import { Lang } from "@/constants/languages" import { GetDestinationOverviewPage, GetDestinationOverviewPageRefs, } from "@/lib/graphql/Query/DestinationOverviewPage/DestinationOverviewPage.graphql" import { request } from "@/lib/graphql/request" import { notFound } from "@/server/errors/trpc" import { contentstackExtendedProcedureUID, router, serviceProcedure, } from "@/server/trpc" import { generateRefsResponseTag, generateTag } from "@/utils/generateTag" import { safeTry } from "@/utils/safeTry" import { getCitiesByCountry, getCountries, getHotelIdsByCityId, } from "../../hotels/utils" import { getCityPageUrls } from "../destinationCityPage/utils" import { getCountryPageUrls } from "../destinationCountryPage/utils" import destinationsDataDa from "./destinations-da.json" with { assert: "json" } import destinationsDataDe from "./destinations-de.json" with { assert: "json" } import destinationsDataEn from "./destinations-en.json" with { assert: "json" } import destinationsDataFi from "./destinations-fi.json" with { assert: "json" } import destinationsDataNo from "./destinations-no.json" with { assert: "json" } import destinationsDataSv from "./destinations-sv.json" with { assert: "json" } import { destinationOverviewPageRefsSchema, destinationOverviewPageSchema, } from "./output" import { getDestinationOverviewPageCounter, getDestinationOverviewPageFailCounter, getDestinationOverviewPageRefsCounter, getDestinationOverviewPageRefsFailCounter, getDestinationOverviewPageRefsSuccessCounter, getDestinationOverviewPageSuccessCounter, } from "./telemetry" import type { Cities, DestinationsData, } from "@/types/components/destinationOverviewPage/destinationsList/destinationsData" import { TrackingChannelEnum, type TrackingSDKPageData, } from "@/types/components/tracking" import type { GetDestinationOverviewPageData, GetDestinationOverviewPageRefsSchema, } from "@/types/trpc/routers/contentstack/destinationOverviewPage" export const destinationOverviewPageQueryRouter = router({ get: contentstackExtendedProcedureUID.query(async ({ ctx }) => { const { lang, uid } = ctx getDestinationOverviewPageRefsCounter.add(1, { lang, uid: `${uid}` }) console.info( "contentstack.destinationOverviewPage.refs start", JSON.stringify({ query: { lang, uid }, }) ) const refsResponse = await request( GetDestinationOverviewPageRefs, { locale: lang, uid, }, { key: generateRefsResponseTag(lang, uid), ttl: "max", } ) if (!refsResponse.data) { const notFoundError = notFound(refsResponse) getDestinationOverviewPageRefsFailCounter.add(1, { lang, uid, error_type: "not_found", error: JSON.stringify({ code: notFoundError.code }), }) console.error( "contentstack.destinationOverviewPage.refs not found error", JSON.stringify({ query: { lang, uid }, error: { code: notFoundError.code }, }) ) throw notFoundError } const validatedRefsData = destinationOverviewPageRefsSchema.safeParse( refsResponse.data ) if (!validatedRefsData.success) { getDestinationOverviewPageRefsFailCounter.add(1, { lang, uid, error_type: "validation_error", error: JSON.stringify(validatedRefsData.error), }) console.error( "contentstack.destinationOverviewPage.refs validation error", JSON.stringify({ query: { lang, uid }, error: validatedRefsData.error, }) ) return null } getDestinationOverviewPageRefsSuccessCounter.add(1, { lang, uid: `${uid}` }) console.info( "contentstack.destinationOverviewPage.refs success", JSON.stringify({ query: { lang, uid }, }) ) getDestinationOverviewPageCounter.add(1, { lang, uid: `${uid}` }) console.info( "contentstack.destinationOverviewPage start", JSON.stringify({ query: { lang, uid }, }) ) const response = await request( GetDestinationOverviewPage, { locale: lang, uid, }, { key: generateTag(lang, uid), ttl: "max", } ) if (!response.data) { const notFoundError = notFound(response) getDestinationOverviewPageFailCounter.add(1, { lang, uid: `${uid}`, error_type: "not_found", error: JSON.stringify({ code: notFoundError.code }), }) console.error( "contentstack.destinationOverviewPage not found error", JSON.stringify({ query: { lang, uid }, error: { code: notFoundError.code }, }) ) throw notFoundError } const destinationOverviewPage = destinationOverviewPageSchema.safeParse( response.data ) if (!destinationOverviewPage.success) { getDestinationOverviewPageFailCounter.add(1, { lang, uid: `${uid}`, error_type: "validation_error", error: JSON.stringify(destinationOverviewPage.error), }) console.error( "contentstack.destinationOverviewPage validation error", JSON.stringify({ query: { lang, uid }, error: destinationOverviewPage.error, }) ) return null } getDestinationOverviewPageSuccessCounter.add(1, { lang, uid: `${uid}` }) console.info( "contentstack.destinationOverviewPage success", JSON.stringify({ query: { lang, uid }, }) ) const system = destinationOverviewPage.data.destination_overview_page.system const tracking: TrackingSDKPageData = { pageId: system.uid, domainLanguage: lang, publishDate: system.updated_at, createDate: system.created_at, channel: TrackingChannelEnum.hotels, pageType: "destinationoverviewpage", pageName: "destinations|overview", siteSections: "destinations|overview", siteVersion: "new-web", } return { destinationOverviewPage: destinationOverviewPage.data.destination_overview_page, tracking, } }), destinations: router({ get: serviceProcedure.query(async function ({ ctx, }): Promise { // For go live we are using static data here, as it rarely changes. // This also improves operational reliance as we are not hammering // a lot of endpoints for a lot of data. // Re-implement once we have better API support and established caching // patterns and mechanisms. // NOTE: To update the static data set `useStaticData = false`. // Then go to the "Hotels & Destinations" page and visit every language. // At the time of commit http://localhost:3000/en/destinations. // This will update the JSON file locally, each page load for each language, // if all data loads correctly. // Set back `useStaticData = true` again and test with the updated JSON file. // Add, commit and push the updated JSON files with useStaticData = true here. const useStaticData = true if (useStaticData) { switch (ctx.lang) { case Lang.da: return destinationsDataDa case Lang.de: return destinationsDataDe case Lang.fi: return destinationsDataFi case Lang.en: return destinationsDataEn case Lang.no: return destinationsDataNo case Lang.sv: return destinationsDataSv default: return [] } } else { return await updateJSONOnDisk() } async function updateJSONOnDisk() { const { lang } = ctx const countries = await getCountries({ lang, serviceToken: ctx.serviceToken, }) if (!countries) { return [] } const countryNames = countries.data.map((country) => country.name) const citiesByCountry = await getCitiesByCountry({ lang, countries: countryNames, serviceToken: ctx.serviceToken, onlyPublished: true, }) const cityPages = await getCityPageUrls(lang) const destinations = await Promise.all( Object.entries(citiesByCountry).map(async ([country, cities]) => { const activeCitiesWithHotelCount: Cities = await Promise.all( cities.map(async (city) => { const [hotels] = await safeTry( getHotelIdsByCityId({ cityId: city.id, serviceToken: ctx.serviceToken, }) ) const cityPage = cityPages.find( (cityPage) => cityPage.city === city.cityIdentifier ) return { id: city.id, name: city.name, hotelIds: hotels || [], hotelCount: hotels ? hotels.length : 0, url: cityPage?.url, } }) ) const countryPages = await getCountryPageUrls(lang) const countryPage = countryPages.find( (countryPage) => countryPage.country === country ) return { country, countryUrl: countryPage?.url, numberOfHotels: activeCitiesWithHotelCount.reduce( (acc, city) => acc + city.hotelCount, 0 ), cities: activeCitiesWithHotelCount, } }) ) const data = destinations.sort((a, b) => a.country.localeCompare(b.country) ) const fs = await import("node:fs") fs.writeFileSync( `./server/routers/contentstack/destinationOverviewPage/destinations-${lang}.json`, JSON.stringify(data), { encoding: "utf-8", } ) return data } }), }), })