bbf9274d37
* chore: upgrade Vitest to 4.0.16 and Vite to 6.4.1 - Update vitest from 2.1.9 to 4.0.16 - Update @vitest/ui from 2.1.9 to 4.0.16 - Update vitest-fetch-mock from 0.3.0 to 0.4.5 - Update vitest-mock-extended from 2.0.2 to 3.1.0 - Update vite from 4.5.14/5.4.21 to 6.4.1 across all packages - Update @vitejs/plugin-react to 5.1.2 - Update @vitejs/plugin-react-swc to 4.2.2 - Update @vitejs/plugin-basic-ssl to 2.1.0 - Update vite-plugin-dts to 4.5.4 - Rename vitest.config.ts to vitest.config.mts for ESM compatibility - Add globals: true to vitest config Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: address Vitest 4.0 and Vite 6 breaking changes - Convert arrow function mockImplementation patterns to regular functions (Vitest 4.0 breaking change: arrow functions can't be constructor mocks) - Fix CSS imports with ?inline suffix for Vite 6 compatibility - Add biome override to disable useArrowFunction rule for test files - Fix syntax errors in test files introduced by regex replacements Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: fix remaining Vitest 4.0 constructor mock patterns Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: fix more Vitest 4.0 constructor mock patterns and exclude API v2 spec files Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert more arrow function mocks to regular functions for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert more arrow function mocks to regular functions for Vitest 4.0 - Fix CrmService.integration.test.ts jsforce.Connection mock - Fix RetellSDKClient.test.ts Retell mock - Fix RetellAIService.test.ts CreditService mocks - Fix GoogleCalendarSubscriptionAdapter.test.ts CalendarAuth mock Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert Google Calendar and OAuthManager arrow function mocks for Vitest 4.0 - Fix googleapis.ts Calendar, OAuth2Client, and JWT mocks - Fix utils.ts JWT mock - Fix OAuthManager.ts defaultMockOAuthManager mock Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: add React plugin, jsdom environment, and fix more constructor mocks for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert HostRepository PrismaClient mock to regular function for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: add useOrgBranding mock to React component tests for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: update TestFunction type for Vitest 4.0 compatibility Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert listBookingReports constructor mocks to regular functions for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert UserRepository constructor mock to regular function for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert OrganizationPaymentService constructor mock to regular function for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert more constructor mocks to regular functions for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: add apps/web path aliases to vitest config Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: fix test issues for Vitest 4.0 compatibility - Fix Response constructor 204 status code issue in testUtils.ts - Fix FeaturesRepository mock persistence in handleNotificationWhenNoSlots.test.ts - Add @vitest-environment node directive to formSubmissionUtils.test.ts - Fix document.querySelector mock in embed.test.ts Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: clear EventManager spy between tests for Vitest 4.0 compatibility Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: update TeamRepository mock pattern for Vitest 4.0 compatibility Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert RoutingFormResponseRepository mock to regular function for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: convert more constructor mocks to regular functions for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: fix mock reset and spy clear issues for Vitest 4.0 Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: fix remaining test failures for Vitest 4.0 upgrade - Fix booking-validations.test.ts: convert UserRepository mock to regular function - Fix route.test.ts: update 500 error test to mock ImageResponse instead of fetch - Fix users-public-view.test.tsx: add missing mocks for getOrgFullOrigin and useRouterQuery - Add @calcom/web path alias to vitest config Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: add vitest-mocks for generated files that don't exist in CI - Add svg-hashes.json mock for route.test.ts - Add tailwind.generated.css mock for embed.test.ts - Update vitest config to use mock files Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: update vitest config aliases for CI compatibility - Use array format for aliases to ensure proper ordering - Add @calcom/platform-constants alias to resolve from source - Add @calcom/embed-react alias to resolve from source - Ensure svg-hashes.json mock alias is matched before @calcom/web Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: add @calcom/embed-snippet alias for CI compatibility Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * Fix wrong test * fix: migrate from CLI flags to VITEST_MODE env var for Vitest 4.0 Vitest 4.0 no longer allows custom CLI flags like --packaged-embed-tests-only. This change migrates to using VITEST_MODE environment variable instead: - VITEST_MODE=packaged-embed for packaged embed tests - VITEST_MODE=integration for integration tests - VITEST_MODE=timezone for timezone-dependent tests Updated vitest.config.mts to handle mode-based include/exclude patterns. Updated CI workflows and package scripts to use the new env var approach. Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: return default include pattern instead of undefined in vitest config The getTestInclude() function was returning undefined for the default case, but Vitest 4.0 expects an array. This caused 'resolved.include is not iterable' error in CI. Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: always set INTEGRATION_TEST_MODE for jsdom environment The getBookingFields.ts file checks for INTEGRATION_TEST_MODE to allow server-side imports in the jsdom environment. Without this, tests fail with 'getBookingFields must not be imported on the client side' error. Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> * fix: support legacy CLI flags for backwards compatibility with main workflow The CI runs workflows from main branch, which uses the old CLI flag approach (yarn test -- --integrationTestsOnly). This commit adds backwards compatibility by checking both VITEST_MODE env var and process.argv for the legacy flags. Co-Authored-By: Volnei Munhoz <volnei.munhoz@gmail.com> --------- Co-authored-by: Devin AI <158243242+devin-ai-integration[bot]@users.noreply.github.com>
490 lines
14 KiB
TypeScript
490 lines
14 KiB
TypeScript
import i18nMock from "../../../tests/libs/__mocks__/libServerI18n";
|
|
import prismaMock from "../../../tests/libs/__mocks__/prismaMock";
|
|
|
|
import { vi, describe, it, beforeAll, afterAll, expect, beforeEach, afterEach } from "vitest";
|
|
|
|
import dayjs from "@calcom/dayjs";
|
|
import * as CalcomEmails from "@calcom/emails/organization-email-service";
|
|
import { getNoSlotsNotificationService } from "@calcom/features/di/containers/NoSlotsNotification";
|
|
import { RedisService } from "@calcom/features/redis/RedisService";
|
|
|
|
vi.mock("@calcom/features/redis/RedisService", () => {
|
|
const mockedRedis = vi.fn();
|
|
mockedRedis.prototype.lrange = vi.fn();
|
|
mockedRedis.prototype.lpush = vi.fn();
|
|
mockedRedis.prototype.expire = vi.fn();
|
|
return {
|
|
RedisService: mockedRedis,
|
|
};
|
|
});
|
|
|
|
vi.mock("@calcom/features/flags/features.repository", () => ({
|
|
FeaturesRepository: vi.fn(function () {
|
|
return {
|
|
checkIfFeatureIsEnabledGlobally: vi.fn().mockResolvedValue(false),
|
|
};
|
|
}),
|
|
}));
|
|
|
|
vi.spyOn(CalcomEmails, "sendOrganizationAdminNoSlotsNotification");
|
|
|
|
describe("(Orgs) Send admin notifications when a user has no availability", () => {
|
|
beforeAll(() => {
|
|
// Setup env vars
|
|
vi.stubEnv("UPSTASH_REDIS_REST_TOKEN", "mocked_token");
|
|
vi.stubEnv("UPSTASH_REDIS_REST_URL", "mocked_url");
|
|
});
|
|
|
|
beforeEach(() => {
|
|
// Setup mocks
|
|
prismaMock.membership.findMany.mockResolvedValue([
|
|
{
|
|
user: {
|
|
email: "test@test.com",
|
|
locale: "en",
|
|
},
|
|
},
|
|
]);
|
|
|
|
i18nMock.getTranslation.mockImplementation(() => {
|
|
return new Promise((resolve) => {
|
|
const identityFn = (key: string) => key;
|
|
// @ts-expect-error Target allows only 1 element(s) but source may have more.
|
|
resolve(identityFn);
|
|
});
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
vi.resetAllMocks();
|
|
});
|
|
|
|
afterAll(() => {
|
|
vi.unstubAllEnvs();
|
|
});
|
|
|
|
it("Should send a notification after 2 times if the org has them enabled", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
prismaMock.team.findFirst.mockResolvedValue({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: true,
|
|
},
|
|
});
|
|
|
|
// Define event and organization details
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(), // Mocking Dayjs format function
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
// Call the function with teamId
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({ eventDetails, orgDetails, teamId: 123 });
|
|
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).not.toHaveBeenCalled();
|
|
|
|
// Mock length to be one then recall to trigger email
|
|
mocked.lrange.mockResolvedValueOnce([""]);
|
|
|
|
const service2 = getNoSlotsNotificationService();
|
|
await service2.handleNotificationWhenNoSlots({ eventDetails, orgDetails, teamId: 123 });
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalled();
|
|
});
|
|
it("Should not send a notification if the org has them disabled", async () => {
|
|
prismaMock.team.findFirst.mockResolvedValueOnce({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: false,
|
|
},
|
|
});
|
|
|
|
// Define event and organization details
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(), // Mocking Dayjs format function
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({ eventDetails, orgDetails, teamId: 123 });
|
|
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).not.toHaveBeenCalled();
|
|
});
|
|
it("Should only send notifications to admins of the specified teamId", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
|
|
prismaMock.team.findFirst.mockResolvedValue({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: true,
|
|
},
|
|
});
|
|
|
|
// Mock finding team members
|
|
prismaMock.membership.findMany.mockResolvedValue([
|
|
{
|
|
user: {
|
|
email: "correctteam@test.com",
|
|
locale: "en",
|
|
},
|
|
},
|
|
]);
|
|
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
// Mock Redis to simulate second no-slots occurrence
|
|
mocked.lrange.mockResolvedValueOnce([""]); // This will trigger email sending
|
|
|
|
// Call with specific teamId
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
teamId: 123, // specific teamId
|
|
});
|
|
|
|
// Verify that membership query included correct teamId
|
|
expect(prismaMock.membership.findMany).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
where: expect.objectContaining({
|
|
team: expect.objectContaining({
|
|
id: 123,
|
|
}),
|
|
}),
|
|
})
|
|
);
|
|
|
|
// Verify email was sent only once (to the one correct team member)
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledTimes(1);
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
to: {
|
|
email: "correctteam@test.com",
|
|
},
|
|
})
|
|
);
|
|
});
|
|
|
|
it("Should not send notifications when no teamId is provided", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
|
|
prismaMock.team.findFirst.mockResolvedValue({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: true,
|
|
},
|
|
});
|
|
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
mocked.lrange.mockResolvedValueOnce([""]);
|
|
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
// teamId intentionally omitted
|
|
});
|
|
|
|
expect(prismaMock.membership.findMany).not.toHaveBeenCalled();
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("Should not send notifications when no orgDomain is provided", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: null, // No org domain
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
mocked.lrange.mockResolvedValueOnce([""]);
|
|
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
expect(prismaMock.team.findFirst).not.toHaveBeenCalled();
|
|
expect(prismaMock.membership.findMany).not.toHaveBeenCalled();
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("Should not send notifications when Redis environment variables are not set", async () => {
|
|
// Temporarily unset Redis env vars
|
|
const originalToken = process.env.UPSTASH_REDIS_REST_TOKEN;
|
|
const originalUrl = process.env.UPSTASH_REDIS_REST_URL;
|
|
delete process.env.UPSTASH_REDIS_REST_TOKEN;
|
|
delete process.env.UPSTASH_REDIS_REST_URL;
|
|
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
expect(prismaMock.team.findFirst).not.toHaveBeenCalled();
|
|
expect(prismaMock.membership.findMany).not.toHaveBeenCalled();
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).not.toHaveBeenCalled();
|
|
|
|
// Restore env vars
|
|
process.env.UPSTASH_REDIS_REST_TOKEN = originalToken;
|
|
process.env.UPSTASH_REDIS_REST_URL = originalUrl;
|
|
});
|
|
|
|
it("Should handle multiple admins correctly", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
|
|
prismaMock.team.findFirst.mockResolvedValue({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: true,
|
|
},
|
|
});
|
|
|
|
// Mock multiple team admins
|
|
prismaMock.membership.findMany.mockResolvedValue([
|
|
{
|
|
user: {
|
|
email: "admin1@test.com",
|
|
locale: "en",
|
|
},
|
|
},
|
|
{
|
|
user: {
|
|
email: "admin2@test.com",
|
|
locale: "fr",
|
|
},
|
|
},
|
|
]);
|
|
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
mocked.lrange.mockResolvedValueOnce([""]);
|
|
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
// Verify emails were sent to both admins
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledTimes(2);
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
to: {
|
|
email: "admin1@test.com",
|
|
},
|
|
})
|
|
);
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
to: {
|
|
email: "admin2@test.com",
|
|
},
|
|
})
|
|
);
|
|
});
|
|
|
|
it("Should not send duplicate notifications within NO_SLOTS_NOTIFICATION_FREQUENCY", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
|
|
prismaMock.team.findFirst.mockResolvedValue({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: true,
|
|
},
|
|
});
|
|
|
|
prismaMock.membership.findMany.mockResolvedValue([
|
|
{
|
|
user: {
|
|
email: "admin@test.com",
|
|
locale: "en",
|
|
},
|
|
},
|
|
]);
|
|
|
|
const eventDetails = {
|
|
username: "user1",
|
|
eventSlug: "event1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
// First notification cycle - simulate having one previous occurrence
|
|
mocked.lrange.mockResolvedValueOnce([""]); // One previous occurrence
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
// Verify first notification was sent
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledTimes(1);
|
|
|
|
// Reset call counts
|
|
vi.clearAllMocks();
|
|
|
|
// For the second attempt, simulate having TWO occurrences already in Redis
|
|
// This better simulates the real Redis state after the first notification
|
|
mocked.lrange.mockResolvedValueOnce(["", ""]); // Two occurrences now
|
|
|
|
const service2 = getNoSlotsNotificationService();
|
|
await service2.handleNotificationWhenNoSlots({
|
|
eventDetails,
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
// Verify no additional notifications were sent
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).not.toHaveBeenCalled();
|
|
|
|
// Verify Redis operations
|
|
expect(mocked.lpush).toHaveBeenCalledTimes(1); // Still records the occurrence
|
|
});
|
|
|
|
it("Should maintain separate notification frequencies for different event types", async () => {
|
|
const redisService = new RedisService();
|
|
const mocked = vi.mocked(redisService);
|
|
|
|
prismaMock.team.findFirst.mockResolvedValue({
|
|
organizationSettings: {
|
|
adminGetsNoSlotsNotification: true,
|
|
},
|
|
});
|
|
|
|
prismaMock.membership.findMany.mockResolvedValue([
|
|
{
|
|
user: {
|
|
email: "admin@test.com",
|
|
locale: "en",
|
|
},
|
|
},
|
|
]);
|
|
|
|
const baseEventDetails = {
|
|
username: "user1",
|
|
startTime: dayjs(),
|
|
endTime: dayjs().add(1, "hour"),
|
|
};
|
|
|
|
const orgDetails = {
|
|
currentOrgDomain: "org1",
|
|
isValidOrgDomain: true,
|
|
};
|
|
|
|
// First event type notification
|
|
mocked.lrange.mockResolvedValueOnce([""]); // Simulate one previous occurrence for first event
|
|
const service = getNoSlotsNotificationService();
|
|
await service.handleNotificationWhenNoSlots({
|
|
eventDetails: { ...baseEventDetails, eventSlug: "event1" },
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
// Verify first notification was sent
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledTimes(1);
|
|
|
|
// Reset only the email mock, keep Redis mocks
|
|
vi.mocked(CalcomEmails.sendOrganizationAdminNoSlotsNotification).mockClear();
|
|
|
|
// For the second event type, also simulate one previous occurrence
|
|
// This needs to be a separate mock since it's a different key in Redis
|
|
mocked.lrange.mockResolvedValueOnce([""]); // Simulate one previous occurrence for second event
|
|
const service2 = getNoSlotsNotificationService();
|
|
await service2.handleNotificationWhenNoSlots({
|
|
eventDetails: { ...baseEventDetails, eventSlug: "event2" },
|
|
orgDetails,
|
|
teamId: 123,
|
|
});
|
|
|
|
// Verify second notification was sent (different event type)
|
|
expect(CalcomEmails.sendOrganizationAdminNoSlotsNotification).toHaveBeenCalledTimes(1);
|
|
|
|
// Get all lpush calls
|
|
const lpushCalls = mocked.lpush.mock.calls;
|
|
expect(lpushCalls.length).toBe(2);
|
|
|
|
// Extract the Redis keys used for each event
|
|
const firstEventKey = lpushCalls[0][0];
|
|
const secondEventKey = lpushCalls[1][0];
|
|
|
|
// Verify different keys were used
|
|
expect(firstEventKey).not.toBe(secondEventKey);
|
|
expect(firstEventKey).toContain("event1");
|
|
expect(secondEventKey).toContain("event2");
|
|
});
|
|
});
|