4710 implement google calendar incremental sync (#4822)

Closes #4710
This commit is contained in:
bosiraphael
2024-04-10 15:53:14 +02:00
committed by GitHub
parent f1cc1c60e0
commit e7d146363c
20 changed files with 333 additions and 152 deletions

View File

@ -10,9 +10,9 @@ import { SaveOrUpdateConnectedAccountInput } from 'src/engine/core-modules/auth/
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service';
import { import {
GoogleCalendarFullSyncJob, GoogleCalendarSyncJob,
GoogleCalendarFullSyncJobData, GoogleCalendarSyncJobData,
} from 'src/modules/calendar/jobs/google-calendar-full-sync.job'; } from 'src/modules/calendar/jobs/google-calendar-sync.job';
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
import { import {
FeatureFlagEntity, FeatureFlagEntity,
@ -135,10 +135,7 @@ export class GoogleAPIsService {
this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') && this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') &&
IsCalendarEnabled IsCalendarEnabled
) { ) {
await this.enqueueGoogleCalendarFullSyncJob( await this.enqueueGoogleCalendarSyncJob(workspaceId, connectedAccountId);
workspaceId,
connectedAccountId,
);
} }
return; return;
@ -170,10 +167,7 @@ export class GoogleAPIsService {
} }
if (this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED')) { if (this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED')) {
await this.enqueueGoogleCalendarFullSyncJob( await this.enqueueGoogleCalendarSyncJob(workspaceId, connectedAccountId);
workspaceId,
connectedAccountId,
);
} }
return; return;
@ -192,12 +186,12 @@ export class GoogleAPIsService {
); );
} }
async enqueueGoogleCalendarFullSyncJob( async enqueueGoogleCalendarSyncJob(
workspaceId: string, workspaceId: string,
connectedAccountId: string, connectedAccountId: string,
) { ) {
await this.calendarQueueService.add<GoogleCalendarFullSyncJobData>( await this.calendarQueueService.add<GoogleCalendarSyncJobData>(
GoogleCalendarFullSyncJob.name, GoogleCalendarSyncJob.name,
{ {
workspaceId, workspaceId,
connectedAccountId, connectedAccountId,

View File

@ -28,12 +28,11 @@ import { StripeModule } from 'src/engine/core-modules/billing/stripe/stripe.modu
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity';
import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity';
import { GoogleCalendarFullSyncJob } from 'src/modules/calendar/jobs/google-calendar-full-sync.job'; import { GoogleCalendarSyncJob } from 'src/modules/calendar/jobs/google-calendar-sync.job';
import { CalendarEventCleanerModule } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module'; import { CalendarEventCleanerModule } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module';
import { RecordPositionBackfillJob } from 'src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job'; import { RecordPositionBackfillJob } from 'src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job';
import { RecordPositionBackfillModule } from 'src/engine/api/graphql/workspace-query-runner/services/record-position-backfill-module'; import { RecordPositionBackfillModule } from 'src/engine/api/graphql/workspace-query-runner/services/record-position-backfill-module';
import { DeleteConnectedAccountAssociatedCalendarDataJob } from 'src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job'; import { GoogleCalendarSyncModule } from 'src/modules/calendar/services/google-calendar-sync.module';
import { GoogleCalendarFullSyncModule } from 'src/modules/calendar/services/google-calendar-full-sync.module';
import { GoogleAPIRefreshAccessTokenModule } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module'; import { GoogleAPIRefreshAccessTokenModule } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module';
import { MessageParticipantModule } from 'src/modules/messaging/services/message-participant/message-participant.module'; import { MessageParticipantModule } from 'src/modules/messaging/services/message-participant/message-participant.module';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
@ -45,12 +44,14 @@ import { EventObjectMetadata } from 'src/modules/event/standard-objects/event.ob
import { HandleWorkspaceMemberDeletedJob } from 'src/engine/core-modules/workspace/handle-workspace-member-deleted.job'; import { HandleWorkspaceMemberDeletedJob } from 'src/engine/core-modules/workspace/handle-workspace-member-deleted.job';
import { GmailFullSynV2Module } from 'src/modules/messaging/services/gmail-full-sync-v2/gmail-full-sync.v2.module'; import { GmailFullSynV2Module } from 'src/modules/messaging/services/gmail-full-sync-v2/gmail-full-sync.v2.module';
import { GmailFetchMessageContentFromCacheModule } from 'src/modules/messaging/services/gmail-fetch-message-content-from-cache/gmail-fetch-message-content-from-cache.module'; import { GmailFetchMessageContentFromCacheModule } from 'src/modules/messaging/services/gmail-fetch-message-content-from-cache/gmail-fetch-message-content-from-cache.module';
import { FetchAllMessagesFromCacheCronJob } from 'src/modules/messaging/jobs/crons/fetch-all-messages-from-cache.cron.job';
import { GmailFullSyncV2Job } from 'src/modules/messaging/jobs/gmail-full-sync-v2.job'; import { GmailFullSyncV2Job } from 'src/modules/messaging/jobs/gmail-full-sync-v2.job';
import { GmailPartialSyncV2Job } from 'src/modules/messaging/jobs/gmail-partial-sync-v2.job'; import { GmailPartialSyncV2Job } from 'src/modules/messaging/jobs/gmail-partial-sync-v2.job';
import { GmailPartialSyncV2Module } from 'src/modules/messaging/services/gmail-partial-sync-v2/gmail-partial-sync-v2.module'; import { GmailPartialSyncV2Module } from 'src/modules/messaging/services/gmail-partial-sync-v2/gmail-partial-sync-v2.module';
import { GoogleCalendarSyncCronJob } from 'src/modules/calendar/jobs/crons/google-calendar-sync.cron.job';
import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module'; import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module';
import { UnmatchParticipantJob } from 'src/modules/connected-account/jobs/unmatch-participant.job'; import { UnmatchParticipantJob } from 'src/modules/connected-account/jobs/unmatch-participant.job';
import { DeleteConnectedAccountAssociatedCalendarDataJob } from 'src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job';
import { FetchAllMessagesFromCacheCronJob } from 'src/modules/messaging/jobs/crons/fetch-all-messages-from-cache.cron.job';
@Module({ @Module({
imports: [ imports: [
@ -60,7 +61,7 @@ import { UnmatchParticipantJob } from 'src/modules/connected-account/jobs/unmatc
DataSeedDemoWorkspaceModule, DataSeedDemoWorkspaceModule,
EnvironmentModule, EnvironmentModule,
HttpModule, HttpModule,
GoogleCalendarFullSyncModule, GoogleCalendarSyncModule,
ObjectMetadataModule, ObjectMetadataModule,
StripeModule, StripeModule,
ThreadCleanerModule, ThreadCleanerModule,
@ -86,8 +87,8 @@ import { UnmatchParticipantJob } from 'src/modules/connected-account/jobs/unmatc
], ],
providers: [ providers: [
{ {
provide: GoogleCalendarFullSyncJob.name, provide: GoogleCalendarSyncJob.name,
useClass: GoogleCalendarFullSyncJob, useClass: GoogleCalendarSyncJob,
}, },
{ {
provide: CallWebhookJobsJob.name, provide: CallWebhookJobsJob.name,
@ -159,6 +160,10 @@ import { UnmatchParticipantJob } from 'src/modules/connected-account/jobs/unmatc
provide: GmailPartialSyncV2Job.name, provide: GmailPartialSyncV2Job.name,
useClass: GmailPartialSyncV2Job, useClass: GmailPartialSyncV2Job,
}, },
{
provide: GoogleCalendarSyncCronJob.name,
useClass: GoogleCalendarSyncCronJob,
},
], ],
}) })
export class JobsModule { export class JobsModule {

View File

@ -6,24 +6,24 @@ import { MessageQueue } from 'src/engine/integrations/message-queue/message-queu
import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service';
import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository';
import { import {
GoogleCalendarFullSyncJobData, GoogleCalendarSyncJobData,
GoogleCalendarFullSyncJob, GoogleCalendarSyncJob,
} from 'src/modules/calendar/jobs/google-calendar-full-sync.job'; } from 'src/modules/calendar/jobs/google-calendar-sync.job';
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata'; import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata';
import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository';
import { CalendarChannelObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel.object-metadata'; import { CalendarChannelObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel.object-metadata';
interface GoogleCalendarFullSyncOptions { interface GoogleCalendarSyncOptions {
workspaceId: string; workspaceId: string;
} }
@Command({ @Command({
name: 'workspace:google-calendar-full-sync', name: 'workspace:google-calendar-sync',
description: description:
'Start google calendar full-sync for all workspaceMembers in a workspace.', 'Start google calendar sync for all workspaceMembers in a workspace.',
}) })
export class GoogleCalendarFullSyncCommand extends CommandRunner { export class GoogleCalendarSyncCommand extends CommandRunner {
constructor( constructor(
@Inject(MessageQueue.messagingQueue) @Inject(MessageQueue.messagingQueue)
private readonly messageQueueService: MessageQueueService, private readonly messageQueueService: MessageQueueService,
@ -37,7 +37,7 @@ export class GoogleCalendarFullSyncCommand extends CommandRunner {
async run( async run(
_passedParam: string[], _passedParam: string[],
options: GoogleCalendarFullSyncOptions, options: GoogleCalendarSyncOptions,
): Promise<void> { ): Promise<void> {
await this.fetchWorkspaceCalendars(options.workspaceId); await this.fetchWorkspaceCalendars(options.workspaceId);
@ -68,8 +68,8 @@ export class GoogleCalendarFullSyncCommand extends CommandRunner {
continue; continue;
} }
await this.messageQueueService.add<GoogleCalendarFullSyncJobData>( await this.messageQueueService.add<GoogleCalendarSyncJobData>(
GoogleCalendarFullSyncJob.name, GoogleCalendarSyncJob.name,
{ {
workspaceId, workspaceId,
connectedAccountId: connectedAccount.id, connectedAccountId: connectedAccount.id,

View File

@ -0,0 +1,31 @@
import { Inject } from '@nestjs/common';
import { Command, CommandRunner } from 'nest-commander';
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service';
import { GoogleCalendarSyncCronJob } from 'src/modules/calendar/jobs/crons/google-calendar-sync.cron.job';
import { googleCalendarSyncCronPattern } from 'src/modules/calendar/jobs/crons/pattern/google-calendar-sync.cron.pattern';
@Command({
name: 'cron:calendar:google-calendar-sync',
description: 'Starts a cron job to sync google calendar for all workspaces.',
})
export class StartGoogleCalendarSyncCronJobCommand extends CommandRunner {
constructor(
@Inject(MessageQueue.cronQueue)
private readonly messageQueueService: MessageQueueService,
) {
super();
}
async run(): Promise<void> {
await this.messageQueueService.addCron<undefined>(
GoogleCalendarSyncCronJob.name,
undefined,
{
repeat: { pattern: googleCalendarSyncCronPattern },
},
);
}
}

View File

@ -1,13 +1,14 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { GoogleCalendarFullSyncCommand } from 'src/modules/calendar/commands/google-calendar-full-sync.command'; import { GoogleCalendarSyncCommand } from 'src/modules/calendar/commands/google-calendar-sync.command';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata'; import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata';
import { StartGoogleCalendarSyncCronJobCommand } from 'src/modules/calendar/commands/start-google-calendar-sync.cron.command';
@Module({ @Module({
imports: [ imports: [
ObjectMetadataRepositoryModule.forFeature([ConnectedAccountObjectMetadata]), ObjectMetadataRepositoryModule.forFeature([ConnectedAccountObjectMetadata]),
], ],
providers: [GoogleCalendarFullSyncCommand], providers: [GoogleCalendarSyncCommand, StartGoogleCalendarSyncCronJobCommand],
}) })
export class WorkspaceCalendarSyncCommandsModule {} export class WorkspaceCalendarSyncCommandsModule {}

View File

@ -0,0 +1,82 @@
import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm';
import { Repository, In } from 'typeorm';
import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface';
import {
FeatureFlagEntity,
FeatureFlagKeys,
} from 'src/engine/core-modules/feature-flag/feature-flag.entity';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity';
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository';
import { CalendarChannelObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel.object-metadata';
import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync.service';
@Injectable()
export class GoogleCalendarSyncCronJob implements MessageQueueJob<undefined> {
constructor(
@InjectRepository(Workspace, 'core')
private readonly workspaceRepository: Repository<Workspace>,
@InjectRepository(DataSourceEntity, 'metadata')
private readonly dataSourceRepository: Repository<DataSourceEntity>,
@InjectObjectMetadataRepository(CalendarChannelObjectMetadata)
private readonly calendarChannelRepository: CalendarChannelRepository,
@InjectRepository(FeatureFlagEntity, 'core')
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
private readonly googleCalendarSyncService: GoogleCalendarSyncService,
) {}
async handle(): Promise<void> {
const workspaceIds = (
await this.workspaceRepository.find({
where: {
subscriptionStatus: 'active',
},
select: ['id'],
})
).map((workspace) => workspace.id);
const workspacesWithFeatureFlagActive =
await this.featureFlagRepository.find({
where: {
workspaceId: In(workspaceIds),
key: FeatureFlagKeys.IsCalendarEnabled,
value: true,
},
});
const dataSources = await this.dataSourceRepository.find({
where: {
workspaceId: In(
workspacesWithFeatureFlagActive.map((w) => w.workspaceId),
),
},
});
const workspaceIdsWithDataSources = new Set(
dataSources.map((dataSource) => dataSource.workspaceId),
);
for (const workspaceId of workspaceIdsWithDataSources) {
await this.startWorkspaceGoogleCalendarSync(workspaceId);
}
}
private async startWorkspaceGoogleCalendarSync(
workspaceId: string,
): Promise<void> {
const calendarChannels =
await this.calendarChannelRepository.getAll(workspaceId);
for (const calendarChannel of calendarChannels) {
await this.googleCalendarSyncService.startGoogleCalendarSync(
workspaceId,
calendarChannel.connectedAccountId,
);
}
}
}

View File

@ -0,0 +1 @@
export const googleCalendarSyncCronPattern = '*/5 * * * *';

View File

@ -3,32 +3,27 @@ import { Injectable, Logger } from '@nestjs/common';
import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface';
import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service'; import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service';
import { GoogleCalendarFullSyncService } from 'src/modules/calendar/services/google-calendar-full-sync.service'; import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync.service';
export type GoogleCalendarFullSyncJobData = { export type GoogleCalendarSyncJobData = {
workspaceId: string; workspaceId: string;
connectedAccountId: string; connectedAccountId: string;
nextPageToken?: string;
}; };
@Injectable() @Injectable()
export class GoogleCalendarFullSyncJob export class GoogleCalendarSyncJob
implements MessageQueueJob<GoogleCalendarFullSyncJobData> implements MessageQueueJob<GoogleCalendarSyncJobData>
{ {
private readonly logger = new Logger(GoogleCalendarFullSyncJob.name); private readonly logger = new Logger(GoogleCalendarSyncJob.name);
constructor( constructor(
private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService, private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService,
private readonly googleCalendarFullSyncService: GoogleCalendarFullSyncService, private readonly googleCalendarSyncService: GoogleCalendarSyncService,
) {} ) {}
async handle(data: GoogleCalendarFullSyncJobData): Promise<void> { async handle(data: GoogleCalendarSyncJobData): Promise<void> {
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${ `google calendar sync for workspace ${data.workspaceId} and account ${data.connectedAccountId}`,
data.workspaceId
} and account ${data.connectedAccountId} ${
data.nextPageToken ? `and ${data.nextPageToken} pageToken` : ''
}`,
); );
try { try {
await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken( await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken(
@ -44,10 +39,9 @@ export class GoogleCalendarFullSyncJob
return; return;
} }
await this.googleCalendarFullSyncService.startGoogleCalendarFullSync( await this.googleCalendarSyncService.startGoogleCalendarSync(
data.workspaceId, data.workspaceId,
data.connectedAccountId, data.connectedAccountId,
data.nextPageToken,
); );
} }
} }

View File

@ -12,6 +12,21 @@ export class CalendarChannelRepository {
private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly workspaceDataSourceService: WorkspaceDataSourceService,
) {} ) {}
public async getAll(
workspaceId: string,
transactionManager?: EntityManager,
): Promise<ObjectRecord<CalendarChannelObjectMetadata>[]> {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
return await this.workspaceDataSourceService.executeRawQuery(
`SELECT * FROM ${dataSourceSchema}."calendarChannel"`,
[],
workspaceId,
transactionManager,
);
}
public async getByConnectedAccountId( public async getByConnectedAccountId(
connectedAccountId: string, connectedAccountId: string,
workspaceId: string, workspaceId: string,

View File

@ -160,14 +160,14 @@ export class CalendarEventParticipantRepository {
); );
} }
public async updateCalendarEventParticipants( public async updateCalendarEventParticipantsAndReturnNewOnes(
calendarEventParticipants: CalendarEventParticipant[], calendarEventParticipants: CalendarEventParticipant[],
iCalUIDCalendarEventIdMap: Map<string, string>, iCalUIDCalendarEventIdMap: Map<string, string>,
workspaceId: string, workspaceId: string,
transactionManager?: EntityManager, transactionManager?: EntityManager,
): Promise<void> { ): Promise<CalendarEventParticipant[]> {
if (calendarEventParticipants.length === 0) { if (calendarEventParticipants.length === 0) {
return; return [];
} }
const dataSourceSchema = const dataSourceSchema =
@ -189,6 +189,14 @@ export class CalendarEventParticipantRepository {
calendarEventParticipant.handle, calendarEventParticipant.handle,
); );
const newCalendarEventParticipants = differenceWith(
calendarEventParticipants,
existingCalendarEventParticipants,
(calendarEventParticipant, existingCalendarEventParticipant) =>
calendarEventParticipant.handle ===
existingCalendarEventParticipant.handle,
);
await this.deleteByIds( await this.deleteByIds(
calendarEventParticipantsToDelete.map( calendarEventParticipantsToDelete.map(
(calendarEventParticipant) => calendarEventParticipant.id, (calendarEventParticipant) => calendarEventParticipant.id,
@ -227,6 +235,8 @@ export class CalendarEventParticipantRepository {
workspaceId, workspaceId,
transactionManager, transactionManager,
); );
return newCalendarEventParticipants;
} }
public async getWithoutPersonIdAndWorkspaceMemberId( public async getWithoutPersonIdAndWorkspaceMemberId(

View File

@ -80,34 +80,36 @@ export class CalendarEventRepository {
} }
public async getICalUIDCalendarEventIdMap( public async getICalUIDCalendarEventIdMap(
iCalUIDs: string[], calendarEventIds: string[],
workspaceId: string, workspaceId: string,
transactionManager?: EntityManager, transactionManager?: EntityManager,
): Promise<Map<string, string>> { ): Promise<Map<string, string>> {
if (iCalUIDs.length === 0) { if (calendarEventIds.length === 0) {
return new Map(); return new Map();
} }
const dataSourceSchema = const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId); this.workspaceDataSourceService.getSchemaName(workspaceId);
const calendarEvents: { const calendarEvents:
id: string; | {
iCalUID: string; id: string;
}[] = await this.workspaceDataSourceService.executeRawQuery( iCalUID: string;
`SELECT id, "iCalUID" FROM ${dataSourceSchema}."calendarEvent" WHERE "iCalUID" = ANY($1)`, }[]
[iCalUIDs], | undefined = await this.workspaceDataSourceService.executeRawQuery(
`SELECT id, "iCalUID" FROM ${dataSourceSchema}."calendarEvent" WHERE "id" = ANY($1)`,
[calendarEventIds],
workspaceId, workspaceId,
transactionManager, transactionManager,
); );
const iCalUIDsCalendarEvnetIdsMap = new Map<string, string>(); const iCalUIDsCalendarEventIdsMap = new Map<string, string>();
calendarEvents.forEach((calendarEvent) => { calendarEvents?.forEach((calendarEvent) => {
iCalUIDsCalendarEvnetIdsMap.set(calendarEvent.iCalUID, calendarEvent.id); iCalUIDsCalendarEventIdsMap.set(calendarEvent.iCalUID, calendarEvent.id);
}); });
return iCalUIDsCalendarEvnetIdsMap; return iCalUIDsCalendarEventIdsMap;
} }
public async saveCalendarEvents( public async saveCalendarEvents(

View File

@ -4,8 +4,9 @@ import { TypeOrmModule } from '@nestjs/typeorm';
import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
import { CalendarEventCleanerModule } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module';
import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module'; import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module';
import { GoogleCalendarFullSyncService } from 'src/modules/calendar/services/google-calendar-full-sync.service'; import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync.service';
import { CalendarProvidersModule } from 'src/modules/calendar/services/providers/calendar-providers.module'; import { CalendarProvidersModule } from 'src/modules/calendar/services/providers/calendar-providers.module';
import { CalendarChannelEventAssociationObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.object-metadata'; import { CalendarChannelEventAssociationObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.object-metadata';
import { CalendarChannelObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel.object-metadata'; import { CalendarChannelObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel.object-metadata';
@ -32,8 +33,9 @@ import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/stan
CalendarEventParticipantModule, CalendarEventParticipantModule,
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
WorkspaceDataSourceModule, WorkspaceDataSourceModule,
CalendarEventCleanerModule,
], ],
providers: [GoogleCalendarFullSyncService], providers: [GoogleCalendarSyncService],
exports: [GoogleCalendarFullSyncService], exports: [GoogleCalendarSyncService],
}) })
export class GoogleCalendarFullSyncModule {} export class GoogleCalendarSyncModule {}

View File

@ -1,8 +1,9 @@
import { Inject, Injectable, Logger } from '@nestjs/common'; import { Injectable, Logger } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm'; import { InjectRepository } from '@nestjs/typeorm';
import { EventEmitter2 } from '@nestjs/event-emitter'; import { EventEmitter2 } from '@nestjs/event-emitter';
import { Repository } from 'typeorm'; import { Repository } from 'typeorm';
import { calendar_v3 as calendarV3 } from 'googleapis';
import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository';
import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository';
@ -14,12 +15,9 @@ import { GoogleCalendarClientProvider } from 'src/modules/calendar/services/prov
import { googleCalendarSearchFilterExcludeEmails } from 'src/modules/calendar/utils/google-calendar-search-filter.util'; import { googleCalendarSearchFilterExcludeEmails } from 'src/modules/calendar/utils/google-calendar-search-filter.util';
import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository'; import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository';
import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository';
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
import { CalendarEventRepository } from 'src/modules/calendar/repositories/calendar-event.repository'; import { CalendarEventRepository } from 'src/modules/calendar/repositories/calendar-event.repository';
import { formatGoogleCalendarEvent } from 'src/modules/calendar/utils/format-google-calendar-event.util'; import { formatGoogleCalendarEvent } from 'src/modules/calendar/utils/format-google-calendar-event.util';
import { GoogleCalendarFullSyncJobData } from 'src/modules/calendar/jobs/google-calendar-full-sync.job';
import { CalendarEventParticipantRepository } from 'src/modules/calendar/repositories/calendar-event-participant.repository'; import { CalendarEventParticipantRepository } from 'src/modules/calendar/repositories/calendar-event-participant.repository';
import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata'; import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata';
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
@ -28,16 +26,16 @@ import { CalendarChannelObjectMetadata } from 'src/modules/calendar/standard-obj
import { CalendarChannelEventAssociationObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.object-metadata'; import { CalendarChannelEventAssociationObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.object-metadata';
import { CalendarEventParticipantObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-event-participant.object-metadata'; import { CalendarEventParticipantObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-event-participant.object-metadata';
import { BlocklistObjectMetadata } from 'src/modules/connected-account/standard-objects/blocklist.object-metadata'; import { BlocklistObjectMetadata } from 'src/modules/connected-account/standard-objects/blocklist.object-metadata';
import { CalendarEventCleanerService } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service';
import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service'; import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service';
import { CalendarEventParticipant } from 'src/modules/calendar/types/calendar-event';
@Injectable() @Injectable()
export class GoogleCalendarFullSyncService { export class GoogleCalendarSyncService {
private readonly logger = new Logger(GoogleCalendarFullSyncService.name); private readonly logger = new Logger(GoogleCalendarSyncService.name);
constructor( constructor(
private readonly googleCalendarClientProvider: GoogleCalendarClientProvider, private readonly googleCalendarClientProvider: GoogleCalendarClientProvider,
@Inject(MessageQueue.calendarQueue)
private readonly messageQueueService: MessageQueueService,
@InjectObjectMetadataRepository(ConnectedAccountObjectMetadata) @InjectObjectMetadataRepository(ConnectedAccountObjectMetadata)
private readonly connectedAccountRepository: ConnectedAccountRepository, private readonly connectedAccountRepository: ConnectedAccountRepository,
@InjectObjectMetadataRepository(CalendarEventObjectMetadata) @InjectObjectMetadataRepository(CalendarEventObjectMetadata)
@ -56,13 +54,13 @@ export class GoogleCalendarFullSyncService {
private readonly featureFlagRepository: Repository<FeatureFlagEntity>, private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly workspaceDataSourceService: WorkspaceDataSourceService,
private readonly eventEmitter: EventEmitter2, private readonly eventEmitter: EventEmitter2,
private readonly calendarEventCleanerService: CalendarEventCleanerService,
private readonly calendarEventParticipantsService: CalendarEventParticipantService, private readonly calendarEventParticipantsService: CalendarEventParticipantService,
) {} ) {}
public async startGoogleCalendarFullSync( public async startGoogleCalendarSync(
workspaceId: string, workspaceId: string,
connectedAccountId: string, connectedAccountId: string,
pageToken?: string,
): Promise<void> { ): Promise<void> {
const connectedAccount = await this.connectedAccountRepository.getById( const connectedAccount = await this.connectedAccountRepository.getById(
connectedAccountId, connectedAccountId,
@ -78,7 +76,7 @@ export class GoogleCalendarFullSyncService {
if (!refreshToken) { if (!refreshToken) {
throw new Error( throw new Error(
`No refresh token found for connected account ${connectedAccountId} in workspace ${workspaceId} during full-sync`, `No refresh token found for connected account ${connectedAccountId} in workspace ${workspaceId} during sync`,
); );
} }
@ -88,6 +86,8 @@ export class GoogleCalendarFullSyncService {
workspaceId, workspaceId,
); );
const syncToken = calendarChannel?.syncCursor || undefined;
if (!calendarChannel) { if (!calendarChannel) {
return; return;
} }
@ -120,30 +120,47 @@ export class GoogleCalendarFullSyncService {
let startTime = Date.now(); let startTime = Date.now();
const googleCalendarEvents = await googleCalendarClient.events.list({ let nextSyncToken: string | null | undefined;
calendarId: 'primary', let nextPageToken: string | undefined;
maxResults: 500, const events: calendarV3.Schema$Event[] = [];
pageToken: pageToken,
q: googleCalendarSearchFilterExcludeEmails(blocklistedEmails), while (true) {
}); const googleCalendarEvents = await googleCalendarClient.events.list({
calendarId: 'primary',
maxResults: 500,
syncToken,
pageToken: nextPageToken,
showDeleted: true,
q: googleCalendarSearchFilterExcludeEmails(blocklistedEmails),
});
nextSyncToken = googleCalendarEvents.data.nextSyncToken;
nextPageToken = googleCalendarEvents.data.nextPageToken || undefined;
const { items } = googleCalendarEvents.data;
if (!items || items.length === 0) {
break;
}
events.push(...items);
if (!nextPageToken) {
break;
}
}
let endTime = Date.now(); let endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId} getting events list in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} getting events list in ${
endTime - startTime endTime - startTime
}ms.`, }ms.`,
); );
const {
items: events,
nextPageToken,
nextSyncToken,
} = googleCalendarEvents.data;
if (!events || events?.length === 0) { if (!events || events?.length === 0) {
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId} done with nothing to import.`, `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} done with nothing to import.`,
); );
return; return;
@ -163,17 +180,11 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: getting existing calendar channel event associations in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: getting existing calendar channel event associations in ${
endTime - startTime endTime - startTime
}ms.`, }ms.`,
); );
// TODO: In V2, we will also import deleted events by doing batch GET queries on the canceled events
// The canceled events start and end are not accessible in the list query
const formattedEvents = events
.filter((event) => event.status !== 'cancelled')
.map((event) => formatGoogleCalendarEvent(event));
// TODO: When we will be able to add unicity contraint on iCalUID, we will do a INSERT ON CONFLICT DO UPDATE // TODO: When we will be able to add unicity contraint on iCalUID, we will do a INSERT ON CONFLICT DO UPDATE
const existingEventExternalIds = const existingEventExternalIds =
@ -181,6 +192,22 @@ export class GoogleCalendarFullSyncService {
(association) => association.eventExternalId, (association) => association.eventExternalId,
); );
const existingEventsIds = existingCalendarChannelEventAssociations.map(
(association) => association.calendarEventId,
);
const iCalUIDCalendarEventIdMap =
await this.calendarEventRepository.getICalUIDCalendarEventIdMap(
existingEventsIds,
workspaceId,
);
const formattedEvents = events
.filter((event) => event.status !== 'cancelled')
.map((event) =>
formatGoogleCalendarEvent(event, iCalUIDCalendarEventIdMap),
);
const eventsToSave = formattedEvents.filter( const eventsToSave = formattedEvents.filter(
(event) => !existingEventExternalIds.includes(event.externalId), (event) => !existingEventExternalIds.includes(event.externalId),
); );
@ -189,6 +216,10 @@ export class GoogleCalendarFullSyncService {
existingEventExternalIds.includes(event.externalId), existingEventExternalIds.includes(event.externalId),
); );
const cancelledEventExternalIds = events
.filter((event) => event.status === 'cancelled')
.map((event) => event.id as string);
const calendarChannelEventAssociationsToSave = eventsToSave.map( const calendarChannelEventAssociationsToSave = eventsToSave.map(
(event) => ({ (event) => ({
calendarEventId: event.id, calendarEventId: event.id,
@ -205,11 +236,7 @@ export class GoogleCalendarFullSyncService {
(event) => event.participants, (event) => event.participants,
); );
const iCalUIDCalendarEventIdMap = let newCalendarEventParticipants: CalendarEventParticipant[] = [];
await this.calendarEventRepository.getICalUIDCalendarEventIdMap(
eventsToUpdate.map((event) => event.iCalUID),
workspaceId,
);
if (events.length > 0) { if (events.length > 0) {
const dataSourceMetadata = const dataSourceMetadata =
@ -230,9 +257,9 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: saving events in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: saving ${
endTime - startTime eventsToSave.length
}ms.`, } events in ${endTime - startTime}ms.`,
); );
startTime = Date.now(); startTime = Date.now();
@ -246,9 +273,9 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: updating events in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: updating ${
endTime - startTime eventsToUpdate.length
}ms.`, } events in ${endTime - startTime}ms.`,
); );
startTime = Date.now(); startTime = Date.now();
@ -262,7 +289,27 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: saving calendar channel event associations in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: saving calendar channel event associations in ${
endTime - startTime
}ms.`,
);
startTime = Date.now();
newCalendarEventParticipants =
await this.calendarEventParticipantsRepository.updateCalendarEventParticipantsAndReturnNewOnes(
participantsToUpdate,
iCalUIDCalendarEventIdMap,
workspaceId,
transactionManager,
);
endTime = Date.now();
participantsToSave.push(...newCalendarEventParticipants);
this.logger.log(
`google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: updating participants in ${
endTime - startTime endTime - startTime
}ms.`, }ms.`,
); );
@ -278,16 +325,16 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: saving participants in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: saving participants in ${
endTime - startTime endTime - startTime
}ms.`, }ms.`,
); );
startTime = Date.now(); startTime = Date.now();
await this.calendarEventParticipantsRepository.updateCalendarEventParticipants( await this.calendarChannelEventAssociationRepository.deleteByEventExternalIdsAndCalendarChannelId(
participantsToUpdate, cancelledEventExternalIds,
iCalUIDCalendarEventIdMap, calendarChannelId,
workspaceId, workspaceId,
transactionManager, transactionManager,
); );
@ -295,35 +342,47 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: updating participants in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: deleting calendar channel event associations in ${
endTime - startTime endTime - startTime
}ms.`, }ms.`,
); );
}); });
if (calendarChannel.isContactAutoCreationEnabled) { startTime = Date.now();
const contactsToCreate = participantsToSave;
await this.calendarEventCleanerService.cleanWorkspaceCalendarEvents(
workspaceId,
);
endTime = Date.now();
this.logger.log(
`google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: cleaning calendar events in ${
endTime - startTime
}ms.`,
);
if (calendarChannel.isContactAutoCreationEnabled) {
this.eventEmitter.emit(`createContacts`, { this.eventEmitter.emit(`createContacts`, {
workspaceId, workspaceId,
connectedAccountHandle: connectedAccount.handle, connectedAccountHandle: connectedAccount.handle,
contactsToCreate, contactsToCreate: participantsToSave,
}); });
} }
} catch (error) { } catch (error) {
this.logger.error( this.logger.error(
`Error during google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: ${error.message}`, `Error during google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: ${error.message}`,
); );
} }
} else { } else {
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId} done with nothing to import.`, `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} done with nothing to import.`,
); );
} }
if (!nextSyncToken) { if (!nextSyncToken) {
throw new Error( throw new Error(
`No next sync token found for connected account ${connectedAccountId} in workspace ${workspaceId} during full-sync`, `No next sync token found for connected account ${connectedAccountId} in workspace ${workspaceId} during sync`,
); );
} }
@ -338,29 +397,15 @@ export class GoogleCalendarFullSyncService {
endTime = Date.now(); endTime = Date.now();
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId}: updating sync cursor in ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: updating sync cursor in ${
endTime - startTime endTime - startTime
}ms.`, }ms.`,
); );
this.logger.log( this.logger.log(
`google calendar full-sync for workspace ${workspaceId} and account ${connectedAccountId} ${ `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} ${
nextPageToken ? `and ${nextPageToken} pageToken` : '' syncToken ? `and ${syncToken} syncToken ` : ''
} done.`, }done.`,
); );
if (nextPageToken) {
await this.messageQueueService.add<GoogleCalendarFullSyncJobData>(
GoogleCalendarFullSyncService.name,
{
workspaceId,
connectedAccountId,
nextPageToken,
},
{
retryLimit: 2,
},
);
}
} }
} }

View File

@ -1,13 +1,15 @@
import { calendar_v3 } from 'googleapis'; import { calendar_v3 as calendarV3 } from 'googleapis';
import { v4 } from 'uuid'; import { v4 } from 'uuid';
import { CalendarEventParticipantResponseStatus } from 'src/modules/calendar/standard-objects/calendar-event-participant.object-metadata'; import { CalendarEventParticipantResponseStatus } from 'src/modules/calendar/standard-objects/calendar-event-participant.object-metadata';
import { CalendarEventWithParticipants } from 'src/modules/calendar/types/calendar-event'; import { CalendarEventWithParticipants } from 'src/modules/calendar/types/calendar-event';
export const formatGoogleCalendarEvent = ( export const formatGoogleCalendarEvent = (
event: calendar_v3.Schema$Event, event: calendarV3.Schema$Event,
iCalUIDCalendarEventIdMap: Map<string, string>,
): CalendarEventWithParticipants => { ): CalendarEventWithParticipants => {
const id = v4(); const id =
(event.iCalUID && iCalUIDCalendarEventIdMap.get(event.iCalUID)) ?? v4();
const formatResponseStatus = (status: string | null | undefined) => { const formatResponseStatus = (status: string | null | undefined) => {
switch (status) { switch (status) {

View File

@ -1,8 +1,8 @@
export const googleCalendarSearchFilterExcludeEmails = ( export const googleCalendarSearchFilterExcludeEmails = (
emails: string[], emails: string[],
): string => { ): string | undefined => {
if (emails.length === 0) { if (emails.length === 0) {
return ''; return undefined;
} }
return `email=-${emails.join(', -')}`; return `email=-${emails.join(', -')}`;

View File

@ -2,9 +2,8 @@ import { Module } from '@nestjs/common';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata'; import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/standard-objects/connected-account.object-metadata';
import { GmailFetchMessagesFromCacheCronCommand } from 'src/modules/messaging/commands/crons/gmail-fetch-messages-from-cache.cron.command'; import { GmailFetchMessagesFromCacheCronCommand } from 'src/modules/messaging/jobs/crons/gmail-fetch-messages-from-cache.cron.command';
import { GmailPartialSyncCronCommand } from 'src/modules/messaging/commands/crons/gmail-partial-sync.cron.command'; import { GmailPartialSyncCronCommand } from 'src/modules/messaging/jobs/crons/gmail-partial-sync.cron.command';
@Module({ @Module({
imports: [ imports: [
ObjectMetadataRepositoryModule.forFeature([ConnectedAccountObjectMetadata]), ObjectMetadataRepositoryModule.forFeature([ConnectedAccountObjectMetadata]),

View File

@ -1,4 +1,4 @@
import { Injectable, Logger } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { InjectRepository } from '@nestjs/typeorm'; import { InjectRepository } from '@nestjs/typeorm';
import { Repository, In } from 'typeorm'; import { Repository, In } from 'typeorm';
@ -16,8 +16,6 @@ import { GmailFetchMessageContentFromCacheService } from 'src/modules/messaging/
export class FetchAllMessagesFromCacheCronJob export class FetchAllMessagesFromCacheCronJob
implements MessageQueueJob<undefined> implements MessageQueueJob<undefined>
{ {
private readonly logger = new Logger(FetchAllMessagesFromCacheCronJob.name);
constructor( constructor(
@InjectRepository(Workspace, 'core') @InjectRepository(Workspace, 'core')
private readonly workspaceRepository: Repository<Workspace>, private readonly workspaceRepository: Repository<Workspace>,

View File

@ -4,8 +4,8 @@ import { Command, CommandRunner } from 'nest-commander';
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service';
import { fetchAllWorkspacesMessagesCronPattern } from 'src/modules/messaging/commands/crons/patterns/fetch-all-workspaces-messages.cron.pattern';
import { GmailPartialSyncCronJob } from 'src/modules/messaging/jobs/crons/gmail-partial-sync.cron.job'; import { GmailPartialSyncCronJob } from 'src/modules/messaging/jobs/crons/gmail-partial-sync.cron.job';
import { fetchAllWorkspacesMessagesCronPattern } from 'src/modules/messaging/jobs/crons/patterns/fetch-all-workspaces-messages.cron.pattern';
@Command({ @Command({
name: 'cron:messaging:gmail-partial-sync', name: 'cron:messaging:gmail-partial-sync',