feat(LOY-268): Feature branch for profiling consent work * feat: Add feature branch for profile and consent work * Merged in feat/LOY-268-profile-consent-banner-comp (pull request #2908) Feat/LOY-358 profile consent banner component * feat: Add feature branch for profile and consent work * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component Approved-by: Chuma Mcphoy (We Ahead) * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * Merged in feat/profile-consent-contentstack (pull request #2921) Feat(LOY-389): Profile consent in Contentstack * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * chore(LOY-348): add profiling consent as CS entry * chore(LOY-348): add banner as dynamic content Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-347-Profile-consent-modal-phase-1 (pull request #2901) Feat(LOY-347): Profiling Consent Modal (phase 1) * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Icons * feat(LOY-347): Implement ability to open modal from banner * fix(LOY-347): minor fixes * fix(LOY-347): replace old spa icon * fix(LOY-347): re-add env vars * fix(LOY-347): File renaming and cleanup * chore(LOY-347): Update readme * fix(LOY-347): use correct space var * fix(LOY-347): Add TODO comment for adding link to accordion Approved-by: Matilda Landström * Merged in fix/LOY-386-profiling-consent-modal-contentstack (pull request #2930) Fix(LOY-386): profiling consent modal contentstack * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Icons * feat(LOY-347): Implement ability to open modal from banner * fix(LOY-347): minor fixes * fix(LOY-347): replace old spa icon * fix(LOY-347): re-add env vars * fix(LOY-347): File renaming and cleanup * fix(LOY-386): Use contentstack content for profile consent modal * fix(LOY-386): beneift cards schema transform * chore(LOY-386): remove usememo * fix(LOY-386): fix modalcontent check * fix(LOY-386): remove uneeded vars Approved-by: Matilda Landström * Merged in feat/LOY-412-profiling-consent-in-signup (pull request #2976) Feat(LOY-412): profiling consent in signup * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * Merged in feat/profile-consent-contentstack (pull request #2921) Feat(LOY-389): Profile consent in Contentstack * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * chore(LOY-348): add profiling consent as CS entry * chore(LOY-348): add banner as dynamic content Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-347-Profile-consent-modal-phase-1 (pull request #2901) Feat(LOY-347): Profiling Consent Modal (phase 1) * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Ico… * Merged in fix/lokalise-ids (pull request #3013) fix: add ids to translations in Profiling Consent * fix: add ids to translations Approved-by: Erik Tiekstra Approved-by: Chuma Mcphoy (We Ahead) * Merged in LOY-436-my-pages-profiling-consent (pull request #3011) LOY-436: Profiling Consent on My Profile, no api Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-418-profiling-consent-ui-text-update (pull request #3080) Feat/LOY-418: Profiling consent ui and text update * chore(LOY-418): update /consent buttons * chore(LOY-418): update legal texts Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-268-profiling-consent-api (pull request #3088) Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-413-Signup-API-Profiling-Consent (pull request #3105) Feat/LOY-413 Signup API Profiling Consent * feat(LOY-413): signup profiling consent * chore(LOY-413): remove todo * fix(LOY-413): only pass in profilingConsent if true * fix(LOY-413): proper spread of profilingConsent in signup input Approved-by: Christel Westerberg * Merged in fix/LOY-413-use-v2-for-signup-call (pull request #3112) fix(LOY-413): use v2 endpoint for profile POST in signup * fix(LOY-413): use v2 endpoint for profile POST in signup Approved-by: Erik Tiekstra * Merged in feat/LOY-268-profiling-consent-improvements (pull request #3094) Feat/LOY-268: Profiling consent improvements * Merged in feat/profile-consent-contentstack (pull request #2921) Feat(LOY-389): Profile consent in Contentstack * feat(LOY-268): create banner * feat(LOY-268): Create personalization banner component * chore(LOY-348): add profiling consent as CS entry * chore(LOY-348): add banner as dynamic content Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-347-Profile-consent-modal-phase-1 (pull request #2901) Feat(LOY-347): Profiling Consent Modal (phase 1) * feat(LOY-347): Profile Consent Modal base functionality * feat(LOY-347): Add Icon * feat(LOY-347): Add Benefit Cards to Profile consent modal * feat(LOY-347): Add accordion to profile consent modal * fix(LOY-347): scroll behaviour * Fix(LOY-347): fade in/out animations of profile consent modal * fix(LOY-347): White Ellipsis Around Icons * feat(LOY-347): Implement ability to open modal from banner * fix(LOY-347): min… * Merged in fix/update-graphql (pull request #3130) fix: update graphql * fix: update graphql Approved-by: Chuma Mcphoy (We Ahead) * Merged in feat/LOY-414-prof-consent-tracking (pull request #3127) Feat/LOY-414 profile consent tracking + credit card ui update * chore(LOY-414): create track link function * chore(LOY-414): add cta tracking * chore(LOY-414): add profileConsent to userInfo datalayer * chore(LOY-414): update credit card ui * chore(LOY-414): update tracking specs * chore(LOY-414): add pageView tracking to modal Approved-by: Chuma Mcphoy (We Ahead) * fix: remove old flag * Merged in fix/LOY-268-prof-consent-button-fix (pull request #3162) fix(LOY-268): add button as link * fix(LOY-268): add button as link Approved-by: Chuma Mcphoy (We Ahead) Approved-by: Matilda Landström
295 lines
8.5 KiB
TypeScript
295 lines
8.5 KiB
TypeScript
import { signupVerify } from "@scandic-hotels/common/constants/routes/signup"
|
|
import { createLogger } from "@scandic-hotels/common/logger/createLogger"
|
|
import { createCounter } from "@scandic-hotels/common/telemetry"
|
|
|
|
import { env } from "../../../env/server"
|
|
import { router } from "../.."
|
|
import * as api from "../../api"
|
|
import { serverErrorByStatus } from "../../errors"
|
|
import { protectedProcedure, serviceProcedure } from "../../procedures"
|
|
import {
|
|
addCreditCardInput,
|
|
addPromoCampaignInput,
|
|
deleteCreditCardInput,
|
|
profilingConsentInput,
|
|
saveCreditCardInput,
|
|
signupInput,
|
|
} from "./input"
|
|
import { initiateSaveCardSchema, subscriberIdSchema } from "./output"
|
|
|
|
const userMutationLogger = createLogger("userMutationRouter")
|
|
export const userMutationRouter = router({
|
|
creditCard: router({
|
|
add: protectedProcedure.input(addCreditCardInput).mutation(async function ({
|
|
ctx,
|
|
input,
|
|
}) {
|
|
userMutationLogger.info(
|
|
"api.user.creditCard.add start",
|
|
JSON.stringify({ query: { language: input.language } })
|
|
)
|
|
const apiResponse = await api.post(
|
|
api.endpoints.v1.Profile.CreditCards.initiateSaveCard,
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
body: {
|
|
language: input.language,
|
|
mobileToken: false,
|
|
redirectUrl: `api/web/add-card-callback/${input.language}`,
|
|
},
|
|
}
|
|
)
|
|
|
|
if (!apiResponse.ok) {
|
|
const text = await apiResponse.text()
|
|
userMutationLogger.error(
|
|
"api.user.creditCard.add error",
|
|
JSON.stringify({
|
|
query: { language: input.language },
|
|
error: {
|
|
status: apiResponse.status,
|
|
statusText: apiResponse.statusText,
|
|
error: text,
|
|
},
|
|
})
|
|
)
|
|
return null
|
|
}
|
|
|
|
const apiJson = await apiResponse.json()
|
|
const verifiedData = initiateSaveCardSchema.safeParse(apiJson)
|
|
if (!verifiedData.success) {
|
|
userMutationLogger.error(
|
|
"api.user.creditCard.add validation error",
|
|
JSON.stringify({
|
|
query: { language: input.language },
|
|
error: verifiedData.error,
|
|
})
|
|
)
|
|
return null
|
|
}
|
|
userMutationLogger.info(
|
|
"api.user.creditCard.add success",
|
|
JSON.stringify({ query: { language: input.language } })
|
|
)
|
|
return verifiedData.data.data
|
|
}),
|
|
save: protectedProcedure
|
|
.input(saveCreditCardInput)
|
|
.mutation(async function ({ ctx, input }) {
|
|
userMutationLogger.info(
|
|
"api.user.creditCard.save start",
|
|
JSON.stringify({})
|
|
)
|
|
const apiResponse = await api.post(
|
|
api.endpoints.v1.Profile.CreditCards.transaction(input.transactionId),
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
}
|
|
)
|
|
|
|
if (!apiResponse.ok) {
|
|
const text = await apiResponse.text()
|
|
userMutationLogger.error(
|
|
"api.user.creditCard.save error",
|
|
JSON.stringify({
|
|
error: {
|
|
status: apiResponse.status,
|
|
statusText: apiResponse.statusText,
|
|
text,
|
|
},
|
|
})
|
|
)
|
|
return false
|
|
}
|
|
userMutationLogger.info(
|
|
"api.user.creditCard.save success",
|
|
JSON.stringify({})
|
|
)
|
|
return true
|
|
}),
|
|
delete: protectedProcedure
|
|
.input(deleteCreditCardInput)
|
|
.mutation(async function ({ ctx, input }) {
|
|
userMutationLogger.info(
|
|
"api.user.creditCard.delete start",
|
|
JSON.stringify({ query: {} })
|
|
)
|
|
const apiResponse = await api.remove(
|
|
api.endpoints.v1.Profile.CreditCards.deleteCreditCard(
|
|
input.creditCardId
|
|
),
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
}
|
|
)
|
|
|
|
if (!apiResponse.ok) {
|
|
const text = await apiResponse.text()
|
|
userMutationLogger.error(
|
|
"api.user.creditCard.delete error",
|
|
JSON.stringify({
|
|
error: {
|
|
status: apiResponse.status,
|
|
statusText: apiResponse.statusText,
|
|
text,
|
|
},
|
|
query: {},
|
|
})
|
|
)
|
|
return false
|
|
}
|
|
userMutationLogger.info(
|
|
"api.user.creditCard.delete success",
|
|
JSON.stringify({})
|
|
)
|
|
return true
|
|
}),
|
|
}),
|
|
generatePreferencesLink: protectedProcedure.mutation(async function ({
|
|
ctx,
|
|
}) {
|
|
const generatePreferencesLinkCounter = createCounter(
|
|
"trpc.user",
|
|
"generatePreferencesLink"
|
|
)
|
|
|
|
const metricsGeneratePreferencesLink = generatePreferencesLinkCounter.init()
|
|
|
|
metricsGeneratePreferencesLink.start()
|
|
|
|
const apiResponse = await api.get(api.endpoints.v1.Profile.subscriberId, {
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
})
|
|
|
|
if (!apiResponse.ok) {
|
|
await metricsGeneratePreferencesLink.httpError(apiResponse)
|
|
return null
|
|
}
|
|
|
|
const data = await apiResponse.json()
|
|
|
|
const validatedData = subscriberIdSchema.safeParse(data)
|
|
|
|
if (!validatedData.success) {
|
|
metricsGeneratePreferencesLink.validationError(validatedData.error)
|
|
return null
|
|
}
|
|
|
|
const preferencesLink = new URL(env.SALESFORCE_PREFERENCE_BASE_URL)
|
|
preferencesLink.searchParams.set("subKey", validatedData.data.subscriberId)
|
|
|
|
metricsGeneratePreferencesLink.success()
|
|
|
|
return preferencesLink.toString()
|
|
}),
|
|
signup: serviceProcedure.input(signupInput).mutation(async function ({
|
|
ctx,
|
|
input,
|
|
}) {
|
|
const signupCounter = createCounter("trpc.user", "signup")
|
|
const metricsSignup = signupCounter.init()
|
|
|
|
const apiResponse = await api.post(api.endpoints.v2.Profile.profile, {
|
|
body: input,
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.serviceToken}`,
|
|
},
|
|
})
|
|
|
|
if (!apiResponse.ok) {
|
|
await metricsSignup.httpError(apiResponse)
|
|
const text = await apiResponse.text()
|
|
throw serverErrorByStatus(apiResponse.status, text)
|
|
}
|
|
|
|
metricsSignup.success()
|
|
|
|
return {
|
|
success: true,
|
|
redirectUrl: signupVerify[input.language],
|
|
}
|
|
}),
|
|
profilingConsent: router({
|
|
update: protectedProcedure
|
|
.input(profilingConsentInput)
|
|
.mutation(async function ({ ctx, input }) {
|
|
const profilingConsentCounter = createCounter(
|
|
"trpc.user",
|
|
"profilingConsent"
|
|
)
|
|
const metricsProfilingConsent = profilingConsentCounter.init()
|
|
|
|
const apiResponse = await api.patch(api.endpoints.v2.Profile.profile, {
|
|
body: input,
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
})
|
|
|
|
if (!apiResponse.ok) {
|
|
await metricsProfilingConsent.httpError(apiResponse)
|
|
const text = await apiResponse.text()
|
|
throw serverErrorByStatus(apiResponse.status, text)
|
|
}
|
|
|
|
metricsProfilingConsent.success()
|
|
|
|
return true
|
|
}),
|
|
}),
|
|
promoCampaign: router({
|
|
add: protectedProcedure
|
|
.input(addPromoCampaignInput)
|
|
.mutation(async function ({ ctx, input }) {
|
|
userMutationLogger.info("api.user.promoCampaign.add start")
|
|
const apiResponse = await api.post(
|
|
api.endpoints.v2.Profile.promoCampaign,
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${ctx.session.token.access_token}`,
|
|
},
|
|
body: {
|
|
promotionId: input.promotionId,
|
|
language: input.language,
|
|
},
|
|
}
|
|
)
|
|
|
|
if (!apiResponse.ok) {
|
|
const text = await apiResponse.text()
|
|
userMutationLogger.error(
|
|
"api.user.promoCampaign.add error",
|
|
JSON.stringify({
|
|
query: {
|
|
promotionId: input.promotionId,
|
|
},
|
|
error: {
|
|
status: apiResponse.status,
|
|
statusText: apiResponse.statusText,
|
|
text,
|
|
},
|
|
})
|
|
)
|
|
return false
|
|
}
|
|
|
|
userMutationLogger.info(
|
|
"api.user.promoCampaign.add success",
|
|
JSON.stringify({
|
|
query: { promotionId: input.promotionId },
|
|
})
|
|
)
|
|
return true
|
|
}),
|
|
}),
|
|
})
|