Feat/SW-3549 pass scandic token * WIP pass scandic token * pass scandic token when booking * Merge branch 'master' of bitbucket.org:scandic-swap/web into feat/SW-3549-pass-scandic-token * pass user token when doing availability search * undo changes * merge * Merged in fix/sw-3551-rsc-bookingflowconfig (pull request #2988) fix(SW-3551): Fix issue with BookingConfigProvider in RSC * wip move config to pages * Move config providing to pages * Merged in fix/update-promo-error-modal-text (pull request #2990) fix: update promo error modal text * fix: update promo error modal text Approved-by: Emma Zettervall * Merged in fix/sw-3514-missing-membership-input-for-multiroom (pull request #2991) fix(SW-3514): Show join Scandic Friends card for SAS multiroom * Show join card for room 2+ Approved-by: Hrishikesh Vaipurkar * Merged in feat/lokalise-rebuild (pull request #2993) Feat/lokalise rebuild * chore(lokalise): update translation ids * chore(lokalise): easier to switch between projects * chore(lokalise): update translation ids * . * . * . * . * . * . * chore(lokalise): update translation ids * chore(lokalise): update translation ids * . * . * . * chore(lokalise): update translation ids * chore(lokalise): update translation ids * . * . * chore(lokalise): update translation ids * chore(lokalise): update translation ids * chore(lokalise): new translations * merge * switch to errors for missing id's * merge * sync translations Approved-by: Linus Flood * Merged in feat/SW-3552-logout-from-social-session-when- (pull request #2994) feat(SW-3552): Removed scandic session on logout Approved-by: Joakim Jäderberg * merge * replace getRedemptionTokenSafely() with context based instead * Refactor user verification and error handling across multiple components; implement safeTry utility for safer async calls * Refactor user verification and error handling across multiple components; implement safeTry utility for safer async calls * merge * Merge branch 'master' of bitbucket.org:scandic-swap/web into feat/SW-3549-pass-scandic-token * add booking scope remove unused getMembershipNumber() Approved-by: Anton Gunnarsson Approved-by: Hrishikesh Vaipurkar
110 lines
3.6 KiB
TypeScript
110 lines
3.6 KiB
TypeScript
import * as Sentry from "@sentry/nextjs"
|
|
import { cookies } from "next/headers"
|
|
import { z } from "zod"
|
|
|
|
import { FriendsMembershipLevels } from "@scandic-hotels/common/constants/membershipLevels"
|
|
import { createLogger } from "@scandic-hotels/common/logger/createLogger"
|
|
import { safeTry } from "@scandic-hotels/common/utils/safeTry"
|
|
|
|
import * as api from "../../../api"
|
|
import { protectedProcedure } from "../../../procedures"
|
|
import { getUserSchema } from "../../user/output"
|
|
import { getVerifiedUser } from "../../user/utils/getVerifiedUser"
|
|
|
|
import type { FriendsTier } from "../../../types/user"
|
|
|
|
const matchedSchema = z.object({
|
|
tierMatchState: z.enum(["matched"]),
|
|
toLevel: z.enum(["L1", "L2", "L3", "L4", "L5", "L6", "L7"]),
|
|
})
|
|
const notMatchedSchema = z.object({
|
|
tierMatchState: z.enum(["alreadyMatched", "notLinked", "error", "cached"]),
|
|
})
|
|
|
|
const outputSchema = z.union([matchedSchema, notMatchedSchema])
|
|
const sasLogger = createLogger("SAS")
|
|
export const performLevelUpgrade = protectedProcedure
|
|
.output(outputSchema)
|
|
.mutation(async function ({ ctx }) {
|
|
const cookieStore = await cookies()
|
|
const sasTierMatch = cookieStore.get("sasTierMatch")
|
|
if (sasTierMatch) {
|
|
return { tierMatchState: "cached" }
|
|
}
|
|
|
|
const [profile, error] = await safeTry(
|
|
getVerifiedUser({ token: ctx.session.token })
|
|
)
|
|
if (!profile?.membership || error) {
|
|
return { tierMatchState: "error" }
|
|
}
|
|
const currentLevel = profile.membership.membershipLevel
|
|
|
|
sasLogger.debug("tier match started")
|
|
|
|
const apiResponse = await api.post(api.endpoints.v1.Profile.matchTier, {
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
body: {
|
|
partner: "sas_eb",
|
|
partnerSpecific: {},
|
|
},
|
|
})
|
|
|
|
cookieStore.set("sasTierMatch", "true", {
|
|
maxAge: 60 * 60 * 24,
|
|
httpOnly: true,
|
|
})
|
|
|
|
const updated = apiResponse.status === 200
|
|
if (updated) {
|
|
sasLogger.debug("tier match complete - boosted")
|
|
const result = await apiResponse.json()
|
|
const user = getUserSchema.parse(result)
|
|
|
|
if (!user.membership) {
|
|
const tierMatchErrorNoMembershipMessage =
|
|
"tier match error - no membership"
|
|
sasLogger.debug(tierMatchErrorNoMembershipMessage)
|
|
Sentry.captureException(new Error(tierMatchErrorNoMembershipMessage))
|
|
return { tierMatchState: "error" }
|
|
}
|
|
|
|
const newLevel = user.membership.membershipLevel
|
|
|
|
if (isHigherLevel(newLevel, currentLevel)) {
|
|
return { tierMatchState: "matched", toLevel: newLevel }
|
|
}
|
|
return { tierMatchState: "alreadyMatched" }
|
|
}
|
|
|
|
const matchedNoChange = apiResponse.status === 204
|
|
if (matchedNoChange) {
|
|
sasLogger.debug("tier match complete - no change")
|
|
return { tierMatchState: "alreadyMatched" }
|
|
}
|
|
|
|
const notLinked = apiResponse.status === 404
|
|
if (notLinked) {
|
|
const tierMatchErrorNotLinkedMessage = "tier match error - not linked"
|
|
sasLogger.error(tierMatchErrorNotLinkedMessage)
|
|
Sentry.captureMessage(tierMatchErrorNotLinkedMessage)
|
|
return { tierMatchState: "notLinked" }
|
|
}
|
|
|
|
const tierMatchErrorMessage = `tier match error with status code ${apiResponse.status} and response ${await apiResponse.text()}`
|
|
sasLogger.error(tierMatchErrorMessage)
|
|
Sentry.captureException(new Error(tierMatchErrorMessage))
|
|
return { tierMatchState: "error" }
|
|
})
|
|
|
|
function isHigherLevel(
|
|
newLevel: FriendsTier,
|
|
currentLevel: FriendsTier
|
|
): boolean {
|
|
const currentIndex = FriendsMembershipLevels.indexOf(currentLevel)
|
|
const newIndex = FriendsMembershipLevels.indexOf(newLevel)
|
|
return newIndex > currentIndex
|
|
}
|