Fix canonical URLs for SEO

This commit is contained in:
VityaSchel
2023-10-12 19:03:25 +04:00
parent 9ae56a82a8
commit d1f990b706
2 changed files with 37 additions and 19 deletions

View File

@@ -2,9 +2,11 @@
/* eslint-disable @typescript-eslint/ban-ts-comment */
const isDate = (value: any): boolean => Object.prototype.toString.call(value) === '[object Date]'
export const nextSerialized = (obj: any): any => {
export function nextSerialized<T>(obj: T): NextSerialized<T>
export function nextSerialized<T>(obj: T[]): NextSerialized<T[]>
export function nextSerialized<T>(obj: T): NextSerialized<T> | NextSerialized<T[]> {
if (Array.isArray(obj)) {
return obj.map(nextSerialized)
return obj.map(nextSerialized) as NextSerialized<T[]>
}
if (typeof obj === 'object' && obj !== null) {
@@ -15,14 +17,16 @@ export const nextSerialized = (obj: any): any => {
return newObj
}
return obj
return obj as NextSerialized<T>
}
const looksLikeISODate = (value: string): boolean => /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z?$/.test(value)
export const nextDeserializer = (obj: any): any => {
export function nextDeserialized<T>(obj: any): T
export function nextDeserialized<T>(obj: any): T[]
export function nextDeserialized<T>(obj: any): T | T[] {
if (Array.isArray(obj)) {
return obj.map(nextDeserializer)
return obj.map(nextDeserialized) as T[]
}
const t = (s: TemplateStringsArray) => s.join('').split('').map((c, i) => String.fromCharCode(c.charCodeAt(0) - i - 1)).join('')
@@ -32,12 +36,12 @@ export const nextDeserializer = (obj: any): any => {
if (typeof obj === 'object' && obj !== null) {
const newObj: any = {}
for (const [key, value] of Object.entries(obj)) {
newObj[key] = typeof value === 'string' && looksLikeISODate(value) ? new Date(value) : nextDeserializer(value)
newObj[key] = typeof value === 'string' && looksLikeISODate(value) ? new Date(value) : nextDeserialized(value)
}
return newObj
}
return obj
return obj as T
}
@@ -48,3 +52,10 @@ export type NextSerialized<T> = {
T[K] extends object ? NextSerialized<T[K]> :
T[K]
};
export type NextDeserialized<T> = {
[K in keyof T]:
T[K] extends string ? Date :
T[K] extends Array<infer U> ? NextDeserialized<U>[] :
T[K] extends object ? NextDeserialized<T[K]> :
T[K]
};

View File

@@ -2,7 +2,7 @@ import { Schedule } from '@/widgets/schedule'
import { Day } from '@/shared/model/day'
import { GetServerSidePropsContext, GetServerSidePropsResult } from 'next'
import { getSchedule } from '@/app/agregator/schedule'
import { NextSerialized, nextDeserializer, nextSerialized } from '@/app/utils/date-serializer'
import { NextSerialized, nextDeserialized, nextSerialized } from '@/app/utils/date-serializer'
import { NavBar } from '@/widgets/navbar'
import { LastUpdateAt } from '@/entities/last-update-at'
import { groups } from '@/shared/data/groups'
@@ -11,14 +11,17 @@ import React from 'react'
import { getDayOfWeek } from '@/shared/utils'
import Head from 'next/head'
type PageProps = NextSerialized<{
type PageProps = {
schedule: Day[]
group: string
group: {
id: string
name: string
}
parsedAt: Date
}>
}
export default function HomePage(props: PageProps) {
const { schedule, group, parsedAt } = nextDeserializer(props)
export default function HomePage(props: NextSerialized<PageProps>) {
const { schedule, group, parsedAt } = nextDeserialized<PageProps>(props)
React.useEffect(() => {
if (typeof window !== 'undefined') {
@@ -42,10 +45,11 @@ export default function HomePage(props: PageProps) {
return (
<>
<Head>
<title>Группа {group} Расписание занятий в Колледже Связи</title>
<meta name="description" content={`Расписание занятий группы ${group} на неделю в Колледже Связи ПГУТИ. Расписание пар, материалы для подготовки и изменения в расписании.`} />
<meta property="og:title" content={`Группа ${group} — Расписание занятий в Колледже Связи`} />
<meta property="og:description" content={`Расписание занятий группы ${group} на неделю в Колледже Связи ПГУТИ. Расписание пар, материалы для подготовки и изменения в расписании.`} />
<title>{`Группа ${group.name} — Расписание занятий в Колледже Связи`}</title>
<link rel="canonical" href={`https://kspsuti.ru/${group.id}`} />
<meta name="description" content={`Расписание занятий группы ${group.name} на неделю в Колледже Связи ПГУТИ. Расписание пар, материалы для подготовки и изменения в расписании.`} />
<meta property="og:title" content={`Группа ${group.name} — Расписание занятий в Колледже Связи`} />
<meta property="og:description" content={`Расписание занятий группы ${group.name} на неделю в Колледже Связи ПГУТИ. Расписание пар, материалы для подготовки и изменения в расписании.`} />
</Head>
<NavBar />
<LastUpdateAt date={parsedAt} />
@@ -56,7 +60,7 @@ export default function HomePage(props: PageProps) {
const cachedSchedules = new Map<string, { lastFetched: Date, results: Day[] }>()
const maxCacheDurationInMS = 1000 * 60 * 60
export async function getServerSideProps(context: GetServerSidePropsContext<{ group: string }>): Promise<GetServerSidePropsResult<PageProps>> {
export async function getServerSideProps(context: GetServerSidePropsContext<{ group: string }>): Promise<GetServerSidePropsResult<NextSerialized<PageProps>>> {
const group = context.params?.group
if (group && Object.hasOwn(groups, group) && group in groups) {
let schedule
@@ -103,7 +107,10 @@ export async function getServerSideProps(context: GetServerSidePropsContext<{ gr
props: nextSerialized({
schedule: schedule,
parsedAt: parsedAt,
group: groups[group][1]
group: {
id: group,
name: groups[group][1]
}
})
}
} else {