Merged in chore/refactor-trpc-booking-routes (pull request #3510)

feat(BOOK-750): refactor booking endpoints

* WIP

* wip

* wip

* parse dates in UTC

* wip

* no more errors

* Merge branch 'master' of bitbucket.org:scandic-swap/web into chore/refactor-trpc-booking-routes

* .

* cleanup

* import named z from zod

* fix(BOOK-750): updateBooking api endpoint expects dateOnly, we passed ISO date


Approved-by: Anton Gunnarsson
This commit is contained in:
Joakim Jäderberg
2026-02-02 14:28:14 +00:00
parent 8ac2c4ba22
commit 16cc26632e
44 changed files with 1621 additions and 1041 deletions

View File

@@ -0,0 +1,76 @@
import { createCounter } from "@scandic-hotels/common/telemetry"
import * as api from "../../../api"
import {
badGatewayError,
extractResponseDetails,
serverErrorByStatus,
} from "../../../errors"
import { toApiLang } from "../../../utils"
import { getBooking } from "../getBooking"
import { cancelBookingSchema } from "./schema"
import type { Lang } from "@scandic-hotels/common/constants/language"
import type { CancelBooking } from "./schema"
export async function cancelBooking(
{
confirmationNumber,
language,
}: { confirmationNumber: string; language: Lang },
token: string
): Promise<CancelBooking | null> {
const cancelBookingCounter = createCounter("booking.cancel")
const metricsCancelBooking = cancelBookingCounter.init({
confirmationNumber,
language,
})
metricsCancelBooking.start()
const headers = {
Authorization: `Bearer ${token}`,
}
const booking = await getBooking(
{ confirmationNumber, lang: language },
token
)
if (!booking) {
metricsCancelBooking.noDataError({ confirmationNumber })
return null
}
const { firstName, lastName, email } = booking.guest
const apiResponse = await api.remove(
api.endpoints.v1.Booking.cancel(confirmationNumber),
{
headers,
body: { firstName, lastName, email },
},
{ language: toApiLang(language) }
)
if (!apiResponse.ok) {
await metricsCancelBooking.httpError(apiResponse)
throw serverErrorByStatus(
apiResponse.status,
await extractResponseDetails(apiResponse),
`cancelBooking failed for ${confirmationNumber}`
)
}
const apiJson = await apiResponse.json()
const verifiedData = cancelBookingSchema.safeParse(apiJson)
if (!verifiedData.success) {
metricsCancelBooking.validationError(verifiedData.error)
throw badGatewayError({
message: "Invalid response from cancelBooking",
errorDetails: { validationError: verifiedData.error },
})
}
metricsCancelBooking.success()
return verifiedData.data
}

View File

@@ -0,0 +1,56 @@
import { z } from "zod"
import { bookingReservationStatusSchema } from "../schema/bookingReservationStatusSchema"
export type CancelBooking = z.infer<typeof cancelBookingSchema>
export const cancelBookingSchema = z
.object({
data: z.object({
attributes: z.object({
reservationStatus: bookingReservationStatusSchema,
paymentUrl: z.string().nullable().optional(),
paymentMethod: z.string().nullable().optional(),
rooms: z
.array(
z.object({
confirmationNumber: z.string(),
cancellationNumber: z.string().nullable(),
priceChangedMetadata: z
.object({
roomPrice: z.number(),
totalPrice: z.number(),
})
.nullable()
.optional(),
})
)
.default([]),
errors: z
.array(
z.object({
confirmationNumber: z.string().nullable().optional(),
errorCode: z.string(),
description: z.string().nullable().optional(),
meta: z
.record(z.string(), z.union([z.string(), z.number()]))
.nullable()
.optional(),
})
)
.default([]),
}),
type: z.string(),
id: z.string(),
}),
})
.transform((apiResponse) => {
return {
id: apiResponse.data.id,
type: apiResponse.data.type,
reservationStatus: apiResponse.data.attributes.reservationStatus,
paymentUrl: apiResponse.data.attributes.paymentUrl,
paymentMethod: apiResponse.data.attributes.paymentMethod,
rooms: apiResponse.data.attributes.rooms,
errors: apiResponse.data.attributes.errors,
}
})