
* --init * added default event types * updated lib path * updated group link design * fixed collective description * added default minimum booking notice * Accept multi user query for a default event type * check types * check types --WIP * check types still --WIP * --WIP * --WIP * fixed single user type not working * check fix * --import path fix * functional collective eventtype page * fixed check type * minor fixes and --WIP * typefix * custominput in defaultevent fix * added booking page compatibility for dynamic group links * added /book compatibility for dynamic group links * checktype fix --WIP * checktype fix * Success page compatibility added * added migrations * added dynamic group booking slug to booking creation * reschedule and database fix * daily integration * daily integration --locationtype fetch * fixed reschedule * added index to key parameter in eventtype list * fix + added after last group slug * added user setting option for dynamic booking * changed defaultEvents location based on recent changes * updated default event name in updated import * disallow booking when one in group disallows it * fixed setting checkbox association * cleanup * udded better error handling for disabled dynamic group bookings * cleanup * added tooltip to allow dynamic setting and enable by default * Update yarn.lock * Fix: Embed Fixes, UI configuration PRO Only, Tests (#2341) * #2325 Followup (#2369) * Adds initial MDX implementation for App Store pages * Adds endpoint to serve app store static files * Replaces zoom icon with dynamic-served one * Fixes zoom icon * Makes Slider reusable * Adds gray-matter for MDX * Adds zoom screenshots * Update yarn.lock * Slider improvements * WIP * Update TrendingAppsSlider.tsx * WIP * Adds MS teams screenshots * Adds stripe screenshots * Cleanup * Update index.ts * WIP * Cleanup * Cleanup * Adds jitsi screenshot * Adds Google meet screenshots * Adds office 365 calendar screenshots * Adds google calendar screenshots * Follow #2325 Co-authored-by: Peer Richelsen <peeroke@gmail.com> Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com> * requested changes * further requested changes * more changes * type fix * fixed prisma/client import path * added e2e test * test-fix * E2E fixes * Fixes circular dependency * Fixed paid bookings seeder * Added missing imports * requested changes * added username slugs as part of event description * updated event description Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com> Co-authored-by: zomars <zomars@me.com> Co-authored-by: Peer Richelsen <peeroke@gmail.com>
146 lines
4.2 KiB
TypeScript
146 lines
4.2 KiB
TypeScript
import { expect, test } from "@playwright/test";
|
|
|
|
import { deleteAllBookingsByEmail } from "./lib/teardown";
|
|
import {
|
|
bookFirstEvent,
|
|
bookTimeSlot,
|
|
selectFirstAvailableTimeSlotNextMonth,
|
|
selectSecondAvailableTimeSlotNextMonth,
|
|
todo,
|
|
} from "./lib/testUtils";
|
|
|
|
test.describe("free user", () => {
|
|
test.beforeEach(async ({ page }) => {
|
|
await page.goto("/free");
|
|
});
|
|
|
|
test.afterEach(async () => {
|
|
// delete test bookings
|
|
await deleteAllBookingsByEmail("free@example.com");
|
|
});
|
|
|
|
test("only one visible event", async ({ page }) => {
|
|
await expect(page.locator(`[href="/free/30min"]`)).toBeVisible();
|
|
await expect(page.locator(`[href="/free/60min"]`)).not.toBeVisible();
|
|
});
|
|
|
|
test("cannot book same slot multiple times", async ({ page }) => {
|
|
// Click first event type
|
|
await page.click('[data-testid="event-type-link"]');
|
|
|
|
await selectFirstAvailableTimeSlotNextMonth(page);
|
|
|
|
// Navigate to book page
|
|
await page.waitForNavigation({
|
|
url(url) {
|
|
return url.pathname.endsWith("/book");
|
|
},
|
|
});
|
|
|
|
// save booking url
|
|
const bookingUrl: string = page.url();
|
|
|
|
// book same time spot twice
|
|
await bookTimeSlot(page);
|
|
|
|
// Make sure we're navigated to the success page
|
|
await page.waitForNavigation({
|
|
url(url) {
|
|
return url.pathname.endsWith("/success");
|
|
},
|
|
});
|
|
|
|
// return to same time spot booking page
|
|
await page.goto(bookingUrl);
|
|
|
|
// book same time spot again
|
|
await bookTimeSlot(page);
|
|
|
|
// check for error message
|
|
await expect(page.locator("[data-testid=booking-fail]")).toBeVisible();
|
|
});
|
|
|
|
// Why do we need this test. The previous test is testing /30min booking only ?
|
|
todo("`/free/30min` is bookable");
|
|
|
|
test("`/free/60min` is not bookable", async ({ page }) => {
|
|
// Not available in listing
|
|
await expect(page.locator('[href="/free/60min"]')).toHaveCount(0);
|
|
|
|
await page.goto("/free/60min");
|
|
// Not available on a direct visit to event type page
|
|
await expect(page.locator('[data-testid="404-page"]')).toBeVisible();
|
|
});
|
|
});
|
|
|
|
test.describe("pro user", () => {
|
|
test.use({ storageState: "playwright/artifacts/proStorageState.json" });
|
|
|
|
test.beforeEach(async ({ page }) => {
|
|
await page.goto("/pro");
|
|
});
|
|
|
|
test.afterAll(async () => {
|
|
// delete test bookings
|
|
await deleteAllBookingsByEmail("pro@example.com");
|
|
});
|
|
|
|
test("pro user's page has at least 2 visible events", async ({ page }) => {
|
|
// await page.pause();
|
|
const $eventTypes = await page.locator("[data-testid=event-types] > *");
|
|
expect(await $eventTypes.count()).toBeGreaterThanOrEqual(2);
|
|
});
|
|
|
|
test("book an event first day in next month", async ({ page }) => {
|
|
// Click first event type
|
|
await page.click('[data-testid="event-type-link"]');
|
|
await selectFirstAvailableTimeSlotNextMonth(page);
|
|
await bookTimeSlot(page);
|
|
|
|
// Make sure we're navigated to the success page
|
|
await page.waitForNavigation({
|
|
url(url) {
|
|
return url.pathname.endsWith("/success");
|
|
},
|
|
});
|
|
});
|
|
test("can reschedule a booking", async ({ page }) => {
|
|
await bookFirstEvent(page);
|
|
|
|
await page.goto("/bookings/upcoming");
|
|
await page.locator('[data-testid="reschedule"]').click();
|
|
await page.waitForNavigation({
|
|
url: (url) => {
|
|
const bookingId = url.searchParams.get("rescheduleUid");
|
|
return !!bookingId;
|
|
},
|
|
});
|
|
await selectSecondAvailableTimeSlotNextMonth(page);
|
|
// --- fill form
|
|
await page.locator('[data-testid="confirm-reschedule-button"]').click();
|
|
await page.waitForNavigation({
|
|
url(url) {
|
|
return url.pathname === "/success" && url.searchParams.get("reschedule") === "true";
|
|
},
|
|
});
|
|
});
|
|
|
|
test("Can cancel the recently created booking", async ({ page }) => {
|
|
await bookFirstEvent(page);
|
|
|
|
await page.goto("/bookings/upcoming");
|
|
await page.locator('[data-testid="cancel"]').first().click();
|
|
await page.waitForNavigation({
|
|
url: (url) => {
|
|
return url.pathname.startsWith("/cancel");
|
|
},
|
|
});
|
|
// --- fill form
|
|
await page.locator('[data-testid="cancel"]').click();
|
|
await page.waitForNavigation({
|
|
url(url) {
|
|
return url.pathname === "/cancel/success";
|
|
},
|
|
});
|
|
});
|
|
});
|