Merged in fix/SW-2827-availability-for-old-dates (pull request #2158)
fix: add error handling for hotel room availability * fix: add error handling for hotel room availability * fix: update error codes * fix: have one error message to rule them all. them as in permutations of invalid dates Approved-by: Linus Flood
This commit is contained in:
@@ -0,0 +1,6 @@
|
|||||||
|
.errorContainer {
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: var(--Spacing-x-one-and-half) 0;
|
||||||
|
width: 100%;
|
||||||
|
max-width: var(--max-width-page);
|
||||||
|
}
|
||||||
@@ -1,7 +1,10 @@
|
|||||||
"use client"
|
"use client"
|
||||||
|
|
||||||
|
import { useIntl } from "react-intl"
|
||||||
|
|
||||||
import { trpc } from "@/lib/trpc/client"
|
import { trpc } from "@/lib/trpc/client"
|
||||||
|
|
||||||
|
import Alert from "@/components/TempDesignSystem/Alert"
|
||||||
import useLang from "@/hooks/useLang"
|
import useLang from "@/hooks/useLang"
|
||||||
import RatesProvider from "@/providers/RatesProvider"
|
import RatesProvider from "@/providers/RatesProvider"
|
||||||
|
|
||||||
@@ -9,7 +12,10 @@ import RateSummary from "./RateSummary"
|
|||||||
import Rooms from "./Rooms"
|
import Rooms from "./Rooms"
|
||||||
import { RoomsContainerSkeleton } from "./RoomsContainerSkeleton"
|
import { RoomsContainerSkeleton } from "./RoomsContainerSkeleton"
|
||||||
|
|
||||||
|
import styles from "./index.module.css"
|
||||||
|
|
||||||
import type { RoomsContainerProps } from "@/types/components/hotelReservation/selectRate/roomsContainer"
|
import type { RoomsContainerProps } from "@/types/components/hotelReservation/selectRate/roomsContainer"
|
||||||
|
import { AlertTypeEnum } from "@/types/enums/alert"
|
||||||
|
|
||||||
export function RoomsContainer({
|
export function RoomsContainer({
|
||||||
booking,
|
booking,
|
||||||
@@ -18,25 +24,45 @@ export function RoomsContainer({
|
|||||||
vat,
|
vat,
|
||||||
}: RoomsContainerProps) {
|
}: RoomsContainerProps) {
|
||||||
const lang = useLang()
|
const lang = useLang()
|
||||||
|
const intl = useIntl()
|
||||||
|
|
||||||
const roomsAvailability = trpc.hotel.availability.selectRate.rooms.useQuery({
|
const { data, isFetching, isError, error } =
|
||||||
booking,
|
trpc.hotel.availability.selectRate.rooms.useQuery(
|
||||||
lang,
|
{
|
||||||
})
|
booking,
|
||||||
|
lang,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
retry(failureCount, error) {
|
||||||
|
if (error.data?.code === "BAD_REQUEST") {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
if (
|
return failureCount <= 3
|
||||||
(roomsAvailability.isFetching || !roomsAvailability.data) &&
|
},
|
||||||
!roomsAvailability.isFetched
|
}
|
||||||
) {
|
)
|
||||||
|
|
||||||
|
if (isFetching) {
|
||||||
return <RoomsContainerSkeleton />
|
return <RoomsContainerSkeleton />
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isError) {
|
||||||
|
const errorMessage = getErrorMessage(error.data?.zodError?.formErrors, intl)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles.errorContainer}>
|
||||||
|
<Alert type={AlertTypeEnum.Alarm} heading={errorMessage} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<RatesProvider
|
<RatesProvider
|
||||||
booking={booking}
|
booking={booking}
|
||||||
hotelType={hotelType}
|
hotelType={hotelType}
|
||||||
roomCategories={roomCategories}
|
roomCategories={roomCategories}
|
||||||
roomsAvailability={roomsAvailability.data}
|
roomsAvailability={data}
|
||||||
vat={vat}
|
vat={vat}
|
||||||
>
|
>
|
||||||
<Rooms />
|
<Rooms />
|
||||||
@@ -44,3 +70,25 @@ export function RoomsContainer({
|
|||||||
</RatesProvider>
|
</RatesProvider>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getErrorMessage(
|
||||||
|
formErrors: string[] | undefined,
|
||||||
|
intl: ReturnType<typeof useIntl>
|
||||||
|
) {
|
||||||
|
const firstError = formErrors?.at(0)
|
||||||
|
|
||||||
|
switch (firstError) {
|
||||||
|
case "FROMDATE_INVALID":
|
||||||
|
case "TODATE_INVALID":
|
||||||
|
case "TODATE_MUST_BE_AFTER_FROMDATE":
|
||||||
|
case "FROMDATE_CANNOT_BE_IN_THE_PAST": {
|
||||||
|
return intl.formatMessage({
|
||||||
|
defaultMessage: "Invalid dates",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return intl.formatMessage({
|
||||||
|
defaultMessage: "Something went wrong",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import dayjs from "dayjs"
|
||||||
import { z } from "zod"
|
import { z } from "zod"
|
||||||
|
|
||||||
import { Lang } from "@/constants/languages"
|
import { Lang } from "@/constants/languages"
|
||||||
@@ -58,17 +59,60 @@ const baseBookingSchema = z.object({
|
|||||||
toDate: z.string(),
|
toDate: z.string(),
|
||||||
})
|
})
|
||||||
|
|
||||||
export const selectRateRoomsAvailabilityInputSchema = z.object({
|
export const selectRateRoomsAvailabilityInputSchema = z
|
||||||
booking: baseBookingSchema.extend({
|
.object({
|
||||||
rooms: z.array(
|
booking: baseBookingSchema.extend({
|
||||||
baseRoomSchema.extend({
|
rooms: z.array(
|
||||||
rateCode: z.string().optional(),
|
baseRoomSchema.extend({
|
||||||
roomTypeCode: z.string().optional(),
|
rateCode: z.string().optional(),
|
||||||
})
|
roomTypeCode: z.string().optional(),
|
||||||
),
|
})
|
||||||
}),
|
),
|
||||||
lang: z.nativeEnum(Lang),
|
}),
|
||||||
})
|
lang: z.nativeEnum(Lang),
|
||||||
|
})
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
const fromDate = dayjs(data.booking.fromDate)
|
||||||
|
|
||||||
|
return fromDate.isValid()
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "FROMDATE_INVALID",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
const toDate = dayjs(data.booking.toDate)
|
||||||
|
|
||||||
|
return toDate.isValid()
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "TODATE_INVALID",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
const fromDate = dayjs(data.booking.fromDate).startOf("day")
|
||||||
|
const toDate = dayjs(data.booking.toDate).startOf("day")
|
||||||
|
|
||||||
|
return fromDate.isBefore(toDate)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "FROMDATE_BEFORE_TODATE",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
const fromDate = dayjs(data.booking.fromDate)
|
||||||
|
const today = dayjs().startOf("day")
|
||||||
|
|
||||||
|
return fromDate.isSameOrAfter(today)
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "FROMDATE_CANNOT_BE_IN_THE_PAST",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
export const selectRateRoomAvailabilityInputSchema = z.object({
|
export const selectRateRoomAvailabilityInputSchema = z.object({
|
||||||
booking: baseBookingSchema.extend({
|
booking: baseBookingSchema.extend({
|
||||||
|
|||||||
Reference in New Issue
Block a user