diff --git a/opensaas-sh/app_diff/main.wasp.diff b/opensaas-sh/app_diff/main.wasp.diff index fc78364..3ea8fee 100644 --- a/opensaas-sh/app_diff/main.wasp.diff +++ b/opensaas-sh/app_diff/main.wasp.diff @@ -88,9 +88,9 @@ + configFn: import { getDiscordAuthConfig } from "@src/auth/userSignupFields" + } }, - onAfterSignup: import { onAfterSignup } from "@src/auth/hooks", onAuthFailedRedirectTo: "/login", -@@ -87,11 +83,11 @@ + onAuthSucceededRedirectTo: "/demo-app", +@@ -86,11 +82,11 @@ // NOTE: "Dummy" provider is just for local development purposes. // Make sure to check the server logs for the email confirmation url (it will not be sent to an address)! // Once you are ready for production, switch to e.g. "SendGrid" or "Mailgun" providers. Check out https://docs.opensaas.sh/guides/email-sending/ . @@ -104,7 +104,7 @@ }, }, } -@@ -207,9 +203,9 @@ +@@ -206,9 +202,9 @@ } api paymentsWebhook { diff --git a/opensaas-sh/app_diff/src/auth/userSignupFields.ts.diff b/opensaas-sh/app_diff/src/auth/userSignupFields.ts.diff index 5272c53..cdc98d7 100644 --- a/opensaas-sh/app_diff/src/auth/userSignupFields.ts.diff +++ b/opensaas-sh/app_diff/src/auth/userSignupFields.ts.diff @@ -1,46 +1,64 @@ --- template/app/src/auth/userSignupFields.ts +++ opensaas-sh/app/src/auth/userSignupFields.ts -@@ -1,11 +1,8 @@ +@@ -1,8 +1,6 @@ import { z } from 'zod'; import { defineUserSignupFields } from 'wasp/auth/providers/types'; -const adminEmails = process.env.ADMIN_EMAILS?.split(',') || []; - - export const getEmailUserFields = defineUserSignupFields({ - username: (data: any) => data.email, -- isAdmin: (data: any) => adminEmails.includes(data.email), - email: (data: any) => data.email, + const emailDataSchema = z.object({ + email: z.string(), + }); +@@ -16,10 +14,6 @@ + const emailData = emailDataSchema.parse(data); + return emailData.email; + }, +- isAdmin: (data) => { +- const emailData = emailDataSchema.parse(data); +- return adminEmails.includes(emailData.email); +- }, }); -@@ -29,10 +26,6 @@ + const githubDataSchema = z.object({ +@@ -45,14 +39,6 @@ const githubData = githubDataSchema.parse(data); return githubData.profile.login; }, - isAdmin: (data) => { - const githubData = githubDataSchema.parse(data); -- return adminEmails.includes(githubData.profile.emails[0].email); +- const emailInfo = getGithubEmailInfo(githubData); +- if (!emailInfo.verified) { +- return false; +- } +- return adminEmails.includes(emailInfo.email); - }, }); - // NOTE: if we don't want to access users' emails, we can use scope ["user:read"] -@@ -58,10 +51,6 @@ + // We are using the first email from the list of emails returned by GitHub. +@@ -85,13 +71,6 @@ const googleData = googleDataSchema.parse(data); return googleData.profile.email; }, - isAdmin: (data) => { - const googleData = googleDataSchema.parse(data); +- if (!googleData.profile.email_verified) { +- return false; +- } - return adminEmails.includes(googleData.profile.email); - }, }); export function getGoogleAuthConfig() { -@@ -86,10 +75,6 @@ +@@ -121,13 +100,6 @@ const discordData = discordDataSchema.parse(data); return discordData.profile.username; }, - isAdmin: (data) => { -- const email = discordDataSchema.parse(data).profile.email; -- return !!email && adminEmails.includes(email); +- const discordData = discordDataSchema.parse(data); +- if (!discordData.profile.email || !discordData.profile.verified) { +- return false; +- } +- return adminEmails.includes(discordData.profile.email); - }, }); diff --git a/opensaas-sh/app_diff/src/payment/PricingPage.tsx.diff b/opensaas-sh/app_diff/src/payment/PricingPage.tsx.diff index a62f8b3..1fd6dcd 100644 --- a/opensaas-sh/app_diff/src/payment/PricingPage.tsx.diff +++ b/opensaas-sh/app_diff/src/payment/PricingPage.tsx.diff @@ -8,16 +8,7 @@ interface PaymentPlanCard { name: string; -@@ -82,7 +83,7 @@ - } - - if (!customerPortalUrl) { -- throw new Error(`Customer Portal does not exist for user ${user.id}`) -+ throw new Error(`Customer Portal does not exist for user ${user.id}`); - } - - window.open(customerPortalUrl, '_blank'); -@@ -96,11 +97,18 @@ +@@ -105,16 +106,24 @@ Pick your pricing @@ -37,11 +28,17 @@ + 4242 4242 4242 4242 4242 +

+ ++ + {errorMessage && ( +
+ {errorMessage} +
+ )} +
{Object.values(PaymentPlanId).map((planId) => (
diff --git a/template/app/main.wasp b/template/app/main.wasp index 4526868..5dfed47 100644 --- a/template/app/main.wasp +++ b/template/app/main.wasp @@ -66,7 +66,6 @@ app OpenSaaS { // configFn: import { getDiscordAuthConfig } from "@src/auth/userSignupFields" // } }, - onAfterSignup: import { onAfterSignup } from "@src/auth/hooks", onAuthFailedRedirectTo: "/login", onAuthSucceededRedirectTo: "/demo-app", }, diff --git a/template/app/src/auth/hooks.ts b/template/app/src/auth/hooks.ts deleted file mode 100644 index b652294..0000000 --- a/template/app/src/auth/hooks.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { HttpError } from 'wasp/server'; -import type { OnAfterSignupHook } from 'wasp/server/auth'; - -export const onAfterSignup: OnAfterSignupHook = async ({ providerId, user, prisma }) => { - // For Stripe to function correctly, we need a valid email associated with the user. - // Discord allows an email address to be optional. If this is the case, we delete the user - // from our DB and throw an error. - if (providerId.providerName === 'discord' && !user.email) { - await prisma.user.delete({ - where: { - id: user.id, - }, - }); - throw new HttpError(403, 'Discord user needs a valid email to sign up'); - } -}; diff --git a/template/app/src/auth/userSignupFields.ts b/template/app/src/auth/userSignupFields.ts index bea820f..43068f8 100644 --- a/template/app/src/auth/userSignupFields.ts +++ b/template/app/src/auth/userSignupFields.ts @@ -3,19 +3,35 @@ import { defineUserSignupFields } from 'wasp/auth/providers/types'; const adminEmails = process.env.ADMIN_EMAILS?.split(',') || []; +const emailDataSchema = z.object({ + email: z.string(), +}); + export const getEmailUserFields = defineUserSignupFields({ - username: (data: any) => data.email, - isAdmin: (data: any) => adminEmails.includes(data.email), - email: (data: any) => data.email, + email: (data) => { + const emailData = emailDataSchema.parse(data); + return emailData.email; + }, + username: (data) => { + const emailData = emailDataSchema.parse(data); + return emailData.email; + }, + isAdmin: (data) => { + const emailData = emailDataSchema.parse(data); + return adminEmails.includes(emailData.email); + }, }); const githubDataSchema = z.object({ profile: z.object({ - emails: z.array( - z.object({ - email: z.string(), - }) - ), + emails: z + .array( + z.object({ + email: z.string(), + verified: z.boolean(), + }) + ) + .min(1, 'You need to have an email address associated with your GitHub account to sign up.'), login: z.string(), }), }); @@ -23,7 +39,7 @@ const githubDataSchema = z.object({ export const getGitHubUserFields = defineUserSignupFields({ email: (data) => { const githubData = githubDataSchema.parse(data); - return githubData.profile.emails[0].email; + return getGithubEmailInfo(githubData).email; }, username: (data) => { const githubData = githubDataSchema.parse(data); @@ -31,10 +47,20 @@ export const getGitHubUserFields = defineUserSignupFields({ }, isAdmin: (data) => { const githubData = githubDataSchema.parse(data); - return adminEmails.includes(githubData.profile.emails[0].email); + const emailInfo = getGithubEmailInfo(githubData); + if (!emailInfo.verified) { + return false; + } + return adminEmails.includes(emailInfo.email); }, }); +// We are using the first email from the list of emails returned by GitHub. +// If you want to use a different email, you can modify this function. +function getGithubEmailInfo(githubData: z.infer) { + return githubData.profile.emails[0]; +} + // NOTE: if we don't want to access users' emails, we can use scope ["user:read"] // instead of ["user"] and access args.profile.username instead export function getGitHubAuthConfig() { @@ -46,6 +72,7 @@ export function getGitHubAuthConfig() { const googleDataSchema = z.object({ profile: z.object({ email: z.string(), + email_verified: z.boolean(), }), }); @@ -60,6 +87,9 @@ export const getGoogleUserFields = defineUserSignupFields({ }, isAdmin: (data) => { const googleData = googleDataSchema.parse(data); + if (!googleData.profile.email_verified) { + return false; + } return adminEmails.includes(googleData.profile.email); }, }); @@ -74,12 +104,17 @@ const discordDataSchema = z.object({ profile: z.object({ username: z.string(), email: z.string().email().nullable(), + verified: z.boolean().nullable(), }), }); export const getDiscordUserFields = defineUserSignupFields({ email: (data) => { const discordData = discordDataSchema.parse(data); + // Users need to have an email for payment processing. + if (!discordData.profile.email) { + throw new Error('You need to have an email address associated with your Discord account to sign up.'); + } return discordData.profile.email; }, username: (data) => { @@ -87,8 +122,11 @@ export const getDiscordUserFields = defineUserSignupFields({ return discordData.profile.username; }, isAdmin: (data) => { - const email = discordDataSchema.parse(data).profile.email; - return !!email && adminEmails.includes(email); + const discordData = discordDataSchema.parse(data); + if (!discordData.profile.email || !discordData.profile.verified) { + return false; + } + return adminEmails.includes(discordData.profile.email); }, }); diff --git a/template/app/src/payment/PricingPage.tsx b/template/app/src/payment/PricingPage.tsx index 41297c4..acde745 100644 --- a/template/app/src/payment/PricingPage.tsx +++ b/template/app/src/payment/PricingPage.tsx @@ -38,9 +38,11 @@ export const paymentPlanCards: Record = { const PricingPage = () => { const [isPaymentLoading, setIsPaymentLoading] = useState(false); - + const [errorMessage, setErrorMessage] = useState(null); + const { data: user } = useAuth(); - const isUserSubscribed = !!user && !!user.subscriptionStatus && user.subscriptionStatus !== SubscriptionStatus.Deleted; + const isUserSubscribed = + !!user && !!user.subscriptionStatus && user.subscriptionStatus !== SubscriptionStatus.Deleted; const { data: customerPortalUrl, @@ -65,8 +67,13 @@ const PricingPage = () => { } else { throw new Error('Error generating checkout session URL'); } - } catch (error) { + } catch (error: unknown) { console.error(error); + if (error instanceof Error) { + setErrorMessage(error.message); + } else { + setErrorMessage('Error processing payment. Please try again later.'); + } setIsPaymentLoading(false); // We only set this to false here and not in the try block because we redirect to the checkout url within the same window } } @@ -78,11 +85,13 @@ const PricingPage = () => { } if (customerPortalUrlError) { - console.error('Error fetching customer portal url'); + setErrorMessage('Error fetching Customer Portal URL'); + return; } if (!customerPortalUrl) { - throw new Error(`Customer Portal does not exist for user ${user.id}`) + setErrorMessage(`Customer Portal does not exist for user ${user.id}`); + return; } window.open(customerPortalUrl, '_blank'); @@ -101,6 +110,11 @@ const PricingPage = () => { out below with test credit card number
4242 4242 4242 4242 4242

+ {errorMessage && ( +
+ {errorMessage} +
+ )}
{Object.values(PaymentPlanId).map((planId) => (