617 lines
17 KiB
TypeScript
617 lines
17 KiB
TypeScript
import { BookingStatus, MembershipRole, Prisma, UserPermissionRole } from "@prisma/client";
|
||
import { uuid } from "short-uuid";
|
||
|
||
import dailyMeta from "@calcom/app-store/dailyvideo/_metadata";
|
||
import googleMeetMeta from "@calcom/app-store/googlevideo/_metadata";
|
||
import zoomMeta from "@calcom/app-store/zoomvideo/_metadata";
|
||
import dayjs from "@calcom/dayjs";
|
||
import { hashPassword } from "@calcom/lib/auth";
|
||
import { DEFAULT_SCHEDULE, getAvailabilityFromSchedule } from "@calcom/lib/availability";
|
||
|
||
import prisma from ".";
|
||
import mainAppStore from "./seed-app-store";
|
||
|
||
async function createUserAndEventType(opts: {
|
||
user: {
|
||
email: string;
|
||
password: string;
|
||
username: string;
|
||
name: string;
|
||
completedOnboarding?: boolean;
|
||
timeZone?: string;
|
||
role?: UserPermissionRole;
|
||
};
|
||
eventTypes: Array<
|
||
Prisma.EventTypeCreateInput & {
|
||
_bookings?: Prisma.BookingCreateInput[];
|
||
}
|
||
>;
|
||
}) {
|
||
const userData = {
|
||
...opts.user,
|
||
password: await hashPassword(opts.user.password),
|
||
emailVerified: new Date(),
|
||
completedOnboarding: opts.user.completedOnboarding ?? true,
|
||
locale: "en",
|
||
schedules:
|
||
opts.user.completedOnboarding ?? true
|
||
? {
|
||
create: {
|
||
name: "Working Hours",
|
||
availability: {
|
||
createMany: {
|
||
data: getAvailabilityFromSchedule(DEFAULT_SCHEDULE),
|
||
},
|
||
},
|
||
},
|
||
}
|
||
: undefined,
|
||
};
|
||
|
||
const user = await prisma.user.upsert({
|
||
where: { email: opts.user.email },
|
||
update: userData,
|
||
create: userData,
|
||
});
|
||
|
||
console.log(
|
||
`👤 Upserted '${opts.user.username}' with email "${opts.user.email}" & password "${opts.user.password}". Booking page 👉 ${process.env.NEXT_PUBLIC_WEBAPP_URL}/${opts.user.username}`
|
||
);
|
||
|
||
for (const eventTypeInput of opts.eventTypes) {
|
||
const { _bookings: bookingInputs = [], ...eventTypeData } = eventTypeInput;
|
||
eventTypeData.userId = user.id;
|
||
eventTypeData.users = { connect: { id: user.id } };
|
||
|
||
const eventType = await prisma.eventType.findFirst({
|
||
where: {
|
||
slug: eventTypeData.slug,
|
||
users: {
|
||
some: {
|
||
id: eventTypeData.userId,
|
||
},
|
||
},
|
||
},
|
||
select: {
|
||
id: true,
|
||
},
|
||
});
|
||
|
||
if (eventType) {
|
||
console.log(
|
||
`\t📆 Event type ${eventTypeData.slug} already seems seeded - ${process.env.NEXT_PUBLIC_WEBAPP_URL}/${user.username}/${eventTypeData.slug}`
|
||
);
|
||
continue;
|
||
}
|
||
const { id } = await prisma.eventType.create({
|
||
data: eventTypeData,
|
||
});
|
||
|
||
console.log(
|
||
`\t📆 Event type ${eventTypeData.slug} with id ${id}, length ${eventTypeData.length}min - ${process.env.NEXT_PUBLIC_WEBAPP_URL}/${user.username}/${eventTypeData.slug}`
|
||
);
|
||
for (const bookingInput of bookingInputs) {
|
||
await prisma.booking.create({
|
||
data: {
|
||
...bookingInput,
|
||
user: {
|
||
connect: {
|
||
email: opts.user.email,
|
||
},
|
||
},
|
||
attendees: {
|
||
create: {
|
||
email: opts.user.email,
|
||
name: opts.user.name,
|
||
timeZone: "Europe/London",
|
||
},
|
||
},
|
||
eventType: {
|
||
connect: {
|
||
id,
|
||
},
|
||
},
|
||
status: bookingInput.status,
|
||
},
|
||
});
|
||
console.log(
|
||
`\t\t☎️ Created booking ${bookingInput.title} at ${new Date(
|
||
bookingInput.startTime
|
||
).toLocaleDateString()}`
|
||
);
|
||
}
|
||
}
|
||
|
||
return user;
|
||
}
|
||
|
||
async function createTeamAndAddUsers(
|
||
teamInput: Prisma.TeamCreateInput,
|
||
users: { id: number; username: string; role?: MembershipRole }[]
|
||
) {
|
||
const createTeam = async (team: Prisma.TeamCreateInput) => {
|
||
try {
|
||
return await prisma.team.create({
|
||
data: {
|
||
...team,
|
||
},
|
||
});
|
||
} catch (_err) {
|
||
if (_err instanceof Error && _err.message.indexOf("Unique constraint failed on the fields") !== -1) {
|
||
console.log(`Team '${team.name}' already exists, skipping.`);
|
||
return;
|
||
}
|
||
throw _err;
|
||
}
|
||
};
|
||
|
||
const team = await createTeam(teamInput);
|
||
if (!team) {
|
||
return;
|
||
}
|
||
|
||
console.log(
|
||
`🏢 Created team '${teamInput.name}' - ${process.env.NEXT_PUBLIC_WEBAPP_URL}/team/${team.slug}`
|
||
);
|
||
|
||
for (const user of users) {
|
||
const { role = MembershipRole.OWNER, id, username } = user;
|
||
await prisma.membership.create({
|
||
data: {
|
||
teamId: team.id,
|
||
userId: id,
|
||
role: role,
|
||
accepted: true,
|
||
},
|
||
});
|
||
console.log(`\t👤 Added '${teamInput.name}' membership for '${username}' with role '${role}'`);
|
||
}
|
||
}
|
||
|
||
async function main() {
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "delete-me@example.com",
|
||
password: "delete-me",
|
||
username: "delete-me",
|
||
name: "delete-me",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "onboarding@example.com",
|
||
password: "onboarding",
|
||
username: "onboarding",
|
||
name: "onboarding",
|
||
completedOnboarding: false,
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "free-first-hidden@example.com",
|
||
password: "free-first-hidden",
|
||
username: "free-first-hidden",
|
||
name: "Free First Hidden Example",
|
||
},
|
||
eventTypes: [
|
||
{
|
||
title: "30min",
|
||
slug: "30min",
|
||
length: 30,
|
||
hidden: true,
|
||
},
|
||
{
|
||
title: "60min",
|
||
slug: "60min",
|
||
length: 30,
|
||
},
|
||
],
|
||
});
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "pro@example.com",
|
||
name: "Pro Example",
|
||
password: "pro",
|
||
username: "pro",
|
||
},
|
||
eventTypes: [
|
||
{
|
||
title: "30min",
|
||
slug: "30min",
|
||
length: 30,
|
||
_bookings: [
|
||
{
|
||
uid: uuid(),
|
||
title: "30min",
|
||
startTime: dayjs().add(1, "day").toDate(),
|
||
endTime: dayjs().add(1, "day").add(30, "minutes").toDate(),
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "30min",
|
||
startTime: dayjs().add(2, "day").toDate(),
|
||
endTime: dayjs().add(2, "day").add(30, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
{
|
||
// hardcode UID so that we can easily test rescheduling in embed
|
||
uid: "qm3kwt3aTnVD7vmP9tiT2f",
|
||
title: "30min Seeded Booking",
|
||
startTime: dayjs().add(3, "day").toDate(),
|
||
endTime: dayjs().add(3, "day").add(30, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
],
|
||
},
|
||
{
|
||
title: "60min",
|
||
slug: "60min",
|
||
length: 60,
|
||
},
|
||
{
|
||
title: "Multiple duration",
|
||
slug: "multiple-duration",
|
||
length: 75,
|
||
metadata: {
|
||
multipleDuration: [30, 75, 90],
|
||
},
|
||
},
|
||
{
|
||
title: "paid",
|
||
slug: "paid",
|
||
length: 60,
|
||
price: 100,
|
||
},
|
||
{
|
||
title: "In person meeting",
|
||
slug: "in-person",
|
||
length: 60,
|
||
locations: [{ type: "inPerson", address: "London" }],
|
||
},
|
||
{
|
||
title: "Zoom Event",
|
||
slug: "zoom",
|
||
length: 60,
|
||
locations: [{ type: zoomMeta.appData?.location.type }],
|
||
},
|
||
{
|
||
title: "Daily Event",
|
||
slug: "daily",
|
||
length: 60,
|
||
locations: [{ type: dailyMeta.appData?.location.type }],
|
||
},
|
||
{
|
||
title: "Google Meet",
|
||
slug: "google-meet",
|
||
length: 60,
|
||
locations: [{ type: googleMeetMeta.appData?.location.type }],
|
||
},
|
||
{
|
||
title: "Yoga class",
|
||
slug: "yoga-class",
|
||
length: 30,
|
||
recurringEvent: { freq: 2, count: 12, interval: 1 },
|
||
_bookings: [
|
||
{
|
||
uid: uuid(),
|
||
title: "Yoga class",
|
||
recurringEventId: Buffer.from("yoga-class").toString("base64"),
|
||
startTime: dayjs().add(1, "day").toDate(),
|
||
endTime: dayjs().add(1, "day").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Yoga class",
|
||
recurringEventId: Buffer.from("yoga-class").toString("base64"),
|
||
startTime: dayjs().add(1, "day").add(1, "week").toDate(),
|
||
endTime: dayjs().add(1, "day").add(1, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Yoga class",
|
||
recurringEventId: Buffer.from("yoga-class").toString("base64"),
|
||
startTime: dayjs().add(1, "day").add(2, "week").toDate(),
|
||
endTime: dayjs().add(1, "day").add(2, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Yoga class",
|
||
recurringEventId: Buffer.from("yoga-class").toString("base64"),
|
||
startTime: dayjs().add(1, "day").add(3, "week").toDate(),
|
||
endTime: dayjs().add(1, "day").add(3, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Yoga class",
|
||
recurringEventId: Buffer.from("yoga-class").toString("base64"),
|
||
startTime: dayjs().add(1, "day").add(4, "week").toDate(),
|
||
endTime: dayjs().add(1, "day").add(4, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Yoga class",
|
||
recurringEventId: Buffer.from("yoga-class").toString("base64"),
|
||
startTime: dayjs().add(1, "day").add(5, "week").toDate(),
|
||
endTime: dayjs().add(1, "day").add(5, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Seeded Yoga class",
|
||
description: "seeded",
|
||
recurringEventId: Buffer.from("seeded-yoga-class").toString("base64"),
|
||
startTime: dayjs().subtract(4, "day").toDate(),
|
||
endTime: dayjs().subtract(4, "day").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Seeded Yoga class",
|
||
description: "seeded",
|
||
recurringEventId: Buffer.from("seeded-yoga-class").toString("base64"),
|
||
startTime: dayjs().subtract(4, "day").add(1, "week").toDate(),
|
||
endTime: dayjs().subtract(4, "day").add(1, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Seeded Yoga class",
|
||
description: "seeded",
|
||
recurringEventId: Buffer.from("seeded-yoga-class").toString("base64"),
|
||
startTime: dayjs().subtract(4, "day").add(2, "week").toDate(),
|
||
endTime: dayjs().subtract(4, "day").add(2, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Seeded Yoga class",
|
||
description: "seeded",
|
||
recurringEventId: Buffer.from("seeded-yoga-class").toString("base64"),
|
||
startTime: dayjs().subtract(4, "day").add(3, "week").toDate(),
|
||
endTime: dayjs().subtract(4, "day").add(3, "week").add(30, "minutes").toDate(),
|
||
status: BookingStatus.ACCEPTED,
|
||
},
|
||
],
|
||
},
|
||
{
|
||
title: "Tennis class",
|
||
slug: "tennis-class",
|
||
length: 60,
|
||
recurringEvent: { freq: 2, count: 10, interval: 2 },
|
||
requiresConfirmation: true,
|
||
_bookings: [
|
||
{
|
||
uid: uuid(),
|
||
title: "Tennis class",
|
||
recurringEventId: Buffer.from("tennis-class").toString("base64"),
|
||
startTime: dayjs().add(2, "day").toDate(),
|
||
endTime: dayjs().add(2, "day").add(60, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Tennis class",
|
||
recurringEventId: Buffer.from("tennis-class").toString("base64"),
|
||
startTime: dayjs().add(2, "day").add(2, "week").toDate(),
|
||
endTime: dayjs().add(2, "day").add(2, "week").add(60, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Tennis class",
|
||
recurringEventId: Buffer.from("tennis-class").toString("base64"),
|
||
startTime: dayjs().add(2, "day").add(4, "week").toDate(),
|
||
endTime: dayjs().add(2, "day").add(4, "week").add(60, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Tennis class",
|
||
recurringEventId: Buffer.from("tennis-class").toString("base64"),
|
||
startTime: dayjs().add(2, "day").add(8, "week").toDate(),
|
||
endTime: dayjs().add(2, "day").add(8, "week").add(60, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
{
|
||
uid: uuid(),
|
||
title: "Tennis class",
|
||
recurringEventId: Buffer.from("tennis-class").toString("base64"),
|
||
startTime: dayjs().add(2, "day").add(10, "week").toDate(),
|
||
endTime: dayjs().add(2, "day").add(10, "week").add(60, "minutes").toDate(),
|
||
status: BookingStatus.PENDING,
|
||
},
|
||
],
|
||
},
|
||
],
|
||
});
|
||
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "trial@example.com",
|
||
password: "trial",
|
||
username: "trial",
|
||
name: "Trial Example",
|
||
},
|
||
eventTypes: [
|
||
{
|
||
title: "30min",
|
||
slug: "30min",
|
||
length: 30,
|
||
},
|
||
{
|
||
title: "60min",
|
||
slug: "60min",
|
||
length: 60,
|
||
},
|
||
],
|
||
});
|
||
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "free@example.com",
|
||
password: "free",
|
||
username: "free",
|
||
name: "Free Example",
|
||
},
|
||
eventTypes: [
|
||
{
|
||
title: "30min",
|
||
slug: "30min",
|
||
length: 30,
|
||
},
|
||
{
|
||
title: "60min",
|
||
slug: "60min",
|
||
length: 30,
|
||
},
|
||
],
|
||
});
|
||
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "usa@example.com",
|
||
password: "usa",
|
||
username: "usa",
|
||
name: "USA Timezone Example",
|
||
timeZone: "America/Phoenix",
|
||
},
|
||
eventTypes: [
|
||
{
|
||
title: "30min",
|
||
slug: "30min",
|
||
length: 30,
|
||
},
|
||
],
|
||
});
|
||
|
||
const freeUserTeam = await createUserAndEventType({
|
||
user: {
|
||
email: "teamfree@example.com",
|
||
password: "teamfree",
|
||
username: "teamfree",
|
||
name: "Team Free Example",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
const proUserTeam = await createUserAndEventType({
|
||
user: {
|
||
email: "teampro@example.com",
|
||
password: "teampro",
|
||
username: "teampro",
|
||
name: "Team Pro Example",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
await createUserAndEventType({
|
||
user: {
|
||
email: "admin@example.com",
|
||
/** To comply with admin password requirements */
|
||
password: "ADMINadmin2022!",
|
||
username: "admin",
|
||
name: "Admin Example",
|
||
role: "ADMIN",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
const pro2UserTeam = await createUserAndEventType({
|
||
user: {
|
||
email: "teampro2@example.com",
|
||
password: "teampro2",
|
||
username: "teampro2",
|
||
name: "Team Pro Example 2",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
const pro3UserTeam = await createUserAndEventType({
|
||
user: {
|
||
email: "teampro3@example.com",
|
||
password: "teampro3",
|
||
username: "teampro3",
|
||
name: "Team Pro Example 3",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
const pro4UserTeam = await createUserAndEventType({
|
||
user: {
|
||
email: "teampro4@example.com",
|
||
password: "teampro4",
|
||
username: "teampro4",
|
||
name: "Team Pro Example 4",
|
||
},
|
||
eventTypes: [],
|
||
});
|
||
|
||
await createTeamAndAddUsers(
|
||
{
|
||
name: "Seeded Team",
|
||
slug: "seeded-team",
|
||
eventTypes: {
|
||
createMany: {
|
||
data: [
|
||
{
|
||
title: "Collective Seeded Team Event",
|
||
slug: "collective-seeded-team-event",
|
||
length: 15,
|
||
schedulingType: "COLLECTIVE",
|
||
},
|
||
{
|
||
title: "Round Robin Seeded Team Event",
|
||
slug: "round-robin-seeded-team-event",
|
||
length: 15,
|
||
schedulingType: "ROUND_ROBIN",
|
||
},
|
||
],
|
||
},
|
||
},
|
||
createdAt: new Date(),
|
||
},
|
||
[
|
||
{
|
||
id: proUserTeam.id,
|
||
username: proUserTeam.name || "Unknown",
|
||
},
|
||
{
|
||
id: freeUserTeam.id,
|
||
username: freeUserTeam.name || "Unknown",
|
||
},
|
||
{
|
||
id: pro2UserTeam.id,
|
||
username: pro2UserTeam.name || "Unknown",
|
||
role: "MEMBER",
|
||
},
|
||
{
|
||
id: pro3UserTeam.id,
|
||
username: pro3UserTeam.name || "Unknown",
|
||
},
|
||
{
|
||
id: pro4UserTeam.id,
|
||
username: pro4UserTeam.name || "Unknown",
|
||
},
|
||
]
|
||
);
|
||
}
|
||
|
||
main()
|
||
.then(() => mainAppStore())
|
||
.catch((e) => {
|
||
console.error(e);
|
||
process.exit(1);
|
||
})
|
||
.finally(async () => {
|
||
await prisma.$disconnect();
|
||
});
|