Skip to content

Commit bc665cb

Browse files
hariombalharadevin-ai-integration[bot]ThyMinimalDev
authored
fix: APIV2 team membership - Member not getting added to event-type automatically (calcom#24780)
* fix: APIV2 team membership addition * feat: Add trimming for email domain and orgAutoAcceptEmail in auto-accept logic - Trim whitespace from both user email domain and orgAutoAcceptEmail - Ensures consistent matching even with accidental whitespace - Addresses feedback from PR review Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * simplify * feat: Use shared OrganizationMembershipService in TRPC for consistent auto-accept logic - Create OrganizationMembershipService.container.ts for DI in TRPC - Update getOrgConnectionInfo to apply trimming + case-insensitive comparison - Precompute auto-accept decisions in createNewUsersConnectToOrgIfExists using the service - Use service in handleNewUsersInvites for consistent auto-accept determination - Ensures both API v2 and TRPC paths use identical trimming and normalization logic Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * Revert "feat: Use shared OrganizationMembershipService in TRPC for consistent auto-accept logic" This reverts commit 0b2bd28. * refactor: Unify OrganizationRepository and remove duplicate PrismaOrganizationRepository (calcom#24869) * refactor: Convert OrganizationRepository from static to instance methods - Add constructor accepting deps object with prismaClient - Convert all static methods to instance methods - Add getOrganizationAutoAcceptSettings method - Create singleton instance export in repository barrel file - Update API v2 OrganizationsRepository to extend from OrganizationRepository - Update all call sites to use singleton instance - Add platform-libraries organizations.ts export - Fix mock imports to use repository barrel - Fix unsafe optional chaining in next-auth-options.ts - Fix any types in test files with proper type inference Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Update all imports to use OrganizationRepository barrel export - Update imports from direct OrganizationRepository file to barrel export - This ensures mocks work correctly in tests - Fixes 202 failing tests related to organizationRepository mock Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Update test mocks to use partial mock pattern - Convert organizationMock to partial mock that preserves real class - Add proper prisma mocks to failing test files - Remove old OrganizationRepository mocks from test files - This fixes test failures related to mock interception Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Export mocked singleton and update tests to use it directly - Export mockedSingleton as organizationRepositoryMock from organizationMock - Update delegationCredential.test.ts to import and use the exported mock - This fixes 'vi.mocked(...).mockResolvedValue is not a function' errors Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Use platform-libraries import for API v2 OrganizationRepository API v2 should import shared features through @calcom/platform-libraries instead of directly from @calcom/features to maintain proper architectural boundaries and packaging/licensing separation. Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * refactor: Implement DI pattern for OrganizationRepository - Create OrganizationRepository.module.ts and .container.ts for DI - Replace singleton pattern with getOrganizationRepository() across 20 files - Update platform-libraries to export getOrganizationRepository - Delete duplicate PrismaOrganizationRepository.ts - Remove singleton export file (repositories/index.ts) - Update test mocks to use DI container pattern - All type checks and unit tests passing Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Implement read/write client separation in OrganizationRepository - Updated OrganizationRepository constructor to accept optional prismaWriteClient parameter - Routed all write operations (create, update) through prismaWrite client - Routed all read operations (find, get) through prismaRead client - Updated API v2 OrganizationsRepository to pass both dbRead.prisma and dbWrite.prisma to super() - Optimized getOrganizationRepository() calls by storing in local variables to avoid repeated function calls - This fixes the critical issue where API v2 was passing read-only client to base class with write methods Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Fix mock setup and optimize getOrganizationRepository() calls - Fixed verify-email.test.ts mock to return mocked repository instance instead of scenario helper object - Added mockReset to organizationMock.ts beforeEach to properly reset mock implementations between tests - Added local variables in page.tsx to store getOrganizationRepository() result for consistency This fixes the issue where getOrganizationRepository() was returning organizationScenarios.organizationRepository (scenario helper) instead of the actual mocked repository instance, causing findUniqueNonPlatformOrgsByMatchingAutoAcceptEmail to be undefined. Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * refactor: Simplify OrganizationRepository to use single prismaClient - Updated base OrganizationRepository constructor to accept only { prismaClient } instead of { prismaClient, prismaWriteClient? } - Replaced this.prismaRead and this.prismaWrite with single this.prismaClient property - Updated API v2 OrganizationsRepository to pass only dbWrite.prisma as prismaClient - Removed unused PrismaReadService import from API v2 - All read and write operations now use the same client instance This simplifies the architecture as requested - API v2 uses write client for all operations, and apps/web uses the client from DI container. Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> --------- Co-authored-by: Devin AI <158243242+devin-ai-integration[bot]@users.noreply.github.com> * fix: Match OrganizationsRepository.findById signature with base class The findById method in OrganizationsRepository was using a different signature than the base OrganizationRepository class, causing type errors in CI. Changed from: findById(organizationId: number) Changed to: findById({ id }: { id: number }) This matches the base class signature and resolves the CI unit test failures. Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Update all findById call sites to use object parameter Fixed 6 call sites in API v2 that were calling findById with a number instead of the required { id: number } object parameter: - is-org.guard.ts - is-admin-api-enabled.guard.ts - is-webhook-in-org.guard.ts - organizations.service.ts - managed-organizations.service.ts (2 call sites) This resolves the API v2 build failure caused by the signature change in OrganizationsRepository.findById to match the base class. Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * fix: Remove redundant findById override from OrganizationsRepository The findById method was duplicating the base class OrganizationRepository implementation. Both methods had identical logic (filtering by isOrganization: true), so the override was unnecessary. Since OrganizationsRepository extends OrganizationRepository and passes dbWrite.prisma to the base constructor, the base class method already provides the exact same functionality. This resolves the API v2 build failure by eliminating the duplicate method that was causing conflicts. Co-Authored-By: hariom@cal.com <hariombalhara@gmail.com> * Remove unncessary changes * Store in variable * Revert "Remove unncessary changes" This reverts commit af93517. * Revert dbRead/dbWrite changes * Add organizations library to tsconfig.json --------- Co-authored-by: Devin AI <158243242+devin-ai-integration[bot]@users.noreply.github.com> Co-authored-by: Morgan <33722304+ThyMinimalDev@users.noreply.github.com>
1 parent 02b1393 commit bc665cb

50 files changed

Lines changed: 583 additions & 159 deletions

File tree

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.
Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
import { OrganizationsRepository } from "@/modules/organizations/index/organizations.repository";
2+
import { Injectable } from "@nestjs/common";
3+
4+
import { OrganizationMembershipService as BaseOrganizationMembershipService } from "@calcom/platform-libraries/organizations";
5+
6+
@Injectable()
7+
export class OrganizationMembershipService extends BaseOrganizationMembershipService {
8+
constructor(organizationsRepository: OrganizationsRepository) {
9+
super({ organizationRepository: organizationsRepository });
10+
}
11+
}
12+

apps/api/v2/src/modules/auth/guards/organizations/is-admin-api-enabled.guard.ts

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ export class IsAdminAPIEnabledGuard implements CanActivate {
5454
}
5555
}
5656

57-
const org = await this.organizationsRepository.findById(Number(organizationId));
57+
const org = await this.organizationsRepository.findById({ id: Number(organizationId) });
5858

5959
if (org?.isOrganization && !org?.isPlatform) {
6060
const adminAPIAccessIsEnabledInOrg = await this.organizationsRepository.fetchOrgAdminApiStatus(

apps/api/v2/src/modules/auth/guards/organizations/is-org.guard.ts

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -55,7 +55,7 @@ export class IsOrgGuard implements CanActivate {
5555
}
5656
}
5757

58-
const org = await this.organizationsRepository.findById(Number(organizationId));
58+
const org = await this.organizationsRepository.findById({ id: Number(organizationId) });
5959

6060
if (org?.isOrganization) {
6161
canAccess = true;

apps/api/v2/src/modules/auth/guards/organizations/is-webhook-in-org.guard.ts

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ export class IsWebhookInOrg implements CanActivate {
4343
}
4444
}
4545

46-
const org = await this.organizationsRepository.findById(Number(organizationId));
46+
const org = await this.organizationsRepository.findById({ id: Number(organizationId) });
4747

4848
if (org?.isOrganization) {
4949
const isWebhookInOrg = await this.organizationsWebhooksRepository.findWebhook(

apps/api/v2/src/modules/organizations/index/organizations.repository.ts

Lines changed: 4 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -4,23 +4,17 @@ import { PrismaWriteService } from "@/modules/prisma/prisma-write.service";
44
import { StripeService } from "@/modules/stripe/stripe.service";
55
import { Injectable } from "@nestjs/common";
66

7+
import { OrganizationRepository } from "@calcom/platform-libraries/organizations";
78
import { Prisma } from "@calcom/prisma/client";
89

910
@Injectable()
10-
export class OrganizationsRepository {
11+
export class OrganizationsRepository extends OrganizationRepository {
1112
constructor(
1213
private readonly dbRead: PrismaReadService,
1314
private readonly dbWrite: PrismaWriteService,
1415
private readonly stripeService: StripeService
15-
) {}
16-
17-
async findById(organizationId: number) {
18-
return this.dbRead.prisma.team.findUnique({
19-
where: {
20-
id: organizationId,
21-
isOrganization: true,
22-
},
23-
});
16+
) {
17+
super({ prismaClient: dbWrite.prisma });
2418
}
2519

2620
async findByIds(organizationIds: number[]) {

apps/api/v2/src/modules/organizations/index/organizations.service.ts

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ export class OrganizationsService {
66
constructor(private readonly organizationsRepository: OrganizationsRepository) {}
77

88
async isPlatform(organizationId: number) {
9-
const organization = await this.organizationsRepository.findById(organizationId);
9+
const organization = await this.organizationsRepository.findById({ id: organizationId });
1010
return organization?.isPlatform;
1111
}
1212
}

apps/api/v2/src/modules/organizations/organizations.module.ts

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import { ZoomVideoService } from "@/modules/conferencing/services/zoom-video.ser
1313
import { CredentialsRepository } from "@/modules/credentials/credentials.repository";
1414
import { EmailModule } from "@/modules/email/email.module";
1515
import { EmailService } from "@/modules/email/email.service";
16+
import { OrganizationMembershipService } from "@/lib/services/organization-membership.service";
1617
import { MembershipsModule } from "@/modules/memberships/memberships.module";
1718
import { OAuthClientRepository } from "@/modules/oauth-clients/oauth-client.repository";
1819
import { UserOOORepository } from "@/modules/ooo/repositories/ooo.repository";
@@ -108,6 +109,7 @@ import { Module } from "@nestjs/common";
108109
],
109110
providers: [
110111
OrganizationsRepository,
112+
OrganizationMembershipService,
111113
OrganizationsTeamsRepository,
112114
OrganizationsService,
113115
OrganizationsTeamsService,
@@ -200,4 +202,4 @@ import { Module } from "@nestjs/common";
200202
OrganizationsEventTypesPrivateLinksController,
201203
],
202204
})
203-
export class OrganizationsModule {}
205+
export class OrganizationsModule { }

apps/api/v2/src/modules/organizations/organizations/services/managed-organizations.service.ts

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -103,12 +103,12 @@ export class ManagedOrganizationsService {
103103
}
104104

105105
private async isManagerOrganizationPlatform(managerOrganizationId: number) {
106-
const organization = await this.organizationsRepository.findById(managerOrganizationId);
106+
const organization = await this.organizationsRepository.findById({ id: managerOrganizationId });
107107
return !!organization?.isPlatform;
108108
}
109109

110110
async getManagedOrganization(managedOrganizationId: number) {
111-
const organization = await this.organizationsRepository.findById(managedOrganizationId);
111+
const organization = await this.organizationsRepository.findById({ id: managedOrganizationId });
112112
if (!organization) {
113113
throw new NotFoundException(`Managed organization with id=${managedOrganizationId} does not exist.`);
114114
}

apps/api/v2/src/modules/organizations/teams/memberships/e2e/organizations-teams-memberships.controller.e2e-spec.ts

Lines changed: 203 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -418,6 +418,209 @@ describe("Organizations Teams Memberships Endpoints", () => {
418418
.expect(404);
419419
});
420420

421+
// Auto-accept tests
422+
describe("auto-accept based on email domain", () => {
423+
let orgWithAutoAccept: Team;
424+
let subteamWithAutoAccept: Team;
425+
let userWithMatchingEmail: User;
426+
let userWithUppercaseEmail: User;
427+
let userWithMatchingEmailForOverride: User;
428+
let userWithNonMatchingEmail: User;
429+
430+
beforeAll(async () => {
431+
// Create org with auto-accept settings
432+
orgWithAutoAccept = await organizationsRepositoryFixture.create({
433+
name: `auto-accept-org-${randomString()}`,
434+
isOrganization: true,
435+
});
436+
437+
// Update organizationSettings with orgAutoAcceptEmail
438+
await organizationsRepositoryFixture.updateSettings(orgWithAutoAccept.id, {
439+
orgAutoAcceptEmail: "acme.com",
440+
isOrganizationVerified: true,
441+
isOrganizationConfigured: true,
442+
});
443+
444+
// Create subteam
445+
subteamWithAutoAccept = await teamsRepositoryFixture.create({
446+
name: `auto-accept-subteam-${randomString()}`,
447+
isOrganization: false,
448+
parent: { connect: { id: orgWithAutoAccept.id } },
449+
});
450+
451+
// Create event type with assignAllTeamMembers
452+
await eventTypesRepositoryFixture.createTeamEventType({
453+
schedulingType: "COLLECTIVE",
454+
team: { connect: { id: subteamWithAutoAccept.id } },
455+
title: "Auto Accept Event Type",
456+
slug: "auto-accept-event-type",
457+
length: 30,
458+
assignAllTeamMembers: true,
459+
bookingFields: [],
460+
locations: [],
461+
});
462+
463+
// Create users
464+
userWithMatchingEmail = await userRepositoryFixture.create({
465+
email: `alice@acme.com`,
466+
username: `alice-${randomString()}`,
467+
});
468+
469+
userWithUppercaseEmail = await userRepositoryFixture.create({
470+
email: `bob@ACME.COM`,
471+
username: `bob-${randomString()}`,
472+
});
473+
474+
userWithMatchingEmailForOverride = await userRepositoryFixture.create({
475+
email: `david@acme.com`,
476+
username: `david-${randomString()}`,
477+
});
478+
479+
userWithNonMatchingEmail = await userRepositoryFixture.create({
480+
email: `charlie@external.com`,
481+
username: `charlie-${randomString()}`,
482+
});
483+
484+
// Add users to org
485+
await membershipsRepositoryFixture.create({
486+
role: "MEMBER",
487+
accepted: true,
488+
user: { connect: { id: userWithMatchingEmail.id } },
489+
team: { connect: { id: orgWithAutoAccept.id } },
490+
});
491+
492+
await membershipsRepositoryFixture.create({
493+
role: "MEMBER",
494+
accepted: true,
495+
user: { connect: { id: userWithUppercaseEmail.id } },
496+
team: { connect: { id: orgWithAutoAccept.id } },
497+
});
498+
499+
await membershipsRepositoryFixture.create({
500+
role: "MEMBER",
501+
accepted: true,
502+
user: { connect: { id: userWithMatchingEmailForOverride.id } },
503+
team: { connect: { id: orgWithAutoAccept.id } },
504+
});
505+
506+
await membershipsRepositoryFixture.create({
507+
role: "MEMBER",
508+
accepted: true,
509+
user: { connect: { id: userWithNonMatchingEmail.id } },
510+
team: { connect: { id: orgWithAutoAccept.id } },
511+
});
512+
513+
// Create profiles for users
514+
await profileRepositoryFixture.create({
515+
uid: `usr-${userWithMatchingEmail.id}`,
516+
username: userWithMatchingEmail.username || `user-${userWithMatchingEmail.id}`,
517+
organization: { connect: { id: orgWithAutoAccept.id } },
518+
user: { connect: { id: userWithMatchingEmail.id } },
519+
});
520+
521+
await profileRepositoryFixture.create({
522+
uid: `usr-${userWithUppercaseEmail.id}`,
523+
username: userWithUppercaseEmail.username || `user-${userWithUppercaseEmail.id}`,
524+
organization: { connect: { id: orgWithAutoAccept.id } },
525+
user: { connect: { id: userWithUppercaseEmail.id } },
526+
});
527+
528+
await profileRepositoryFixture.create({
529+
uid: `usr-${userWithMatchingEmailForOverride.id}`,
530+
username: userWithMatchingEmailForOverride.username || `user-${userWithMatchingEmailForOverride.id}`,
531+
organization: { connect: { id: orgWithAutoAccept.id } },
532+
user: { connect: { id: userWithMatchingEmailForOverride.id } },
533+
});
534+
535+
await profileRepositoryFixture.create({
536+
uid: `usr-${userWithNonMatchingEmail.id}`,
537+
username: userWithNonMatchingEmail.username || `user-${userWithNonMatchingEmail.id}`,
538+
organization: { connect: { id: orgWithAutoAccept.id } },
539+
user: { connect: { id: userWithNonMatchingEmail.id } },
540+
});
541+
542+
// Make user an admin of the org for API access
543+
await membershipsRepositoryFixture.create({
544+
role: "ADMIN",
545+
accepted: true,
546+
user: { connect: { id: user.id } },
547+
team: { connect: { id: orgWithAutoAccept.id } },
548+
});
549+
});
550+
551+
it("should auto-accept when email matches orgAutoAcceptEmail", async () => {
552+
const response = await request(app.getHttpServer())
553+
.post(`/v2/organizations/${orgWithAutoAccept.id}/teams/${subteamWithAutoAccept.id}/memberships`)
554+
.send({
555+
userId: userWithMatchingEmail.id,
556+
role: "MEMBER",
557+
} satisfies CreateOrgTeamMembershipDto)
558+
.expect(201);
559+
560+
const responseBody: ApiSuccessResponse<TeamMembershipOutput> = response.body;
561+
expect(responseBody.data.accepted).toBe(true);
562+
563+
// Verify EventTypes assignment
564+
const eventTypes = await eventTypesRepositoryFixture.getAllTeamEventTypes(
565+
subteamWithAutoAccept.id
566+
);
567+
const eventTypeWithAssignAll = eventTypes.find((et) => et.assignAllTeamMembers);
568+
expect(eventTypeWithAssignAll).toBeTruthy();
569+
const userIsHost = eventTypeWithAssignAll?.hosts.some((h) => h.userId === userWithMatchingEmail.id);
570+
expect(userIsHost).toBe(true);
571+
});
572+
573+
it("should handle case-insensitive email domain matching", async () => {
574+
// User with email="bob@ACME.COM" should match orgAutoAcceptEmail="acme.com"
575+
const response = await request(app.getHttpServer())
576+
.post(`/v2/organizations/${orgWithAutoAccept.id}/teams/${subteamWithAutoAccept.id}/memberships`)
577+
.send({
578+
userId: userWithUppercaseEmail.id,
579+
role: "MEMBER",
580+
} satisfies CreateOrgTeamMembershipDto)
581+
.expect(201);
582+
583+
const responseBody: ApiSuccessResponse<TeamMembershipOutput> = response.body;
584+
expect(responseBody.data.accepted).toBe(true);
585+
});
586+
587+
it("should ALWAYS auto-accept when email matches, even if accepted:false", async () => {
588+
const response = await request(app.getHttpServer())
589+
.post(`/v2/organizations/${orgWithAutoAccept.id}/teams/${subteamWithAutoAccept.id}/memberships`)
590+
.send({
591+
userId: userWithMatchingEmailForOverride.id,
592+
role: "MEMBER",
593+
accepted: false,
594+
} satisfies CreateOrgTeamMembershipDto)
595+
.expect(201);
596+
597+
const responseBody: ApiSuccessResponse<TeamMembershipOutput> = response.body;
598+
// Should override to true because email matches
599+
expect(responseBody.data.accepted).toBe(true);
600+
});
601+
602+
it("should NOT auto-accept when email does not match orgAutoAcceptEmail", async () => {
603+
const response = await request(app.getHttpServer())
604+
.post(`/v2/organizations/${orgWithAutoAccept.id}/teams/${subteamWithAutoAccept.id}/memberships`)
605+
.send({
606+
userId: userWithNonMatchingEmail.id,
607+
role: "MEMBER",
608+
} satisfies CreateOrgTeamMembershipDto)
609+
.expect(201);
610+
611+
const responseBody: ApiSuccessResponse<TeamMembershipOutput> = response.body;
612+
expect(responseBody.data.accepted).toBe(false);
613+
});
614+
615+
afterAll(async () => {
616+
await userRepositoryFixture.deleteByEmail(userWithMatchingEmail.email);
617+
await userRepositoryFixture.deleteByEmail(userWithUppercaseEmail.email);
618+
await userRepositoryFixture.deleteByEmail(userWithMatchingEmailForOverride.email);
619+
await userRepositoryFixture.deleteByEmail(userWithNonMatchingEmail.email);
620+
await organizationsRepositoryFixture.delete(orgWithAutoAccept.id);
621+
});
622+
});
623+
421624
afterAll(async () => {
422625
await userRepositoryFixture.deleteByEmail(user.email);
423626
await userRepositoryFixture.deleteByEmail(userToInviteViaApi.email);

apps/api/v2/src/modules/organizations/teams/memberships/organizations-teams-memberships.controller.ts

Lines changed: 22 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import { IsAdminAPIEnabledGuard } from "@/modules/auth/guards/organizations/is-a
1212
import { IsOrgGuard } from "@/modules/auth/guards/organizations/is-org.guard";
1313
import { RolesGuard } from "@/modules/auth/guards/roles/roles.guard";
1414
import { IsTeamInOrg } from "@/modules/auth/guards/teams/is-team-in-org.guard";
15+
import { OrganizationMembershipService } from "@/lib/services/organization-membership.service";
1516
import { OrganizationsRepository } from "@/modules/organizations/index/organizations.repository";
1617
import { CreateOrgTeamMembershipDto } from "@/modules/organizations/teams/memberships/inputs/create-organization-team-membership.input";
1718
import { UpdateOrgTeamMembershipDto } from "@/modules/organizations/teams/memberships/inputs/update-organization-team-membership.input";
@@ -58,8 +59,9 @@ export class OrganizationsTeamsMembershipsController {
5859

5960
constructor(
6061
private organizationsTeamsMembershipsService: OrganizationsTeamsMembershipsService,
61-
private readonly organizationsRepository: OrganizationsRepository
62-
) {}
62+
private readonly organizationsRepository: OrganizationsRepository,
63+
private readonly orgMembershipService: OrganizationMembershipService
64+
) { }
6365

6466
@Get("/")
6567
@ApiOperation({ summary: "Get all memberships" })
@@ -168,6 +170,9 @@ export class OrganizationsTeamsMembershipsController {
168170
};
169171
}
170172

173+
174+
// TODO: Refactor to use inviteMembersWithNoInviterPermissionCheck when it is moved to a Service
175+
// See: packages/trpc/server/routers/viewer/teams/inviteMember/inviteMember.handler.ts
171176
@Roles("TEAM_ADMIN")
172177
@PlatformPlan("ESSENTIALS")
173178
@Post("/")
@@ -184,7 +189,21 @@ export class OrganizationsTeamsMembershipsController {
184189
throw new UnprocessableEntityException("User is not part of the Organization");
185190
}
186191

187-
const membership = await this.organizationsTeamsMembershipsService.createOrgTeamMembership(teamId, data);
192+
const shouldAutoAccept = await this.orgMembershipService.shouldAutoAccept({
193+
organizationId: orgId,
194+
userEmail: user.email,
195+
});
196+
197+
// ALWAYS override when email matches - prevents pending memberships
198+
// Remember organizations expect added team member to automatically start receiving bookings for the team event
199+
const acceptedStatus = shouldAutoAccept ? true : (data.accepted ?? false);
200+
201+
const membershipData = { ...data, accepted: acceptedStatus };
202+
const membership = await this.organizationsTeamsMembershipsService.createOrgTeamMembership(
203+
teamId,
204+
membershipData
205+
);
206+
188207
if (membership.accepted) {
189208
try {
190209
await updateNewTeamMemberEventTypes(user.id, teamId);

0 commit comments

Comments
 (0)