 7bc7b241ac
			
		
	
	
		7bc7b241ac
		
			
		
	
	
	
	
		
			
			* Email input UX improvements * Makes email queries case insensitive * Lowercases all emails * Type fixes * Re adds lowercase email to login * Removes citext dependency * Updates schema * Migration fixes * Added failsafes to team invites * Team invite improvements * Deleting the index, lowercasing ``` calendso=> UPDATE users SET email=LOWER(email); ERROR: duplicate key value violates unique constraint "users.email_unique" DETAIL: Key (email)=(free@example.com) already exists. ``` vs. ``` calendso=> CREATE UNIQUE INDEX "users.email_unique" ON "users" (email); ERROR: could not create unique index "users.email_unique" DETAIL: Key (email)=(Free@example.com) is duplicated. ``` I think it'll be easier to rectify for users if they try to run the migrations if the index stays in place. Co-authored-by: Alex van Andel <me@alexvanandel.com>
		
			
				
	
	
		
			102 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			102 lines
		
	
	
	
		
			2.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { GetServerSidePropsContext } from "next";
 | |
| import "react-phone-number-input/style.css";
 | |
| 
 | |
| import { asStringOrThrow } from "@lib/asStringOrNull";
 | |
| import prisma from "@lib/prisma";
 | |
| import { inferSSRProps } from "@lib/types/inferSSRProps";
 | |
| 
 | |
| import BookingPage from "@components/booking/pages/BookingPage";
 | |
| 
 | |
| export type TeamBookingPageProps = inferSSRProps<typeof getServerSideProps>;
 | |
| 
 | |
| export default function TeamBookingPage(props: TeamBookingPageProps) {
 | |
|   return <BookingPage {...props} />;
 | |
| }
 | |
| 
 | |
| export async function getServerSideProps(context: GetServerSidePropsContext) {
 | |
|   const eventTypeId = parseInt(asStringOrThrow(context.query.type));
 | |
|   if (typeof eventTypeId !== "number" || eventTypeId % 1 !== 0) {
 | |
|     return {
 | |
|       notFound: true,
 | |
|     } as const;
 | |
|   }
 | |
| 
 | |
|   const eventType = await prisma.eventType.findUnique({
 | |
|     where: {
 | |
|       id: eventTypeId,
 | |
|     },
 | |
|     select: {
 | |
|       id: true,
 | |
|       title: true,
 | |
|       slug: true,
 | |
|       description: true,
 | |
|       length: true,
 | |
|       locations: true,
 | |
|       customInputs: true,
 | |
|       periodType: true,
 | |
|       periodDays: true,
 | |
|       periodStartDate: true,
 | |
|       periodEndDate: true,
 | |
|       periodCountCalendarDays: true,
 | |
|       disableGuests: true,
 | |
|       price: true,
 | |
|       currency: true,
 | |
|       team: {
 | |
|         select: {
 | |
|           slug: true,
 | |
|           name: true,
 | |
|           logo: true,
 | |
|         },
 | |
|       },
 | |
|       users: {
 | |
|         select: {
 | |
|           avatar: true,
 | |
|           name: true,
 | |
|         },
 | |
|       },
 | |
|     },
 | |
|   });
 | |
| 
 | |
|   if (!eventType) return { notFound: true };
 | |
| 
 | |
|   const eventTypeObject = [eventType].map((e) => {
 | |
|     return {
 | |
|       ...e,
 | |
|       periodStartDate: e.periodStartDate?.toString() ?? null,
 | |
|       periodEndDate: e.periodEndDate?.toString() ?? null,
 | |
|     };
 | |
|   })[0];
 | |
| 
 | |
|   let booking = null;
 | |
| 
 | |
|   if (context.query.rescheduleUid) {
 | |
|     booking = await prisma.booking.findFirst({
 | |
|       where: {
 | |
|         uid: asStringOrThrow(context.query.rescheduleUid),
 | |
|       },
 | |
|       select: {
 | |
|         description: true,
 | |
|         attendees: {
 | |
|           select: {
 | |
|             email: true,
 | |
|             name: true,
 | |
|           },
 | |
|         },
 | |
|       },
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     props: {
 | |
|       profile: {
 | |
|         ...eventTypeObject.team,
 | |
|         slug: "team/" + eventTypeObject.slug,
 | |
|         image: eventTypeObject.team?.logo || null,
 | |
|         theme: null /* Teams don't have a theme, and `BookingPage` uses it */,
 | |
|         brandColor: null /* Teams don't have a brandColor, and `BookingPage` uses it */,
 | |
|       },
 | |
|       eventType: eventTypeObject,
 | |
|       booking,
 | |
|     },
 | |
|   };
 | |
| }
 |