feat: enhance stays with api data

This commit is contained in:
Christel Westerberg
2024-05-08 14:56:39 +02:00
parent 57fd59ebe2
commit 661a8019d3
18 changed files with 223 additions and 108 deletions

View File

@@ -4,7 +4,7 @@ import Title from "@/components/Title"
import styles from "./header.module.css"
import type { HeaderProps } from "@/types/components/myPages/myStays/title"
import type { HeaderProps } from "@/types/components/myPages/stays/title"
export default function Header({ title, subtitle, link }: HeaderProps) {
return (

View File

@@ -10,7 +10,7 @@ import StayList from "../StayList"
import EmptyPreviousStaysBlock from "./EmptyPreviousStays"
import { AccountPageComponentProps } from "@/types/components/myPages/myPage/accountPage"
import type { Page } from "@/types/components/myPages/myStays/page"
import type { Page } from "@/types/components/myPages/stays/page"
export default function PreviousStays({
lang,

View File

@@ -2,7 +2,7 @@ import Button from "@/components/TempDesignSystem/Button"
import styles from "./button.module.css"
import type { ShowMoreButtonParams } from "@/types/components/myPages/myStays/button"
import type { ShowMoreButtonParams } from "@/types/components/myPages/stays/button"
export default function ShowMoreButton({
disabled,

View File

@@ -16,7 +16,7 @@ export default async function SoonestStays({
subtitle,
link,
}: AccountPageComponentProps) {
const stays = await serverClient().user.stays.soonestUpcoming()
const { data: stays } = await serverClient().user.stays.upcoming({ limit: 3 })
return (
<MaxWidth className={styles.container} tag="section">

View File

@@ -5,15 +5,15 @@ import Title from "@/components/Title"
import styles from "./stay.module.css"
import type { StayCardProps } from "@/types/components/myPages/myStays/stayCard"
import type { StayCardProps } from "@/types/components/myPages/stays/stayCard"
export default function StayCard({ stay, lang }: StayCardProps) {
const { dateArrive, dateDepart, guests, hotel } = stay
const { checkinDate, checkoutDate, hotelInformation } = stay.attributes
const arrival = dt(dateArrive).locale(lang)
const arrival = dt(checkinDate).locale(lang)
const arrivalDate = arrival.format("DD MMM")
const arrivalDateTime = arrival.format("YYYY-MM-DD")
const depart = dt(dateDepart).locale(lang)
const depart = dt(checkoutDate).locale(lang)
const departDate = depart.format("DD MMM YYYY")
const departDateTime = depart.format("YYYY-MM-DD")
@@ -27,7 +27,7 @@ export default function StayCard({ stay, lang }: StayCardProps) {
uppercase
className={styles.hotel}
>
{hotel}
{hotelInformation.hotelName}
</Title>
<section className={styles.container}>
<div className={styles.date}>
@@ -43,17 +43,6 @@ export default function StayCard({ stay, lang }: StayCardProps) {
<time dateTime={departDateTime}>{departDate}</time>
</p>
</div>
<div className={styles.guests}>
<Image
alt="Guests Icon"
height={20}
src="/_static/icons/person.svg"
width={20}
/>
<span>
{guests} guest{guests > 1 ? "s" : ""}
</span>
</div>
</section>
</footer>
</article>

View File

@@ -2,13 +2,17 @@ import StayCard from "../StayCard"
import styles from "./stayList.module.css"
import { StayListProps } from "@/types/components/myPages/myStays/stayList"
import { StayListProps } from "@/types/components/myPages/stays/stayList"
export default function StayList({ lang, stays }: StayListProps) {
return (
<section className={styles.stays}>
{stays.map((stay) => (
<StayCard key={stay.uid} stay={stay} lang={lang} />
<StayCard
key={stay.attributes.confirmationNumber}
stay={stay}
lang={lang}
/>
))}
</section>
)

View File

@@ -1,6 +1,5 @@
"use client"
import { Lang } from "@/constants/languages"
import { _ } from "@/lib/translation"
import { trpc } from "@/lib/trpc/client"
@@ -12,7 +11,7 @@ import StayList from "../StayList"
import EmptyUpcomingStaysBlock from "./EmptyUpcomingStays"
import { AccountPageComponentProps } from "@/types/components/myPages/myPage/accountPage"
import type { Page } from "@/types/components/myPages/myStays/page"
import type { Page } from "@/types/components/myPages/stays/page"
export default function UpcomingStays({
lang,
@@ -22,14 +21,16 @@ export default function UpcomingStays({
}: AccountPageComponentProps) {
const { data, hasNextPage, isFetching, fetchNextPage } =
trpc.user.stays.upcoming.useInfiniteQuery(
{},
{ limit: 6 },
{
getNextPageParam: (lastPage: Page) => lastPage.nextCursor,
}
)
function loadMoreData() {
fetchNextPage()
if (hasNextPage) {
fetchNextPage()
}
}
return (

View File

@@ -5,6 +5,10 @@ export namespace endpoints {
export const enum v0 {
profile = "profile/v0/Profile",
}
export const enum v1 {
upcomingStays = "booking/v1/Stays/future",
previousStays = "booking/v1/Stays/past",
}
}
export type Endpoint = endpoints.v0
export type Endpoint = endpoints.v0 | endpoints.v1

View File

@@ -20,10 +20,14 @@ const defaultOptions: RequestInit = {
export async function get(
endpoint: Endpoint,
options: RequestOptionsWithOutBody
options: RequestOptionsWithOutBody,
params?: URLSearchParams
) {
defaultOptions.method = "GET"
return fetch(`${env.API_BASEURL}/${endpoint}`, merge(defaultOptions, options))
const url = new URL(
`${env.API_BASEURL}/${endpoint}${params ? `?${params.toString()}` : ""}`
)
return fetch(url, merge(defaultOptions, options))
}
export async function patch(

View File

@@ -2,15 +2,13 @@ import { z } from "zod"
export const staysInput = z
.object({
perPage: z.number().min(0).default(6),
page: z.number().min(0).default(0),
limit: z.number().min(0).default(6),
cursor: z.number().nullish(),
})
.default({})
export const soonestUpcomingStaysInput = z
.object({
limit: z.number().int().positive(),
})
.default({ limit: 3 })
export const soonestUpcomingStaysInput = z
.object({
limit: z.number().int().positive(),
})
.default({ limit: 3 })

View File

@@ -23,3 +23,69 @@ export const getUserSchema = z.object({
phoneNumber: z.string(),
profileId: z.string(),
})
// Schema is the same for upcoming and previous stays endpoints
export const getStaysSchema = z.object({
data: z.array(
z.object({
attributes: z.object({
hotelOperaId: z.string(),
hotelInformation: z.object({
hotelContent: z.object({
images: z.object({
metaData: z.object({
title: z.string(),
altText: z.string(),
altText_En: z.string(),
copyRight: z.string(),
}),
imageSizes: z.object({
tiny: z.string(),
small: z.string(),
medium: z.string(),
large: z.string(),
}),
}),
}),
hotelName: z.string(),
cityName: z.string(),
}),
confirmationNumber: z.string(),
checkinDate: z.string(),
checkoutDate: z.string(),
isWebAppOrigin: z.boolean(),
}),
relationships: z.object({
hotel: z.object({
links: z.object({
related: z.string(),
}),
data: z.object({
id: z.string(),
type: z.string(),
}),
}),
}),
type: z.string(),
id: z.string(),
links: z.object({
self: z.object({
href: z.string(),
meta: z.object({
method: z.string(),
}),
}),
}),
})
),
links: z.object({
self: z.string(),
offset: z.number(),
limit: z.number(),
totalCount: z.number(),
}),
})
type GetStaysData = z.infer<typeof getStaysSchema>
export type Stay = GetStaysData["data"][number]

View File

@@ -9,15 +9,9 @@ import {
} from "@/server/errors/trpc"
import { protectedProcedure, router } from "@/server/trpc"
import { soonestUpcomingStaysInput, staysInput } from "./input"
import { getUserSchema } from "./output"
import {
benefits,
extendedUser,
nextLevelPerks,
previousStays,
upcomingStays,
} from "./temp"
import { staysInput } from "./input"
import { getStaysSchema, getUserSchema } from "./output"
import { benefits, extendedUser, nextLevelPerks } from "./temp"
function fakingRequest<T>(payload: T): Promise<T> {
return new Promise((resolve) => {
@@ -84,66 +78,130 @@ export const userQueryRouter = router({
}),
stays: router({
soonestUpcoming: protectedProcedure
.input(soonestUpcomingStaysInput)
.query(async ({ input }) => {
return upcomingStays.slice(0, input.limit)
}),
previous: protectedProcedure.input(staysInput).query(async (opts) => {
const { perPage, page, cursor } = opts.input
let nextCursor: typeof cursor | undefined = undefined
const nrPages = Math.ceil(previousStays.length / perPage)
try {
const { limit: perPage, cursor } = opts.input
let stays, nextPage
if (cursor) {
stays = previousStays.slice(cursor, perPage + cursor + 1)
nextPage = cursor / perPage + 1
} else {
stays = previousStays.slice(
page * perPage,
page * perPage + perPage + 1
)
}
const params = new URLSearchParams()
params.set("limit", perPage.toString())
if (
(nextPage && nextPage < nrPages && stays.length == perPage + 1) ||
(!nextPage && nrPages > 1)
) {
const nextItem = stays.pop()
if (nextItem) {
nextCursor = previousStays.indexOf(nextItem)
if (cursor) {
params.set("offset", cursor.toString())
}
} // TODO: Make request to get user data from Scandic API
return { data: stays, nextCursor }
const apiResponse = await api.get(
api.endpoints.v1.previousStays,
{
headers: {
Authorization: `Bearer ${opts.ctx.session.token.access_token}`,
},
},
params
)
if (!apiResponse.ok) {
switch (apiResponse.status) {
case 400:
throw badRequestError()
case 401:
throw unauthorizedError()
case 403:
throw forbiddenError()
default:
throw internalServerError()
}
}
const apiJson = await apiResponse.json()
if (!apiJson.data?.length) {
throw internalServerError()
}
const verifiedData = getStaysSchema.safeParse(apiJson)
if (!verifiedData.success) {
console.info(`Get Previous Stays - Verified Data Error`)
console.error(verifiedData.error)
throw badRequestError()
}
const nextCursor =
verifiedData.data.links.offset < verifiedData.data.links.totalCount
? verifiedData.data.links.offset
: undefined
return {
data: verifiedData.data.data,
nextCursor,
}
} catch (error) {
console.info(`Get Previous Stays Error`)
console.error(error)
throw internalServerError()
}
}),
upcoming: protectedProcedure.input(staysInput).query(async (opts) => {
const { perPage, page, cursor } = opts.input
let nextCursor: typeof cursor | undefined = undefined
const nrPages = Math.ceil(upcomingStays.length / perPage)
try {
const { limit: perPage, cursor } = opts.input
let stays, nextPage
if (cursor) {
stays = upcomingStays.slice(cursor, perPage + cursor + 1)
nextPage = cursor / perPage + 1
} else {
stays = upcomingStays.slice(
page * perPage,
page * perPage + perPage + 1
)
}
const params = new URLSearchParams()
params.set("limit", perPage.toString())
if (
(nextPage && nextPage < nrPages && stays.length == perPage + 1) ||
(!nextPage && nrPages > 1)
) {
const nextItem = stays.pop()
if (nextItem) {
nextCursor = upcomingStays.indexOf(nextItem)
if (cursor) {
params.set("offset", cursor.toString())
}
} // TODO: Make request to get user data from Scandic API
return { data: stays, nextCursor }
const apiResponse = await api.get(
api.endpoints.v1.upcomingStays,
{
headers: {
Authorization: `Bearer ${opts.ctx.session.token.access_token}`,
},
},
params
)
if (!apiResponse.ok) {
switch (apiResponse.status) {
case 400:
throw badRequestError()
case 401:
throw unauthorizedError()
case 403:
throw forbiddenError()
default:
throw internalServerError()
}
}
const apiJson = await apiResponse.json()
if (!apiJson.data?.length) {
throw internalServerError()
}
const verifiedData = getStaysSchema.safeParse(apiJson)
if (!verifiedData.success) {
console.info(`Get Upcoming Stays - Verified Data Error`)
console.error(verifiedData.error)
throw badRequestError()
}
const nextCursor =
verifiedData.data.links.offset < verifiedData.data.links.totalCount
? verifiedData.data.links.offset
: undefined
return {
data: verifiedData.data.data,
nextCursor,
}
} catch (error) {
console.info(`Get Upcoming Stays Error`)
console.error(error)
throw internalServerError()
}
}),
}),
})

View File

@@ -1,9 +0,0 @@
import { UUID } from "crypto"
export type Stay = {
uid: UUID
dateArrive: Date
dateDepart: Date
guests: number
hotel: string
}

View File

@@ -1,4 +1,4 @@
import type { Stay } from "../myPage/stay"
import { Stay } from "@/server/routers/user/output"
export type Page = {
data: Stay[]

View File

@@ -1,5 +1,5 @@
import type { Lang } from "@/constants/languages"
import type { Stay } from "../myPage/stay"
import type { Stay } from "@/server/routers/user/output"
export type StayCardProps = {
lang: Lang

View File

@@ -1,5 +1,5 @@
import type { Lang } from "@/constants/languages"
import type { Stay } from "../myPage/stay"
import type { Stay } from "@/server/routers/user/output"
export type StayListProps = {
stays: Stay[]