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:
76
packages/trpc/lib/services/booking/cancelBooking/index.ts
Normal file
76
packages/trpc/lib/services/booking/cancelBooking/index.ts
Normal 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
|
||||
}
|
||||
56
packages/trpc/lib/services/booking/cancelBooking/schema.ts
Normal file
56
packages/trpc/lib/services/booking/cancelBooking/schema.ts
Normal 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,
|
||||
}
|
||||
})
|
||||
Reference in New Issue
Block a user