feat(SW-2863): Move contentstack router to trpc package * Add exports to packages and lint rule to prevent relative imports * Add env to trpc package * Add eslint to trpc package * Apply lint rules * Use direct imports from trpc package * Add lint-staged config to trpc * Move lang enum to common * Restructure trpc package folder structure * WIP first step * update internal imports in trpc * Fix most errors in scandic-web Just 100 left... * Move Props type out of trpc * Fix CategorizedFilters types * Move more schemas in hotel router * Fix deps * fix getNonContentstackUrls * Fix import error * Fix entry error handling * Fix generateMetadata metrics * Fix alertType enum * Fix duplicated types * lint:fix * Merge branch 'master' into feat/sw-2863-move-contentstack-router-to-trpc-package * Fix broken imports * Merge branch 'master' into feat/sw-2863-move-contentstack-router-to-trpc-package Approved-by: Linus Flood
132 lines
3.4 KiB
TypeScript
132 lines
3.4 KiB
TypeScript
import { type RawMetadataSchema } from "@scandic-hotels/trpc/routers/contentstack/metadata/output"
|
|
|
|
import { env } from "@/env/server"
|
|
import { serverClient } from "@/lib/trpc/server"
|
|
|
|
import { getDescription } from "./description"
|
|
import { getImage } from "./image"
|
|
import { getTitle } from "./title"
|
|
|
|
import type { Lang } from "@scandic-hotels/common/constants/language"
|
|
import type { Metadata } from "next"
|
|
import type { AlternateURLs } from "next/dist/lib/metadata/types/alternative-urls-types"
|
|
|
|
import type {
|
|
ContentTypeParams,
|
|
LangParams,
|
|
PageArgs,
|
|
UIDParams,
|
|
} from "@/types/params"
|
|
|
|
export async function generateMetadata({
|
|
searchParams,
|
|
params,
|
|
}: PageArgs<
|
|
LangParams & ContentTypeParams & UIDParams,
|
|
{ subpage?: string; filterFromUrl?: string }
|
|
>) {
|
|
const { lang } = await params
|
|
const { subpage, filterFromUrl, ...otherSearchParams } = await searchParams
|
|
// If there are other (real) search params, we don't want to index the page as this will
|
|
// cause duplicate content issues.
|
|
const noIndexOnSearchParams = !!Object.keys(otherSearchParams).length
|
|
const caller = await serverClient()
|
|
const { rawMetadata, alternates, robots } =
|
|
await caller.contentstack.metadata.get({
|
|
subpage,
|
|
filterFromUrl,
|
|
noIndex: noIndexOnSearchParams,
|
|
})
|
|
|
|
if (!rawMetadata) {
|
|
return {
|
|
robots: {
|
|
index: env.isLangLive(lang),
|
|
follow: env.isLangLive(lang),
|
|
},
|
|
}
|
|
}
|
|
|
|
const metadata = await getTransformedMetadata(rawMetadata, alternates, robots)
|
|
|
|
if (typeof metadata?.robots === "string") {
|
|
return metadata
|
|
}
|
|
|
|
return {
|
|
...metadata,
|
|
robots: {
|
|
...(metadata.robots ?? {}),
|
|
index: isIndexable(metadata.robots?.index, lang, alternates),
|
|
follow: isIndexable(metadata.robots?.follow, lang, alternates),
|
|
},
|
|
}
|
|
}
|
|
|
|
function isIndexable(
|
|
pageIndexableFromSettings: boolean | null | undefined,
|
|
lang: Lang,
|
|
alternates: AlternateURLs | null
|
|
) {
|
|
// This is a special case for whitelisting the scandic friends pages, this can be removed when all pages are live
|
|
const url = getUrl(alternates)
|
|
const firstNonLangSegment = (url ?? "").substring(3)
|
|
if (firstNonLangSegment.startsWith("/scandic-friends")) {
|
|
return true
|
|
}
|
|
|
|
// If we are live we want to index the page, but if the page has been marked as noindex in contentstack we don't
|
|
return (pageIndexableFromSettings ?? true) && env.isLangLive(lang)
|
|
}
|
|
|
|
function getUrl(alternates: AlternateURLs | null): string | null {
|
|
try {
|
|
if (!alternates?.canonical) {
|
|
return null
|
|
}
|
|
|
|
if (typeof alternates.canonical === "string") {
|
|
return alternates.canonical
|
|
}
|
|
|
|
if ("href" in alternates.canonical) {
|
|
return alternates.canonical.href
|
|
}
|
|
|
|
if (typeof alternates.canonical.url === "string") {
|
|
return alternates.canonical.url
|
|
}
|
|
|
|
return alternates.canonical.url.href
|
|
} catch {
|
|
return null
|
|
}
|
|
}
|
|
|
|
async function getTransformedMetadata(
|
|
data: RawMetadataSchema,
|
|
alternates: Metadata["alternates"] | null,
|
|
robots: Metadata["robots"] | null = null
|
|
) {
|
|
const noIndex = !!data.web?.seo_metadata?.noindex
|
|
|
|
const metadata: Metadata = {
|
|
metadataBase: env.PUBLIC_URL ? new URL(env.PUBLIC_URL) : undefined,
|
|
title: await getTitle(data),
|
|
description: await getDescription(data),
|
|
openGraph: {
|
|
images: getImage(data),
|
|
},
|
|
alternates,
|
|
robots,
|
|
}
|
|
|
|
if (noIndex) {
|
|
metadata.robots = {
|
|
index: false,
|
|
follow: false,
|
|
}
|
|
}
|
|
return metadata
|
|
}
|