Merged develop into feat/hotel-pages-intro-section

This commit is contained in:
Chuma Mcphoy (We Ahead)
2024-07-08 15:55:01 +00:00
14 changed files with 136 additions and 83 deletions

View File

@@ -1,7 +1,6 @@
import { logout } from "@/constants/routes/handleAuth" import { logout } from "@/constants/routes/handleAuth"
import { serverClient } from "@/lib/trpc/server" import { serverClient } from "@/lib/trpc/server"
import { auth } from "@/auth"
import Link from "@/components/TempDesignSystem/Link" import Link from "@/components/TempDesignSystem/Link"
import { getIntl } from "@/i18n" import { getIntl } from "@/i18n"
@@ -23,9 +22,7 @@ export default async function TopMenu({
lang, lang,
}: TopMenuProps) { }: TopMenuProps) {
const { formatMessage } = await getIntl() const { formatMessage } = await getIntl()
const session = await auth() const user = await serverClient().user.name()
const user = session ? await serverClient().user.get() : null
return ( return (
<div className={styles.topMenu}> <div className={styles.topMenu}>
<div className={styles.container}> <div className={styles.container}>
@@ -46,7 +43,7 @@ export default async function TopMenu({
</li> </li>
))} ))}
<li className={styles.sessionContainer}> <li className={styles.sessionContainer}>
{session ? ( {user ? (
<> <>
{user ? ( {user ? (
<Link <Link

View File

@@ -19,14 +19,11 @@ export default async function Header({
}: LangParams & { languageSwitcher: React.ReactNode } & { }: LangParams & { languageSwitcher: React.ReactNode } & {
myPagesMobileDropdown: React.ReactNode myPagesMobileDropdown: React.ReactNode
}) { }) {
const [data, session] = await Promise.all([ const data = await serverClient().contentstack.base.header({
serverClient().contentstack.base.header({ lang,
lang, })
}),
auth(),
])
const user = !!session ? await serverClient().user.get() : null const user = await serverClient().user.name()
if (!data) { if (!data) {
return null return null

View File

@@ -14,20 +14,20 @@
gap: var(--Spacing-x1); gap: var(--Spacing-x1);
} }
.ol:has(li:nth-last-child(n + 4)), .ol:has(li:nth-last-child(n + 5)),
.ul:has(li:nth-last-child(n + 4)) { .ul:has(li:nth-last-child(n + 5)) {
grid-template-columns: 1fr 1fr; grid-template-columns: 1fr 1fr;
grid-auto-flow: column;
} }
.ol > li::marker { .ol > li::marker {
color: var(--Primary-Light-On-Surface-Accent); color: var(--Primary-Light-On-Surface-Accent);
} }
.ul:has(.heart), .li:has(.heart),
.ul:has(.check) { .li:has(.check) {
list-style: none; list-style: none;
} }
.li:has(.heart), .li:has(.heart),
.li:has(.check) { .li:has(.check) {
display: flex; display: flex;
@@ -54,7 +54,7 @@
} }
.container { .container {
display: "grid"; display: grid;
gap: var(--Spacing-x3); gap: var(--Spacing-x3);
max-width: 1197px; max-width: 1197px;
} }

View File

@@ -497,7 +497,10 @@ export const renderOptions: RenderOptions = {
className={styles.ul} className={styles.ul}
style={ style={
numberOfRows numberOfRows
? { gridTemplateRows: `repeat(${numberOfRows}, auto)` } ? {
gridTemplateRows: `repeat(${numberOfRows}, auto)`,
gridAutoFlow: "column",
}
: {} : {}
} }
> >

View File

@@ -124,13 +124,15 @@ function reducer(state: any, action: OverviewTableReducerAction) {
} }
} }
export default function OverviewTable({ user }: OverviewTableProps) { export default function OverviewTable({
activeMembership,
}: OverviewTableProps) {
const intl = useIntl() const intl = useIntl()
const lang = Lang.en const lang = Lang.en
const levelsData = levelsTranslations[lang] const levelsData = levelsTranslations[lang]
const [selectionState, dispatch] = useReducer( const [selectionState, dispatch] = useReducer(
reducer, reducer,
{ user, lang }, { activeMembership, lang },
getInitialState getInitialState
) )
@@ -150,10 +152,6 @@ export default function OverviewTable({ user }: OverviewTableProps) {
value: level.level, value: level.level,
})) }))
const activeMembership = user?.memberships
? getMembership(user.memberships)
: null
let activeMembershipLevel: membershipLevels | null = null let activeMembershipLevel: membershipLevels | null = null
if (activeMembership?.membershipLevel) { if (activeMembership?.membershipLevel) {
activeMembershipLevel = membershipLevels[activeMembership?.membershipLevel] activeMembershipLevel = membershipLevels[activeMembership?.membershipLevel]

View File

@@ -21,9 +21,7 @@ import type {
import { LoyaltyComponentEnum } from "@/types/components/loyalty/enums" import { LoyaltyComponentEnum } from "@/types/components/loyalty/enums"
async function DynamicComponentBlock({ component }: DynamicComponentProps) { async function DynamicComponentBlock({ component }: DynamicComponentProps) {
const session = await auth() const membershipLevel = await serverClient().user.membershipLevel()
const user = session ? await serverClient().user.get() : null
switch (component) { switch (component) {
case LoyaltyComponentEnum.how_it_works: case LoyaltyComponentEnum.how_it_works:
@@ -31,7 +29,7 @@ async function DynamicComponentBlock({ component }: DynamicComponentProps) {
case LoyaltyComponentEnum.loyalty_levels: case LoyaltyComponentEnum.loyalty_levels:
return <LoyaltyLevels /> return <LoyaltyLevels />
case LoyaltyComponentEnum.overview_table: case LoyaltyComponentEnum.overview_table:
return <OverviewTable user={user} /> return <OverviewTable activeMembership={membershipLevel} />
default: default:
return null return null
} }

View File

@@ -1,6 +1,5 @@
import { login } from "@/constants/routes/handleAuth" import { serverClient } from "@/lib/trpc/server"
import { auth } from "@/auth"
import ArrowRight from "@/components/Icons/ArrowRight" import ArrowRight from "@/components/Icons/ArrowRight"
import { ScandicFriends } from "@/components/Levels" import { ScandicFriends } from "@/components/Levels"
import Button from "@/components/TempDesignSystem/Button" import Button from "@/components/TempDesignSystem/Button"
@@ -21,9 +20,10 @@ export default async function JoinLoyaltyContact({
lang, lang,
}: JoinLoyaltyContactProps & LangParams) { }: JoinLoyaltyContactProps & LangParams) {
const { formatMessage } = await getIntl() const { formatMessage } = await getIntl()
const session = await auth() const user = await serverClient().user.name()
if (session) { // Check if we have user, that means we are logged in.
if (user) {
return null return null
} }
return ( return (

View File

@@ -1,12 +1,14 @@
import { auth } from "@/auth" import { serverClient } from "@/lib/trpc/server"
import MyPagesSidebar from "@/components/MyPages/Sidebar" import MyPagesSidebar from "@/components/MyPages/Sidebar"
import { LangParams } from "@/types/params" import { LangParams } from "@/types/params"
export default async function MyPagesNavigation({ lang }: LangParams) { export default async function MyPagesNavigation({ lang }: LangParams) {
const session = await auth() const user = await serverClient().user.name()
if (!session) { // Check if we have user, that means we are logged in.
if (!user) {
return null return null
} }

View File

@@ -10,7 +10,7 @@ import { unauthorizedError } from "./errors/trpc"
typeof auth typeof auth
type CreateContextOptions = { type CreateContextOptions = {
auth: () => Promise<Session> auth: () => Promise<Session | null>
lang: Lang lang: Lang
pathname: string pathname: string
uid?: string | null uid?: string | null
@@ -50,7 +50,7 @@ export function createContext() {
const session = await auth() const session = await auth()
const webToken = webviewTokenCookie?.value const webToken = webviewTokenCookie?.value
if (!session?.token && !webToken) { if (!session?.token && !webToken) {
throw unauthorizedError() return null
} }
return session || ({ token: { access_token: webToken } } as Session) return session || ({ token: { access_token: webToken } } as Session)

View File

@@ -1,9 +1,13 @@
import * as api from "@/lib/api" import * as api from "@/lib/api"
import { protectedProcedure, router } from "@/server/trpc" import {
protectedProcedure,
router,
safeProtectedProcedure,
} from "@/server/trpc"
import { countries } from "@/components/TempDesignSystem/Form/Country/countries" import { countries } from "@/components/TempDesignSystem/Form/Country/countries"
import * as maskValue from "@/utils/maskValue" import * as maskValue from "@/utils/maskValue"
import { getMembershipCards } from "@/utils/user" import { getMembership, getMembershipCards } from "@/utils/user"
import { import {
friendTransactionsInput, friendTransactionsInput,
@@ -19,6 +23,37 @@ import {
} from "./output" } from "./output"
import { benefits, extendedUser, nextLevelPerks } from "./temp" import { benefits, extendedUser, nextLevelPerks } from "./temp"
import type { Session } from "next-auth"
async function getVerifiedUser({ session }: { session: Session }) {
const apiResponse = await api.get(api.endpoints.v1.profile, {
cache: "no-store",
headers: {
Authorization: `Bearer ${session.token.access_token}`,
},
})
if (!apiResponse.ok) {
return null
}
const apiJson = await apiResponse.json()
if (!apiJson.data?.attributes) {
console.error(`User has no data - (user: ${JSON.stringify(session.user)})`)
return null
}
const verifiedData = getUserSchema.safeParse(apiJson.data.attributes)
if (!verifiedData.success) {
console.info(
`Failed to validate User - (User: ${JSON.stringify(session.user)})`
)
console.error(verifiedData.error)
return null
}
return verifiedData
}
function fakingRequest<T>(payload: T): Promise<T> { function fakingRequest<T>(payload: T): Promise<T> {
return new Promise((resolve) => { return new Promise((resolve) => {
setTimeout(() => { setTimeout(() => {
@@ -31,45 +66,9 @@ export const userQueryRouter = router({
get: protectedProcedure get: protectedProcedure
.input(getUserInputSchema) .input(getUserInputSchema)
.query(async function getUser({ ctx, input }) { .query(async function getUser({ ctx, input }) {
const apiResponse = await api.get(api.endpoints.v1.profile, { const verifiedData = await getVerifiedUser({ session: ctx.session })
cache: "no-store",
headers: {
Authorization: `Bearer ${ctx.session.token.access_token}`,
},
})
if (!apiResponse.ok) { if (!verifiedData) {
// switch (apiResponse.status) {
// case 400:
// throw badRequestError()
// case 401:
// throw unauthorizedError()
// case 403:
// throw forbiddenError()
// default:
// throw internalServerError()
// }
console.info(`API Response Failed - Getting User`)
console.info(`User: (${JSON.stringify(ctx.session.user)})`)
console.error(apiResponse)
return null
}
const apiJson = await apiResponse.json()
if (!apiJson.data?.attributes) {
// throw notFound(apiJson)
console.error(
`User has no data - (user: ${JSON.stringify(ctx.session.user)})`
)
return null
}
const verifiedData = getUserSchema.safeParse(apiJson.data.attributes)
if (!verifiedData.success) {
console.info(
`Failed to validate User - (User: ${JSON.stringify(ctx.session.user)})`
)
console.error(verifiedData.error)
return null return null
} }
@@ -119,7 +118,33 @@ export const userQueryRouter = router({
return user return user
}), }),
name: safeProtectedProcedure.query(async function ({ ctx }) {
if (!ctx.session) {
return null
}
const verifiedData = await getVerifiedUser({ session: ctx.session })
if (!verifiedData) {
return null
}
return {
firstName: verifiedData.data.firstName,
lastName: verifiedData.data.lastName,
}
}),
membershipLevel: safeProtectedProcedure.query(async function ({ ctx }) {
if (!ctx.session) {
return null
}
const verifiedData = await getVerifiedUser({ session: ctx.session })
if (!verifiedData) {
return null
}
const membershipLevel = getMembership(verifiedData.data.memberships)
return membershipLevel
}),
benefits: router({ benefits: router({
current: protectedProcedure.query(async function (opts) { current: protectedProcedure.query(async function (opts) {
// TODO: Make request to get user data from Scandic API // TODO: Make request to get user data from Scandic API

View File

@@ -2,10 +2,16 @@ import { initTRPC } from "@trpc/server"
import { env } from "@/env/server" import { env } from "@/env/server"
import { badRequestError, sessionExpiredError } from "./errors/trpc" import {
badRequestError,
sessionExpiredError,
unauthorizedError,
} from "./errors/trpc"
import { transformer } from "./transformer" import { transformer } from "./transformer"
import { langInput } from "./utils" import { langInput } from "./utils"
import type { Session } from "next-auth"
import type { Meta } from "@/types/trpc/meta" import type { Meta } from "@/types/trpc/meta"
import type { Context } from "./context" import type { Context } from "./context"
@@ -57,6 +63,10 @@ export const protectedProcedure = t.procedure.use(async function (opts) {
console.info(`path: ${opts.path} | type: ${opts.type}`) console.info(`path: ${opts.path} | type: ${opts.type}`)
} }
if (!session) {
throw unauthorizedError()
}
if (session?.error === "RefreshAccessTokenError") { if (session?.error === "RefreshAccessTokenError") {
throw sessionExpiredError() throw sessionExpiredError()
} }
@@ -67,3 +77,25 @@ export const protectedProcedure = t.procedure.use(async function (opts) {
}, },
}) })
}) })
export const safeProtectedProcedure = t.procedure.use(async function (opts) {
const authRequired = opts.meta?.authRequired ?? true
let session: Session | null = await opts.ctx.auth()
if (!authRequired && env.NODE_ENV === "development") {
console.info(
`❌❌❌❌ You are opting out of authorization, if its done on purpose maybe you should use the publicProcedure instead. ❌❌❌❌`
)
console.info(`path: ${opts.path} | type: ${opts.type}`)
}
if (!session || session.error === "RefreshAccessTokenError") {
session = null
}
return opts.next({
ctx: {
session,
},
})
})

View File

@@ -16,6 +16,6 @@ export type MainMenuProps = {
languageSwitcher: React.ReactNode | null languageSwitcher: React.ReactNode | null
myPagesMobileDropdown: React.ReactNode | null myPagesMobileDropdown: React.ReactNode | null
bookingHref: string bookingHref: string
user: User | null user: Pick<User, "firstName" | "lastName"> | null
lang: Lang lang: Lang
} }

View File

@@ -7,9 +7,9 @@ import {
RteBlockContent, RteBlockContent,
} from "@/server/routers/contentstack/loyaltyPage/output" } from "@/server/routers/contentstack/loyaltyPage/output"
import type { IntlFormatters } from "@formatjs/intl" import { MembershipLevel } from "@/utils/user"
import { User } from "@/types/user" import type { IntlFormatters } from "@formatjs/intl"
export type BlocksProps = { export type BlocksProps = {
blocks: Block[] blocks: Block[]
@@ -32,7 +32,7 @@ export type Content = { content: RteBlockContent["content"]["content"] }
type Benefit = { title: string } type Benefit = { title: string }
export type OverviewTableProps = { user: User | null } export type OverviewTableProps = { activeMembership: MembershipLevel | null }
export type Level = { export type Level = {
level: membershipLevels level: membershipLevels

View File

@@ -15,6 +15,7 @@ export function getMembership(memberships: User["memberships"]) {
membership.membershipType.toLowerCase() === scandicMemberships.guestpr membership.membershipType.toLowerCase() === scandicMemberships.guestpr
) )
} }
export type MembershipLevel = ReturnType<typeof getMembership>
export function getMembershipCards( export function getMembershipCards(
memberships: z.infer<typeof getMembershipCardsSchema> memberships: z.infer<typeof getMembershipCardsSchema>