mirror of
https://github.com/twentyhq/twenty.git
synced 2024-11-23 22:12:24 +03:00
4398 decouple contacts and companies creation from messages import (#4590)
* emit event
* create queue and listener
* filter participants with role 'from'
* create job
* Add job to job module
* Refactoring
* Refactor contact creation in CreateCompanyAndContactService
* update job
* wip
* add getByHandlesWithoutPersonIdAndWorkspaceMemberId to calendar event attendee repository
* refactoring
* refactoring
* Revert "refactoring"
This reverts commit e5434f0b87
.
* fix nest imports
* add await
* fix contact creation condition
* emit contact creation event after calendar-full-sync
* add await
* add missing transactionManager
* calendar event attendees personId update is working
* messageParticipant and calendarEventAttendee update is working as intended
* rename module
* fix lodash import
* add test
* update package.json
This commit is contained in:
parent
1a763263c9
commit
96cad2accd
@ -44,6 +44,8 @@
|
|||||||
"graphql-middleware": "^6.1.35",
|
"graphql-middleware": "^6.1.35",
|
||||||
"jwt-decode": "^4.0.0",
|
"jwt-decode": "^4.0.0",
|
||||||
"lodash.differencewith": "^4.5.0",
|
"lodash.differencewith": "^4.5.0",
|
||||||
|
"lodash.uniq": "^4.5.0",
|
||||||
|
"lodash.uniqby": "^4.7.0",
|
||||||
"passport": "^0.7.0",
|
"passport": "^0.7.0",
|
||||||
"psl": "^1.9.0",
|
"psl": "^1.9.0",
|
||||||
"tsconfig-paths": "^4.2.0"
|
"tsconfig-paths": "^4.2.0"
|
||||||
@ -57,6 +59,8 @@
|
|||||||
"@types/lodash.isobject": "^3.0.7",
|
"@types/lodash.isobject": "^3.0.7",
|
||||||
"@types/lodash.omit": "^4.5.9",
|
"@types/lodash.omit": "^4.5.9",
|
||||||
"@types/lodash.snakecase": "^4.1.7",
|
"@types/lodash.snakecase": "^4.1.7",
|
||||||
|
"@types/lodash.uniq": "^4.5.9",
|
||||||
|
"@types/lodash.uniqby": "^4.7.9",
|
||||||
"@types/lodash.upperfirst": "^4.3.7",
|
"@types/lodash.upperfirst": "^4.3.7",
|
||||||
"@types/react": "^18.2.39",
|
"@types/react": "^18.2.39",
|
||||||
"rimraf": "^5.0.5",
|
"rimraf": "^5.0.5",
|
||||||
|
@ -18,7 +18,7 @@ import { EnvironmentModule } from 'src/engine/integrations/environment/environme
|
|||||||
import { FetchAllWorkspacesMessagesJob } from 'src/modules/messaging/commands/crons/fetch-all-workspaces-messages.job';
|
import { FetchAllWorkspacesMessagesJob } from 'src/modules/messaging/commands/crons/fetch-all-workspaces-messages.job';
|
||||||
import { MatchMessageParticipantJob } from 'src/modules/messaging/jobs/match-message-participant.job';
|
import { MatchMessageParticipantJob } from 'src/modules/messaging/jobs/match-message-participant.job';
|
||||||
import { CreateCompaniesAndContactsAfterSyncJob } from 'src/modules/messaging/jobs/create-companies-and-contacts-after-sync.job';
|
import { CreateCompaniesAndContactsAfterSyncJob } from 'src/modules/messaging/jobs/create-companies-and-contacts-after-sync.job';
|
||||||
import { CreateCompaniesAndContactsModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company-and-contact/create-company-and-contact.module';
|
import { AutoCompaniesAndContactsCreationModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module';
|
||||||
import { DataSeedDemoWorkspaceModule } from 'src/database/commands/data-seed-demo-workspace/data-seed-demo-workspace.module';
|
import { DataSeedDemoWorkspaceModule } from 'src/database/commands/data-seed-demo-workspace/data-seed-demo-workspace.module';
|
||||||
import { DataSeedDemoWorkspaceJob } from 'src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job';
|
import { DataSeedDemoWorkspaceJob } from 'src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job';
|
||||||
import { DeleteConnectedAccountAssociatedMessagingDataJob } from 'src/modules/messaging/jobs/delete-connected-account-associated-messaging-data.job';
|
import { DeleteConnectedAccountAssociatedMessagingDataJob } from 'src/modules/messaging/jobs/delete-connected-account-associated-messaging-data.job';
|
||||||
@ -44,6 +44,7 @@ import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repos
|
|||||||
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 { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
||||||
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
||||||
|
import { CreateCompanyAndContactJob } from 'src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job';
|
||||||
import { SaveEventToDbJob } from 'src/engine/api/graphql/workspace-query-runner/jobs/save-event-to-db.job';
|
import { SaveEventToDbJob } from 'src/engine/api/graphql/workspace-query-runner/jobs/save-event-to-db.job';
|
||||||
import { EventObjectMetadata } from 'src/modules/event/standard-objects/event.object-metadata';
|
import { EventObjectMetadata } from 'src/modules/event/standard-objects/event.object-metadata';
|
||||||
|
|
||||||
@ -51,7 +52,7 @@ import { EventObjectMetadata } from 'src/modules/event/standard-objects/event.ob
|
|||||||
imports: [
|
imports: [
|
||||||
BillingModule,
|
BillingModule,
|
||||||
DataSourceModule,
|
DataSourceModule,
|
||||||
CreateCompaniesAndContactsModule,
|
AutoCompaniesAndContactsCreationModule,
|
||||||
DataSeedDemoWorkspaceModule,
|
DataSeedDemoWorkspaceModule,
|
||||||
EnvironmentModule,
|
EnvironmentModule,
|
||||||
HttpModule,
|
HttpModule,
|
||||||
@ -133,6 +134,10 @@ import { EventObjectMetadata } from 'src/modules/event/standard-objects/event.ob
|
|||||||
provide: RecordPositionBackfillJob.name,
|
provide: RecordPositionBackfillJob.name,
|
||||||
useClass: RecordPositionBackfillJob,
|
useClass: RecordPositionBackfillJob,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
provide: CreateCompanyAndContactJob.name,
|
||||||
|
useClass: CreateCompanyAndContactJob,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
provide: SaveEventToDbJob.name,
|
provide: SaveEventToDbJob.name,
|
||||||
useClass: SaveEventToDbJob,
|
useClass: SaveEventToDbJob,
|
||||||
|
@ -7,6 +7,7 @@ export enum MessageQueue {
|
|||||||
cronQueue = 'cron-queue',
|
cronQueue = 'cron-queue',
|
||||||
emailQueue = 'email-queue',
|
emailQueue = 'email-queue',
|
||||||
calendarQueue = 'calendar-queue',
|
calendarQueue = 'calendar-queue',
|
||||||
|
contactCreationQueue = 'contact-creation-queue',
|
||||||
billingQueue = 'billing-queue',
|
billingQueue = 'billing-queue',
|
||||||
recordPositionBackfillQueue = 'record-position-backfill-queue',
|
recordPositionBackfillQueue = 'record-position-backfill-queue',
|
||||||
entityEventsToDbQueue = 'entity-events-to-db-queue',
|
entityEventsToDbQueue = 'entity-events-to-db-queue',
|
||||||
|
@ -7,7 +7,10 @@ import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/work
|
|||||||
import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record';
|
import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record';
|
||||||
import { CalendarEventAttendeeObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-event-attendee.object-metadata';
|
import { CalendarEventAttendeeObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-event-attendee.object-metadata';
|
||||||
import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/getFlattenedValuesAndValuesStringForBatchRawQuery.util';
|
import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/getFlattenedValuesAndValuesStringForBatchRawQuery.util';
|
||||||
import { CalendarEventAttendee } from 'src/modules/calendar/types/calendar-event';
|
import {
|
||||||
|
CalendarEventAttendee,
|
||||||
|
CalendarEventAttendeeWithId,
|
||||||
|
} from 'src/modules/calendar/types/calendar-event';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class CalendarEventAttendeeRepository {
|
export class CalendarEventAttendeeRepository {
|
||||||
@ -172,4 +175,29 @@ export class CalendarEventAttendeeRepository {
|
|||||||
transactionManager,
|
transactionManager,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async getWithoutPersonIdAndWorkspaceMemberId(
|
||||||
|
workspaceId: string,
|
||||||
|
transactionManager?: EntityManager,
|
||||||
|
): Promise<CalendarEventAttendeeWithId[]> {
|
||||||
|
if (!workspaceId) {
|
||||||
|
throw new Error('WorkspaceId is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
const dataSourceSchema =
|
||||||
|
this.workspaceDataSourceService.getSchemaName(workspaceId);
|
||||||
|
|
||||||
|
const calendarEventAttendees: CalendarEventAttendeeWithId[] =
|
||||||
|
await this.workspaceDataSourceService.executeRawQuery(
|
||||||
|
`SELECT "calendarEventAttendee".*
|
||||||
|
FROM ${dataSourceSchema}."calendarEventAttendee" AS "calendarEventAttendee"
|
||||||
|
WHERE "calendarEventAttendee"."personId" IS NULL
|
||||||
|
AND "calendarEventAttendee"."workspaceMemberId" IS NULL`,
|
||||||
|
[],
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
return calendarEventAttendees;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,16 @@
|
|||||||
|
import { Module } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
|
||||||
|
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
|
||||||
|
import { CalendarEventAttendeeService } from 'src/modules/calendar/services/calendar-event-attendee/calendar-event-attendee.service';
|
||||||
|
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
||||||
|
|
||||||
|
@Module({
|
||||||
|
imports: [
|
||||||
|
WorkspaceDataSourceModule,
|
||||||
|
ObjectMetadataRepositoryModule.forFeature([PersonObjectMetadata]),
|
||||||
|
],
|
||||||
|
providers: [CalendarEventAttendeeService],
|
||||||
|
exports: [CalendarEventAttendeeService],
|
||||||
|
})
|
||||||
|
export class CalendarEventAttendeeModule {}
|
@ -0,0 +1,65 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { EntityManager } from 'typeorm';
|
||||||
|
|
||||||
|
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
||||||
|
import { PersonRepository } from 'src/modules/person/repositories/person.repository';
|
||||||
|
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
||||||
|
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
|
||||||
|
import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/getFlattenedValuesAndValuesStringForBatchRawQuery.util';
|
||||||
|
import { CalendarEventAttendeeWithId } from 'src/modules/calendar/types/calendar-event';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class CalendarEventAttendeeService {
|
||||||
|
constructor(
|
||||||
|
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
||||||
|
@InjectObjectMetadataRepository(PersonObjectMetadata)
|
||||||
|
private readonly personRepository: PersonRepository,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async updateCalendarEventAttendeesAfterContactCreation(
|
||||||
|
attendees: CalendarEventAttendeeWithId[],
|
||||||
|
workspaceId: string,
|
||||||
|
transactionManager?: EntityManager,
|
||||||
|
): Promise<void> {
|
||||||
|
if (!attendees) return;
|
||||||
|
|
||||||
|
const dataSourceSchema =
|
||||||
|
this.workspaceDataSourceService.getSchemaName(workspaceId);
|
||||||
|
|
||||||
|
const handles = attendees.map((attendee) => attendee.handle);
|
||||||
|
|
||||||
|
const attendeePersonIds = await this.personRepository.getByEmails(
|
||||||
|
handles,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
const calendarEventAttendeesToUpdate = attendees.map((attendee) => ({
|
||||||
|
id: attendee.id,
|
||||||
|
personId: attendeePersonIds.find(
|
||||||
|
(e: { id: string; email: string }) => e.email === attendee.handle,
|
||||||
|
)?.id,
|
||||||
|
}));
|
||||||
|
|
||||||
|
if (calendarEventAttendeesToUpdate.length === 0) return;
|
||||||
|
|
||||||
|
const { flattenedValues, valuesString } =
|
||||||
|
getFlattenedValuesAndValuesStringForBatchRawQuery(
|
||||||
|
calendarEventAttendeesToUpdate,
|
||||||
|
{
|
||||||
|
id: 'uuid',
|
||||||
|
personId: 'uuid',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.workspaceDataSourceService.executeRawQuery(
|
||||||
|
`UPDATE ${dataSourceSchema}."calendarEventAttendee" AS "calendarEventAttendee" SET "personId" = "data"."personId"
|
||||||
|
FROM (VALUES ${valuesString}) AS "data"("id", "personId")
|
||||||
|
WHERE "calendarEventAttendee"."id" = "data"."id"`,
|
||||||
|
flattenedValues,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,6 @@
|
|||||||
import { Inject, Injectable, Logger } from '@nestjs/common';
|
import { Inject, Injectable, Logger } from '@nestjs/common';
|
||||||
import { InjectRepository } from '@nestjs/typeorm';
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
||||||
|
|
||||||
import { Repository } from 'typeorm';
|
import { Repository } from 'typeorm';
|
||||||
|
|
||||||
@ -53,6 +54,7 @@ export class GoogleCalendarFullSyncService {
|
|||||||
@InjectRepository(FeatureFlagEntity, 'core')
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
||||||
|
private readonly eventEmitter: EventEmitter2,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
public async startGoogleCalendarFullSync(
|
public async startGoogleCalendarFullSync(
|
||||||
@ -294,6 +296,16 @@ export class GoogleCalendarFullSyncService {
|
|||||||
}ms.`,
|
}ms.`,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (calendarChannel.isContactAutoCreationEnabled) {
|
||||||
|
const contactsToCreate = attendeesToSave;
|
||||||
|
|
||||||
|
this.eventEmitter.emit(`createContacts`, {
|
||||||
|
workspaceId,
|
||||||
|
connectedAccountHandle: connectedAccount.handle,
|
||||||
|
contactsToCreate,
|
||||||
|
});
|
||||||
|
}
|
||||||
} 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 full-sync for workspace ${workspaceId} and account ${connectedAccountId}: ${error.message}`,
|
||||||
|
@ -29,3 +29,7 @@ export type CalendarEventWithAttendees = CalendarEvent & {
|
|||||||
externalId: string;
|
externalId: string;
|
||||||
attendees: CalendarEventAttendee[];
|
attendees: CalendarEventAttendee[];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type CalendarEventAttendeeWithId = CalendarEventAttendee & {
|
||||||
|
id: string;
|
||||||
|
};
|
||||||
|
@ -0,0 +1,31 @@
|
|||||||
|
import { Module } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service';
|
||||||
|
import { CreateCompanyModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.module';
|
||||||
|
import { CreateContactModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.module';
|
||||||
|
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
|
||||||
|
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
||||||
|
import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/standard-objects/workspace-member.object-metadata';
|
||||||
|
import { MessageParticipantModule } from 'src/modules/messaging/services/message-participant/message-participant.module';
|
||||||
|
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
|
||||||
|
import { CreateCompanyAndContactListener } from 'src/modules/connected-account/auto-companies-and-contacts-creation/listeners/create-company-and-contact.listener';
|
||||||
|
import { CalendarEventAttendeeObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-event-attendee.object-metadata';
|
||||||
|
import { CalendarEventAttendeeModule } from 'src/modules/calendar/services/calendar-event-attendee/calendar-event-attendee.module';
|
||||||
|
|
||||||
|
@Module({
|
||||||
|
imports: [
|
||||||
|
CreateContactModule,
|
||||||
|
CreateCompanyModule,
|
||||||
|
ObjectMetadataRepositoryModule.forFeature([
|
||||||
|
PersonObjectMetadata,
|
||||||
|
WorkspaceMemberObjectMetadata,
|
||||||
|
CalendarEventAttendeeObjectMetadata,
|
||||||
|
]),
|
||||||
|
MessageParticipantModule,
|
||||||
|
WorkspaceDataSourceModule,
|
||||||
|
CalendarEventAttendeeModule,
|
||||||
|
],
|
||||||
|
providers: [CreateCompanyAndContactService, CreateCompanyAndContactListener],
|
||||||
|
exports: [CreateCompanyAndContactService],
|
||||||
|
})
|
||||||
|
export class AutoCompaniesAndContactsCreationModule {}
|
@ -1,22 +0,0 @@
|
|||||||
import { Module } from '@nestjs/common';
|
|
||||||
|
|
||||||
import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company-and-contact/create-company-and-contact.service';
|
|
||||||
import { CreateCompanyModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.module';
|
|
||||||
import { CreateContactModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.module';
|
|
||||||
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
|
|
||||||
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
|
||||||
import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/standard-objects/workspace-member.object-metadata';
|
|
||||||
|
|
||||||
@Module({
|
|
||||||
imports: [
|
|
||||||
CreateContactModule,
|
|
||||||
CreateCompanyModule,
|
|
||||||
ObjectMetadataRepositoryModule.forFeature([
|
|
||||||
PersonObjectMetadata,
|
|
||||||
WorkspaceMemberObjectMetadata,
|
|
||||||
]),
|
|
||||||
],
|
|
||||||
providers: [CreateCompanyAndContactService],
|
|
||||||
exports: [CreateCompanyAndContactService],
|
|
||||||
})
|
|
||||||
export class CreateCompaniesAndContactsModule {}
|
|
@ -1,117 +0,0 @@
|
|||||||
import { Injectable } from '@nestjs/common';
|
|
||||||
|
|
||||||
import { EntityManager } from 'typeorm';
|
|
||||||
import compact from 'lodash/compact';
|
|
||||||
|
|
||||||
import { Participant } from 'src/modules/messaging/types/gmail-message';
|
|
||||||
import { getDomainNameFromHandle } from 'src/modules/messaging/utils/get-domain-name-from-handle.util';
|
|
||||||
import { CreateCompanyService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service';
|
|
||||||
import { CreateContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service';
|
|
||||||
import { PersonRepository } from 'src/modules/person/repositories/person.repository';
|
|
||||||
import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository';
|
|
||||||
import { getUniqueParticipantsAndHandles } from 'src/modules/messaging/utils/get-unique-participants-and-handles.util';
|
|
||||||
import { filterOutParticipantsFromCompanyOrWorkspace } from 'src/modules/messaging/utils/filter-out-participants-from-company-or-workspace.util';
|
|
||||||
import { isWorkEmail } from 'src/utils/is-work-email';
|
|
||||||
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
|
||||||
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
|
||||||
import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/standard-objects/workspace-member.object-metadata';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class CreateCompanyAndContactService {
|
|
||||||
constructor(
|
|
||||||
private readonly createContactService: CreateContactService,
|
|
||||||
private readonly createCompaniesService: CreateCompanyService,
|
|
||||||
@InjectObjectMetadataRepository(PersonObjectMetadata)
|
|
||||||
private readonly personRepository: PersonRepository,
|
|
||||||
@InjectObjectMetadataRepository(WorkspaceMemberObjectMetadata)
|
|
||||||
private readonly workspaceMemberRepository: WorkspaceMemberRepository,
|
|
||||||
) {}
|
|
||||||
|
|
||||||
async createCompaniesAndContacts(
|
|
||||||
selfHandle: string,
|
|
||||||
participants: Participant[],
|
|
||||||
workspaceId: string,
|
|
||||||
transactionManager?: EntityManager,
|
|
||||||
) {
|
|
||||||
if (!participants || participants.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: This is a feature that may be implemented in the future
|
|
||||||
const isContactAutoCreationForNonWorkEmailsEnabled = false;
|
|
||||||
|
|
||||||
const workspaceMembers =
|
|
||||||
await this.workspaceMemberRepository.getAllByWorkspaceId(
|
|
||||||
workspaceId,
|
|
||||||
transactionManager,
|
|
||||||
);
|
|
||||||
|
|
||||||
const participantsFromOtherCompanies =
|
|
||||||
filterOutParticipantsFromCompanyOrWorkspace(
|
|
||||||
participants,
|
|
||||||
selfHandle,
|
|
||||||
workspaceMembers,
|
|
||||||
);
|
|
||||||
|
|
||||||
const { uniqueParticipants, uniqueHandles } =
|
|
||||||
getUniqueParticipantsAndHandles(participantsFromOtherCompanies);
|
|
||||||
|
|
||||||
if (uniqueHandles.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const alreadyCreatedContacts = await this.personRepository.getByEmails(
|
|
||||||
uniqueHandles,
|
|
||||||
workspaceId,
|
|
||||||
);
|
|
||||||
|
|
||||||
const alreadyCreatedContactEmails: string[] = alreadyCreatedContacts?.map(
|
|
||||||
({ email }) => email,
|
|
||||||
);
|
|
||||||
|
|
||||||
const filteredParticipants = uniqueParticipants.filter(
|
|
||||||
(participant) =>
|
|
||||||
!alreadyCreatedContactEmails.includes(participant.handle) &&
|
|
||||||
participant.handle.includes('@') &&
|
|
||||||
(isContactAutoCreationForNonWorkEmailsEnabled ||
|
|
||||||
isWorkEmail(participant.handle)),
|
|
||||||
);
|
|
||||||
|
|
||||||
const filteredParticipantsWithCompanyDomainNames =
|
|
||||||
filteredParticipants?.map((participant) => ({
|
|
||||||
handle: participant.handle,
|
|
||||||
displayName: participant.displayName,
|
|
||||||
companyDomainName: isWorkEmail(participant.handle)
|
|
||||||
? getDomainNameFromHandle(participant.handle)
|
|
||||||
: undefined,
|
|
||||||
}));
|
|
||||||
|
|
||||||
const domainNamesToCreate = compact(
|
|
||||||
filteredParticipantsWithCompanyDomainNames.map(
|
|
||||||
(participant) => participant.companyDomainName,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
const companiesObject = await this.createCompaniesService.createCompanies(
|
|
||||||
domainNamesToCreate,
|
|
||||||
workspaceId,
|
|
||||||
transactionManager,
|
|
||||||
);
|
|
||||||
|
|
||||||
const contactsToCreate = filteredParticipantsWithCompanyDomainNames.map(
|
|
||||||
(participant) => ({
|
|
||||||
handle: participant.handle,
|
|
||||||
displayName: participant.displayName,
|
|
||||||
companyId:
|
|
||||||
participant.companyDomainName &&
|
|
||||||
companiesObject[participant.companyDomainName],
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
await this.createContactService.createContacts(
|
|
||||||
contactsToCreate,
|
|
||||||
workspaceId,
|
|
||||||
transactionManager,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
@ -0,0 +1,36 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface';
|
||||||
|
|
||||||
|
import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service';
|
||||||
|
|
||||||
|
export type CreateCompanyAndContactJobData = {
|
||||||
|
workspaceId: string;
|
||||||
|
connectedAccountHandle: string;
|
||||||
|
contactsToCreate: {
|
||||||
|
displayName: string;
|
||||||
|
handle: string;
|
||||||
|
}[];
|
||||||
|
};
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class CreateCompanyAndContactJob
|
||||||
|
implements MessageQueueJob<CreateCompanyAndContactJobData>
|
||||||
|
{
|
||||||
|
constructor(
|
||||||
|
private readonly createCompanyAndContactService: CreateCompanyAndContactService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
async handle(data: CreateCompanyAndContactJobData): Promise<void> {
|
||||||
|
const { workspaceId, connectedAccountHandle, contactsToCreate } = data;
|
||||||
|
|
||||||
|
await this.createCompanyAndContactService.createCompaniesAndContactsAndUpdateParticipants(
|
||||||
|
connectedAccountHandle,
|
||||||
|
contactsToCreate.map((contact) => ({
|
||||||
|
handle: contact.handle,
|
||||||
|
displayName: contact.displayName,
|
||||||
|
})),
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,32 @@
|
|||||||
|
import { Injectable, Inject } from '@nestjs/common';
|
||||||
|
import { OnEvent } from '@nestjs/event-emitter';
|
||||||
|
|
||||||
|
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
|
||||||
|
import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service';
|
||||||
|
import {
|
||||||
|
CreateCompanyAndContactJobData,
|
||||||
|
CreateCompanyAndContactJob,
|
||||||
|
} from 'src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class CreateCompanyAndContactListener {
|
||||||
|
constructor(
|
||||||
|
@Inject(MessageQueue.contactCreationQueue)
|
||||||
|
private readonly messageQueueService: MessageQueueService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
@OnEvent('createContacts')
|
||||||
|
async handleContactCreationEvent(payload: {
|
||||||
|
workspaceId: string;
|
||||||
|
connectedAccountHandle: string;
|
||||||
|
contactsToCreate: {
|
||||||
|
displayName: string;
|
||||||
|
handle: string;
|
||||||
|
}[];
|
||||||
|
}) {
|
||||||
|
await this.messageQueueService.add<CreateCompanyAndContactJobData>(
|
||||||
|
CreateCompanyAndContactJob.name,
|
||||||
|
payload,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,179 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { EntityManager } from 'typeorm';
|
||||||
|
import compact from 'lodash/compact';
|
||||||
|
|
||||||
|
import { getDomainNameFromHandle } from 'src/modules/messaging/utils/get-domain-name-from-handle.util';
|
||||||
|
import { CreateCompanyService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service';
|
||||||
|
import { CreateContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service';
|
||||||
|
import { PersonRepository } from 'src/modules/person/repositories/person.repository';
|
||||||
|
import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository';
|
||||||
|
import { isWorkEmail } from 'src/utils/is-work-email';
|
||||||
|
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
||||||
|
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
||||||
|
import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/standard-objects/workspace-member.object-metadata';
|
||||||
|
import { getUniqueContactsAndHandles } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util';
|
||||||
|
import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type';
|
||||||
|
import { MessageParticipantRepository } from 'src/modules/messaging/repositories/message-participant.repository';
|
||||||
|
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
|
||||||
|
import { MessageParticipantService } from 'src/modules/messaging/services/message-participant/message-participant.service';
|
||||||
|
import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
||||||
|
import { CalendarEventAttendeeService } from 'src/modules/calendar/services/calendar-event-attendee/calendar-event-attendee.service';
|
||||||
|
import { CalendarEventAttendeeRepository } from 'src/modules/calendar/repositories/calendar-event-attendee.repository';
|
||||||
|
import { CalendarEventAttendeeObjectMetadata } from 'src/modules/calendar/standard-objects/calendar-event-attendee.object-metadata';
|
||||||
|
import { filterOutContactsFromCompanyOrWorkspace } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class CreateCompanyAndContactService {
|
||||||
|
constructor(
|
||||||
|
private readonly createContactService: CreateContactService,
|
||||||
|
private readonly createCompaniesService: CreateCompanyService,
|
||||||
|
@InjectObjectMetadataRepository(PersonObjectMetadata)
|
||||||
|
private readonly personRepository: PersonRepository,
|
||||||
|
@InjectObjectMetadataRepository(WorkspaceMemberObjectMetadata)
|
||||||
|
private readonly workspaceMemberRepository: WorkspaceMemberRepository,
|
||||||
|
@InjectObjectMetadataRepository(MessageParticipantObjectMetadata)
|
||||||
|
private readonly messageParticipantRepository: MessageParticipantRepository,
|
||||||
|
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
||||||
|
private readonly messageParticipantService: MessageParticipantService,
|
||||||
|
@InjectObjectMetadataRepository(CalendarEventAttendeeObjectMetadata)
|
||||||
|
private readonly calendarEventAttendeeRepository: CalendarEventAttendeeRepository,
|
||||||
|
private readonly calendarEventAttendeeService: CalendarEventAttendeeService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
async createCompaniesAndContacts(
|
||||||
|
connectedAccountHandle: string,
|
||||||
|
contactsToCreate: Contacts,
|
||||||
|
workspaceId: string,
|
||||||
|
transactionManager?: EntityManager,
|
||||||
|
) {
|
||||||
|
if (!contactsToCreate || contactsToCreate.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: This is a feature that may be implemented in the future
|
||||||
|
const isContactAutoCreationForNonWorkEmailsEnabled = false;
|
||||||
|
|
||||||
|
const workspaceMembers =
|
||||||
|
await this.workspaceMemberRepository.getAllByWorkspaceId(
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
const contactsToCreateFromOtherCompanies = contactsToCreate;
|
||||||
|
|
||||||
|
filterOutContactsFromCompanyOrWorkspace(
|
||||||
|
contactsToCreate,
|
||||||
|
connectedAccountHandle,
|
||||||
|
workspaceMembers,
|
||||||
|
);
|
||||||
|
|
||||||
|
const { uniqueContacts, uniqueHandles } = getUniqueContactsAndHandles(
|
||||||
|
contactsToCreateFromOtherCompanies,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (uniqueHandles.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const alreadyCreatedContacts = await this.personRepository.getByEmails(
|
||||||
|
uniqueHandles,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
const alreadyCreatedContactEmails: string[] = alreadyCreatedContacts?.map(
|
||||||
|
({ email }) => email,
|
||||||
|
);
|
||||||
|
|
||||||
|
const filteredContactsToCreate = uniqueContacts.filter(
|
||||||
|
(participant) =>
|
||||||
|
!alreadyCreatedContactEmails.includes(participant.handle) &&
|
||||||
|
participant.handle.includes('@') &&
|
||||||
|
(isContactAutoCreationForNonWorkEmailsEnabled ||
|
||||||
|
isWorkEmail(participant.handle)),
|
||||||
|
);
|
||||||
|
|
||||||
|
const filteredContactsToCreateWithCompanyDomainNames =
|
||||||
|
filteredContactsToCreate?.map((participant) => ({
|
||||||
|
handle: participant.handle,
|
||||||
|
displayName: participant.displayName,
|
||||||
|
companyDomainName: isWorkEmail(participant.handle)
|
||||||
|
? getDomainNameFromHandle(participant.handle)
|
||||||
|
: undefined,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const domainNamesToCreate = compact(
|
||||||
|
filteredContactsToCreateWithCompanyDomainNames.map(
|
||||||
|
(participant) => participant.companyDomainName,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
const companiesObject = await this.createCompaniesService.createCompanies(
|
||||||
|
domainNamesToCreate,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
const formattedContactsToCreate =
|
||||||
|
filteredContactsToCreateWithCompanyDomainNames.map((contact) => ({
|
||||||
|
handle: contact.handle,
|
||||||
|
displayName: contact.displayName,
|
||||||
|
companyId:
|
||||||
|
contact.companyDomainName &&
|
||||||
|
companiesObject[contact.companyDomainName],
|
||||||
|
}));
|
||||||
|
|
||||||
|
await this.createContactService.createContacts(
|
||||||
|
formattedContactsToCreate,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async createCompaniesAndContactsAndUpdateParticipants(
|
||||||
|
connectedAccountHandle: string,
|
||||||
|
contactsToCreate: Contacts,
|
||||||
|
workspaceId: string,
|
||||||
|
) {
|
||||||
|
const { dataSource: workspaceDataSource } =
|
||||||
|
await this.workspaceDataSourceService.connectedToWorkspaceDataSourceAndReturnMetadata(
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
|
||||||
|
await workspaceDataSource?.transaction(
|
||||||
|
async (transactionManager: EntityManager) => {
|
||||||
|
await this.createCompaniesAndContacts(
|
||||||
|
connectedAccountHandle,
|
||||||
|
contactsToCreate,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
const messageParticipantsWithoutPersonIdAndWorkspaceMemberId =
|
||||||
|
await this.messageParticipantRepository.getWithoutPersonIdAndWorkspaceMemberId(
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.messageParticipantService.updateMessageParticipantsAfterPeopleCreation(
|
||||||
|
messageParticipantsWithoutPersonIdAndWorkspaceMemberId,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
const calendarEventAttendeesWithoutPersonIdAndWorkspaceMemberId =
|
||||||
|
await this.calendarEventAttendeeRepository.getWithoutPersonIdAndWorkspaceMemberId(
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.calendarEventAttendeeService.updateCalendarEventAttendeesAfterContactCreation(
|
||||||
|
calendarEventAttendeesWithoutPersonIdAndWorkspaceMemberId,
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,6 @@
|
|||||||
|
export type Contact = {
|
||||||
|
handle: string;
|
||||||
|
displayName: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Contacts = Contact[];
|
@ -0,0 +1,32 @@
|
|||||||
|
import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type';
|
||||||
|
import { getUniqueContactsAndHandles } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util';
|
||||||
|
|
||||||
|
describe('getUniqueContactsAndHandles', () => {
|
||||||
|
it('should return empty arrays when contacts is empty', () => {
|
||||||
|
const contacts: Contacts = [];
|
||||||
|
const result = getUniqueContactsAndHandles(contacts);
|
||||||
|
|
||||||
|
expect(result.uniqueContacts).toEqual([]);
|
||||||
|
expect(result.uniqueHandles).toEqual([]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return unique contacts and handles', () => {
|
||||||
|
const contacts: Contacts = [
|
||||||
|
{ handle: 'john@twenty.com', displayName: 'John Doe' },
|
||||||
|
{ handle: 'john@twenty.com', displayName: 'John Doe' },
|
||||||
|
{ handle: 'jane@twenty.com', displayName: 'Jane Smith' },
|
||||||
|
{ handle: 'jane@twenty.com', displayName: 'Jane Smith' },
|
||||||
|
{ handle: 'jane@twenty.com', displayName: 'Jane Smith' },
|
||||||
|
];
|
||||||
|
const result = getUniqueContactsAndHandles(contacts);
|
||||||
|
|
||||||
|
expect(result.uniqueContacts).toEqual([
|
||||||
|
{ handle: 'john@twenty.com', displayName: 'John Doe' },
|
||||||
|
{ handle: 'jane@twenty.com', displayName: 'Jane Smith' },
|
||||||
|
]);
|
||||||
|
expect(result.uniqueHandles).toEqual([
|
||||||
|
'john@twenty.com',
|
||||||
|
'jane@twenty.com',
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
@ -1,13 +1,13 @@
|
|||||||
import { Participant } from 'src/modules/messaging/types/gmail-message';
|
|
||||||
import { getDomainNameFromHandle } from 'src/modules/messaging/utils/get-domain-name-from-handle.util';
|
import { getDomainNameFromHandle } from 'src/modules/messaging/utils/get-domain-name-from-handle.util';
|
||||||
import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/standard-objects/workspace-member.object-metadata';
|
import { WorkspaceMemberObjectMetadata } from 'src/modules/workspace-member/standard-objects/workspace-member.object-metadata';
|
||||||
import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record';
|
import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record';
|
||||||
|
import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type';
|
||||||
|
|
||||||
export function filterOutParticipantsFromCompanyOrWorkspace(
|
export function filterOutContactsFromCompanyOrWorkspace(
|
||||||
participants: Participant[],
|
contacts: Contacts,
|
||||||
selfHandle: string,
|
selfHandle: string,
|
||||||
workspaceMembers: ObjectRecord<WorkspaceMemberObjectMetadata>[],
|
workspaceMembers: ObjectRecord<WorkspaceMemberObjectMetadata>[],
|
||||||
): Participant[] {
|
): Contacts {
|
||||||
const selfDomainName = getDomainNameFromHandle(selfHandle);
|
const selfDomainName = getDomainNameFromHandle(selfHandle);
|
||||||
|
|
||||||
const workspaceMembersMap = workspaceMembers.reduce(
|
const workspaceMembersMap = workspaceMembers.reduce(
|
||||||
@ -19,9 +19,9 @@ export function filterOutParticipantsFromCompanyOrWorkspace(
|
|||||||
new Map<string, boolean>(),
|
new Map<string, boolean>(),
|
||||||
);
|
);
|
||||||
|
|
||||||
return participants.filter(
|
return contacts.filter(
|
||||||
(participant) =>
|
(contact) =>
|
||||||
getDomainNameFromHandle(participant.handle) !== selfDomainName &&
|
getDomainNameFromHandle(contact.handle) !== selfDomainName &&
|
||||||
!workspaceMembersMap[participant.handle],
|
!workspaceMembersMap[contact.handle],
|
||||||
);
|
);
|
||||||
}
|
}
|
@ -0,0 +1,19 @@
|
|||||||
|
import uniq from 'lodash.uniq';
|
||||||
|
import uniqBy from 'lodash.uniqby';
|
||||||
|
|
||||||
|
import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type';
|
||||||
|
|
||||||
|
export function getUniqueContactsAndHandles(contacts: Contacts): {
|
||||||
|
uniqueContacts: Contacts;
|
||||||
|
uniqueHandles: string[];
|
||||||
|
} {
|
||||||
|
if (contacts.length === 0) {
|
||||||
|
return { uniqueContacts: [], uniqueHandles: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
const uniqueHandles = uniq(contacts.map((participant) => participant.handle));
|
||||||
|
|
||||||
|
const uniqueContacts = uniqBy(contacts, 'handle');
|
||||||
|
|
||||||
|
return { uniqueContacts, uniqueHandles };
|
||||||
|
}
|
@ -3,7 +3,7 @@ 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 { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
||||||
import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company-and-contact/create-company-and-contact.service';
|
import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service';
|
||||||
import { MessageChannelRepository } from 'src/modules/messaging/repositories/message-channel.repository';
|
import { MessageChannelRepository } from 'src/modules/messaging/repositories/message-channel.repository';
|
||||||
import { MessageParticipantRepository } from 'src/modules/messaging/repositories/message-participant.repository';
|
import { MessageParticipantRepository } from 'src/modules/messaging/repositories/message-participant.repository';
|
||||||
import { MessageParticipantService } from 'src/modules/messaging/services/message-participant/message-participant.service';
|
import { MessageParticipantService } from 'src/modules/messaging/services/message-participant/message-participant.service';
|
||||||
|
@ -102,8 +102,7 @@ export class MessageParticipantRepository {
|
|||||||
return messageParticipants;
|
return messageParticipants;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async getByHandlesWithoutPersonIdAndWorkspaceMemberId(
|
public async getWithoutPersonIdAndWorkspaceMemberId(
|
||||||
handles: string[],
|
|
||||||
workspaceId: string,
|
workspaceId: string,
|
||||||
transactionManager?: EntityManager,
|
transactionManager?: EntityManager,
|
||||||
): Promise<ParticipantWithId[]> {
|
): Promise<ParticipantWithId[]> {
|
||||||
@ -116,18 +115,11 @@ export class MessageParticipantRepository {
|
|||||||
|
|
||||||
const messageParticipants: ParticipantWithId[] =
|
const messageParticipants: ParticipantWithId[] =
|
||||||
await this.workspaceDataSourceService.executeRawQuery(
|
await this.workspaceDataSourceService.executeRawQuery(
|
||||||
`SELECT "messageParticipant".id,
|
`SELECT "messageParticipant".*
|
||||||
"messageParticipant"."role",
|
|
||||||
"messageParticipant"."handle",
|
|
||||||
"messageParticipant"."displayName",
|
|
||||||
"messageParticipant"."personId",
|
|
||||||
"messageParticipant"."workspaceMemberId",
|
|
||||||
"messageParticipant"."messageId"
|
|
||||||
FROM ${dataSourceSchema}."messageParticipant" "messageParticipant"
|
FROM ${dataSourceSchema}."messageParticipant" "messageParticipant"
|
||||||
WHERE "messageParticipant"."personId" IS NULL
|
WHERE "messageParticipant"."personId" IS NULL
|
||||||
AND "messageParticipant"."workspaceMemberId" IS NULL
|
AND "messageParticipant"."workspaceMemberId" IS NULL`,
|
||||||
AND "messageParticipant"."handle" = ANY($1)`,
|
[],
|
||||||
[handles],
|
|
||||||
workspaceId,
|
workspaceId,
|
||||||
transactionManager,
|
transactionManager,
|
||||||
);
|
);
|
||||||
|
@ -8,7 +8,7 @@ import { ConnectedAccountObjectMetadata } from 'src/modules/connected-account/st
|
|||||||
import { FetchMessagesByBatchesModule } from 'src/modules/messaging/services/fetch-messages-by-batches/fetch-messages-by-batches.module';
|
import { FetchMessagesByBatchesModule } from 'src/modules/messaging/services/fetch-messages-by-batches/fetch-messages-by-batches.module';
|
||||||
import { GmailFullSyncService } from 'src/modules/messaging/services/gmail-full-sync/gmail-full-sync.service';
|
import { GmailFullSyncService } from 'src/modules/messaging/services/gmail-full-sync/gmail-full-sync.service';
|
||||||
import { MessagingProvidersModule } from 'src/modules/messaging/services/providers/messaging-providers.module';
|
import { MessagingProvidersModule } from 'src/modules/messaging/services/providers/messaging-providers.module';
|
||||||
import { SaveMessagesAndCreateContactsModule } from 'src/modules/messaging/services/save-message-and-create-contact/save-message-and-create-contacts.module';
|
import { SaveMessageAndEmitContactCreationEventModule } from 'src/modules/messaging/services/save-message-and-emit-contact-creation-event/save-message-and-emit-contact-creation-event.module';
|
||||||
import { MessageChannelMessageAssociationObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel-message-association.object-metadata';
|
import { MessageChannelMessageAssociationObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel-message-association.object-metadata';
|
||||||
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
||||||
|
|
||||||
@ -22,7 +22,7 @@ import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-obj
|
|||||||
MessageChannelMessageAssociationObjectMetadata,
|
MessageChannelMessageAssociationObjectMetadata,
|
||||||
BlocklistObjectMetadata,
|
BlocklistObjectMetadata,
|
||||||
]),
|
]),
|
||||||
SaveMessagesAndCreateContactsModule,
|
SaveMessageAndEmitContactCreationEventModule,
|
||||||
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
|
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
|
||||||
],
|
],
|
||||||
providers: [GmailFullSyncService],
|
providers: [GmailFullSyncService],
|
||||||
|
@ -17,7 +17,7 @@ import { MessageChannelMessageAssociationRepository } from 'src/modules/messagin
|
|||||||
import { createQueriesFromMessageIds } from 'src/modules/messaging/utils/create-queries-from-message-ids.util';
|
import { createQueriesFromMessageIds } from 'src/modules/messaging/utils/create-queries-from-message-ids.util';
|
||||||
import { gmailSearchFilterExcludeEmails } from 'src/modules/messaging/utils/gmail-search-filter.util';
|
import { gmailSearchFilterExcludeEmails } from 'src/modules/messaging/utils/gmail-search-filter.util';
|
||||||
import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository';
|
import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository';
|
||||||
import { SaveMessagesAndCreateContactsService } from 'src/modules/messaging/services/save-message-and-create-contact/save-messages-and-create-contacts.service';
|
import { SaveMessageAndEmitContactCreationEventService } from 'src/modules/messaging/services/save-message-and-emit-contact-creation-event/save-message-and-emit-contact-creation-event.service';
|
||||||
import {
|
import {
|
||||||
FeatureFlagEntity,
|
FeatureFlagEntity,
|
||||||
FeatureFlagKeys,
|
FeatureFlagKeys,
|
||||||
@ -47,7 +47,7 @@ export class GmailFullSyncService {
|
|||||||
private readonly messageChannelMessageAssociationRepository: MessageChannelMessageAssociationRepository,
|
private readonly messageChannelMessageAssociationRepository: MessageChannelMessageAssociationRepository,
|
||||||
@InjectObjectMetadataRepository(BlocklistObjectMetadata)
|
@InjectObjectMetadataRepository(BlocklistObjectMetadata)
|
||||||
private readonly blocklistRepository: BlocklistRepository,
|
private readonly blocklistRepository: BlocklistRepository,
|
||||||
private readonly saveMessagesAndCreateContactsService: SaveMessagesAndCreateContactsService,
|
private readonly saveMessagesAndCreateContactsService: SaveMessageAndEmitContactCreationEventService,
|
||||||
@InjectRepository(FeatureFlagEntity, 'core')
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
) {}
|
) {}
|
||||||
|
@ -9,7 +9,7 @@ import { FetchMessagesByBatchesModule } from 'src/modules/messaging/services/fet
|
|||||||
import { GmailPartialSyncService } from 'src/modules/messaging/services/gmail-partial-sync/gmail-partial-sync.service';
|
import { GmailPartialSyncService } from 'src/modules/messaging/services/gmail-partial-sync/gmail-partial-sync.service';
|
||||||
import { MessageModule } from 'src/modules/messaging/services/message/message.module';
|
import { MessageModule } from 'src/modules/messaging/services/message/message.module';
|
||||||
import { MessagingProvidersModule } from 'src/modules/messaging/services/providers/messaging-providers.module';
|
import { MessagingProvidersModule } from 'src/modules/messaging/services/providers/messaging-providers.module';
|
||||||
import { SaveMessagesAndCreateContactsModule } from 'src/modules/messaging/services/save-message-and-create-contact/save-message-and-create-contacts.module';
|
import { SaveMessageAndEmitContactCreationEventModule } from 'src/modules/messaging/services/save-message-and-emit-contact-creation-event/save-message-and-emit-contact-creation-event.module';
|
||||||
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
@ -22,7 +22,7 @@ import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-obj
|
|||||||
BlocklistObjectMetadata,
|
BlocklistObjectMetadata,
|
||||||
]),
|
]),
|
||||||
MessageModule,
|
MessageModule,
|
||||||
SaveMessagesAndCreateContactsModule,
|
SaveMessageAndEmitContactCreationEventModule,
|
||||||
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
|
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
|
||||||
],
|
],
|
||||||
providers: [GmailPartialSyncService],
|
providers: [GmailPartialSyncService],
|
||||||
|
@ -18,7 +18,7 @@ import { createQueriesFromMessageIds } from 'src/modules/messaging/utils/create-
|
|||||||
import { GmailMessage } from 'src/modules/messaging/types/gmail-message';
|
import { GmailMessage } from 'src/modules/messaging/types/gmail-message';
|
||||||
import { isPersonEmail } from 'src/modules/messaging/utils/is-person-email.util';
|
import { isPersonEmail } from 'src/modules/messaging/utils/is-person-email.util';
|
||||||
import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository';
|
import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository';
|
||||||
import { SaveMessagesAndCreateContactsService } from 'src/modules/messaging/services/save-message-and-create-contact/save-messages-and-create-contacts.service';
|
import { SaveMessageAndEmitContactCreationEventService } from 'src/modules/messaging/services/save-message-and-emit-contact-creation-event/save-message-and-emit-contact-creation-event.service';
|
||||||
import {
|
import {
|
||||||
FeatureFlagEntity,
|
FeatureFlagEntity,
|
||||||
FeatureFlagKeys,
|
FeatureFlagKeys,
|
||||||
@ -45,7 +45,7 @@ export class GmailPartialSyncService {
|
|||||||
private readonly messageService: MessageService,
|
private readonly messageService: MessageService,
|
||||||
@InjectObjectMetadataRepository(BlocklistObjectMetadata)
|
@InjectObjectMetadataRepository(BlocklistObjectMetadata)
|
||||||
private readonly blocklistRepository: BlocklistRepository,
|
private readonly blocklistRepository: BlocklistRepository,
|
||||||
private readonly saveMessagesAndCreateContactsService: SaveMessagesAndCreateContactsService,
|
private readonly saveMessagesAndCreateContactsService: SaveMessageAndEmitContactCreationEventService,
|
||||||
@InjectRepository(FeatureFlagEntity, 'core')
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
) {}
|
) {}
|
||||||
|
@ -7,6 +7,7 @@ import { ParticipantWithId } from 'src/modules/messaging/types/gmail-message';
|
|||||||
import { PersonRepository } from 'src/modules/person/repositories/person.repository';
|
import { PersonRepository } from 'src/modules/person/repositories/person.repository';
|
||||||
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
import { PersonObjectMetadata } from 'src/modules/person/standard-objects/person.object-metadata';
|
||||||
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
|
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
|
||||||
|
import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/getFlattenedValuesAndValuesStringForBatchRawQuery.util';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class MessageParticipantService {
|
export class MessageParticipantService {
|
||||||
@ -34,24 +35,29 @@ export class MessageParticipantService {
|
|||||||
transactionManager,
|
transactionManager,
|
||||||
);
|
);
|
||||||
|
|
||||||
const messageParticipantsToUpdate = participants.map((participant) => [
|
const messageParticipantsToUpdate = participants.map((participant) => ({
|
||||||
participant.id,
|
id: participant.id,
|
||||||
participantPersonIds.find(
|
personId: participantPersonIds.find(
|
||||||
(e: { id: string; email: string }) => e.email === participant.handle,
|
(e: { id: string; email: string }) => e.email === participant.handle,
|
||||||
)?.id,
|
)?.id,
|
||||||
]);
|
}));
|
||||||
|
|
||||||
if (messageParticipantsToUpdate.length === 0) return;
|
if (messageParticipantsToUpdate.length === 0) return;
|
||||||
|
|
||||||
const valuesString = messageParticipantsToUpdate
|
const { flattenedValues, valuesString } =
|
||||||
.map((_, index) => `($${index * 2 + 1}::uuid, $${index * 2 + 2}::uuid)`)
|
getFlattenedValuesAndValuesStringForBatchRawQuery(
|
||||||
.join(', ');
|
messageParticipantsToUpdate,
|
||||||
|
{
|
||||||
|
id: 'uuid',
|
||||||
|
personId: 'uuid',
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
await this.workspaceDataSourceService.executeRawQuery(
|
await this.workspaceDataSourceService.executeRawQuery(
|
||||||
`UPDATE ${dataSourceSchema}."messageParticipant" AS "messageParticipant" SET "personId" = "data"."personId"
|
`UPDATE ${dataSourceSchema}."messageParticipant" AS "messageParticipant" SET "personId" = "data"."personId"
|
||||||
FROM (VALUES ${valuesString}) AS "data"("id", "personId")
|
FROM (VALUES ${valuesString}) AS "data"("id", "personId")
|
||||||
WHERE "messageParticipant"."id" = "data"."id"`,
|
WHERE "messageParticipant"."id" = "data"."id"`,
|
||||||
messageParticipantsToUpdate.flat(),
|
flattenedValues,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
transactionManager,
|
transactionManager,
|
||||||
);
|
);
|
||||||
|
@ -2,10 +2,10 @@ 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 { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
|
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
|
||||||
import { CreateCompaniesAndContactsModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company-and-contact/create-company-and-contact.module';
|
import { AutoCompaniesAndContactsCreationModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.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 { MessageModule } from 'src/modules/messaging/services/message/message.module';
|
import { MessageModule } from 'src/modules/messaging/services/message/message.module';
|
||||||
import { SaveMessagesAndCreateContactsService } from 'src/modules/messaging/services/save-message-and-create-contact/save-messages-and-create-contacts.service';
|
import { SaveMessageAndEmitContactCreationEventService } from 'src/modules/messaging/services/save-message-and-emit-contact-creation-event/save-message-and-emit-contact-creation-event.service';
|
||||||
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
||||||
import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
||||||
|
|
||||||
@ -16,11 +16,11 @@ import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard
|
|||||||
MessageChannelObjectMetadata,
|
MessageChannelObjectMetadata,
|
||||||
MessageParticipantObjectMetadata,
|
MessageParticipantObjectMetadata,
|
||||||
]),
|
]),
|
||||||
CreateCompaniesAndContactsModule,
|
AutoCompaniesAndContactsCreationModule,
|
||||||
MessageParticipantModule,
|
MessageParticipantModule,
|
||||||
WorkspaceDataSourceModule,
|
WorkspaceDataSourceModule,
|
||||||
],
|
],
|
||||||
providers: [SaveMessagesAndCreateContactsService],
|
providers: [SaveMessageAndEmitContactCreationEventService],
|
||||||
exports: [SaveMessagesAndCreateContactsService],
|
exports: [SaveMessageAndEmitContactCreationEventService],
|
||||||
})
|
})
|
||||||
export class SaveMessagesAndCreateContactsModule {}
|
export class SaveMessageAndEmitContactCreationEventModule {}
|
@ -1,10 +1,8 @@
|
|||||||
import { Injectable, Logger } from '@nestjs/common';
|
import { Injectable, Logger } from '@nestjs/common';
|
||||||
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
||||||
import { EntityManager } from 'typeorm';
|
|
||||||
|
|
||||||
import { MessageChannelRepository } from 'src/modules/messaging/repositories/message-channel.repository';
|
import { MessageChannelRepository } from 'src/modules/messaging/repositories/message-channel.repository';
|
||||||
import { MessageParticipantRepository } from 'src/modules/messaging/repositories/message-participant.repository';
|
import { MessageParticipantRepository } from 'src/modules/messaging/repositories/message-participant.repository';
|
||||||
import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company-and-contact/create-company-and-contact.service';
|
|
||||||
import {
|
import {
|
||||||
GmailMessage,
|
GmailMessage,
|
||||||
ParticipantWithMessageId,
|
ParticipantWithMessageId,
|
||||||
@ -16,12 +14,11 @@ import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repos
|
|||||||
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
import { MessageChannelObjectMetadata } from 'src/modules/messaging/standard-objects/message-channel.object-metadata';
|
||||||
import { MessageService } from 'src/modules/messaging/services/message/message.service';
|
import { MessageService } from 'src/modules/messaging/services/message/message.service';
|
||||||
import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
import { MessageParticipantObjectMetadata } from 'src/modules/messaging/standard-objects/message-participant.object-metadata';
|
||||||
import { MessageParticipantService } from 'src/modules/messaging/services/message-participant/message-participant.service';
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class SaveMessagesAndCreateContactsService {
|
export class SaveMessageAndEmitContactCreationEventService {
|
||||||
private readonly logger = new Logger(
|
private readonly logger = new Logger(
|
||||||
SaveMessagesAndCreateContactsService.name,
|
SaveMessageAndEmitContactCreationEventService.name,
|
||||||
);
|
);
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@ -30,8 +27,7 @@ export class SaveMessagesAndCreateContactsService {
|
|||||||
private readonly messageChannelRepository: MessageChannelRepository,
|
private readonly messageChannelRepository: MessageChannelRepository,
|
||||||
@InjectObjectMetadataRepository(MessageParticipantObjectMetadata)
|
@InjectObjectMetadataRepository(MessageParticipantObjectMetadata)
|
||||||
private readonly messageParticipantRepository: MessageParticipantRepository,
|
private readonly messageParticipantRepository: MessageParticipantRepository,
|
||||||
private readonly createCompaniesAndContactsService: CreateCompanyAndContactService,
|
private readonly eventEmitter: EventEmitter2,
|
||||||
private readonly messageParticipantService: MessageParticipantService,
|
|
||||||
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@ -80,68 +76,28 @@ export class SaveMessagesAndCreateContactsService {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const isContactAutoCreationEnabled =
|
const participantsWithMessageId: (ParticipantWithMessageId & {
|
||||||
gmailMessageChannel.isContactAutoCreationEnabled;
|
shouldCreateContact: boolean;
|
||||||
|
})[] = messagesToSave.flatMap((message) => {
|
||||||
const participantsWithMessageId: ParticipantWithMessageId[] =
|
|
||||||
messagesToSave.flatMap((message) => {
|
|
||||||
const messageId = messageExternalIdsAndIdsMap.get(message.externalId);
|
const messageId = messageExternalIdsAndIdsMap.get(message.externalId);
|
||||||
|
|
||||||
return messageId
|
return messageId
|
||||||
? message.participants.map((participant) => ({
|
? message.participants.map((participant) => ({
|
||||||
...participant,
|
...participant,
|
||||||
messageId,
|
messageId,
|
||||||
|
shouldCreateContact:
|
||||||
|
gmailMessageChannel.isContactAutoCreationEnabled &&
|
||||||
|
message.participants.find((p) => p.role === 'from')?.handle ===
|
||||||
|
connectedAccount.handle,
|
||||||
}))
|
}))
|
||||||
: [];
|
: [];
|
||||||
});
|
});
|
||||||
|
|
||||||
const contactsToCreate = messagesToSave
|
|
||||||
.filter((message) => connectedAccount.handle === message.fromHandle)
|
|
||||||
.flatMap((message) => message.participants);
|
|
||||||
|
|
||||||
if (isContactAutoCreationEnabled) {
|
|
||||||
startTime = Date.now();
|
|
||||||
|
|
||||||
await workspaceDataSource?.transaction(
|
|
||||||
async (transactionManager: EntityManager) => {
|
|
||||||
await this.createCompaniesAndContactsService.createCompaniesAndContacts(
|
|
||||||
connectedAccount.handle,
|
|
||||||
contactsToCreate,
|
|
||||||
workspaceId,
|
|
||||||
transactionManager,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
const handles = participantsWithMessageId.map(
|
|
||||||
(participant) => participant.handle,
|
|
||||||
);
|
|
||||||
|
|
||||||
const messageParticipantsWithoutPersonIdAndWorkspaceMemberId =
|
|
||||||
await this.messageParticipantRepository.getByHandlesWithoutPersonIdAndWorkspaceMemberId(
|
|
||||||
handles,
|
|
||||||
workspaceId,
|
|
||||||
);
|
|
||||||
|
|
||||||
await this.messageParticipantService.updateMessageParticipantsAfterPeopleCreation(
|
|
||||||
messageParticipantsWithoutPersonIdAndWorkspaceMemberId,
|
|
||||||
workspaceId,
|
|
||||||
);
|
|
||||||
|
|
||||||
endTime = Date.now();
|
|
||||||
|
|
||||||
this.logger.log(
|
|
||||||
`${jobName} creating companies and contacts for workspace ${workspaceId} and account ${
|
|
||||||
connectedAccount.id
|
|
||||||
} in ${endTime - startTime}ms`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
startTime = Date.now();
|
startTime = Date.now();
|
||||||
|
|
||||||
await this.tryToSaveMessageParticipantsOrDeleteMessagesIfError(
|
await this.tryToSaveMessageParticipantsOrDeleteMessagesIfError(
|
||||||
participantsWithMessageId,
|
participantsWithMessageId,
|
||||||
gmailMessageChannelId,
|
gmailMessageChannel,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
connectedAccount,
|
connectedAccount,
|
||||||
jobName,
|
jobName,
|
||||||
@ -157,8 +113,10 @@ export class SaveMessagesAndCreateContactsService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async tryToSaveMessageParticipantsOrDeleteMessagesIfError(
|
private async tryToSaveMessageParticipantsOrDeleteMessagesIfError(
|
||||||
participantsWithMessageId: ParticipantWithMessageId[],
|
participantsWithMessageId: (ParticipantWithMessageId & {
|
||||||
gmailMessageChannelId: string,
|
shouldCreateContact: boolean;
|
||||||
|
})[],
|
||||||
|
gmailMessageChannel: ObjectRecord<MessageChannelObjectMetadata>,
|
||||||
workspaceId: string,
|
workspaceId: string,
|
||||||
connectedAccount: ObjectRecord<ConnectedAccountObjectMetadata>,
|
connectedAccount: ObjectRecord<ConnectedAccountObjectMetadata>,
|
||||||
jobName?: string,
|
jobName?: string,
|
||||||
@ -168,6 +126,18 @@ export class SaveMessagesAndCreateContactsService {
|
|||||||
participantsWithMessageId,
|
participantsWithMessageId,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (gmailMessageChannel.isContactAutoCreationEnabled) {
|
||||||
|
const contactsToCreate = participantsWithMessageId.filter(
|
||||||
|
(participant) => participant.shouldCreateContact,
|
||||||
|
);
|
||||||
|
|
||||||
|
this.eventEmitter.emit(`createContacts`, {
|
||||||
|
workspaceId,
|
||||||
|
connectedAccountHandle: connectedAccount.handle,
|
||||||
|
contactsToCreate,
|
||||||
|
});
|
||||||
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
this.logger.error(
|
this.logger.error(
|
||||||
`${jobName} error saving message participants for workspace ${workspaceId} and account ${connectedAccount.id}`,
|
`${jobName} error saving message participants for workspace ${workspaceId} and account ${connectedAccount.id}`,
|
||||||
@ -180,7 +150,7 @@ export class SaveMessagesAndCreateContactsService {
|
|||||||
|
|
||||||
await this.messageService.deleteMessages(
|
await this.messageService.deleteMessages(
|
||||||
messagesToDelete,
|
messagesToDelete,
|
||||||
gmailMessageChannelId,
|
gmailMessageChannel.id,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
);
|
);
|
||||||
}
|
}
|
@ -1,19 +0,0 @@
|
|||||||
import { uniq, uniqBy } from 'lodash';
|
|
||||||
|
|
||||||
import { Participant } from 'src/modules/messaging/types/gmail-message';
|
|
||||||
export function getUniqueParticipantsAndHandles(participants: Participant[]): {
|
|
||||||
uniqueParticipants: Participant[];
|
|
||||||
uniqueHandles: string[];
|
|
||||||
} {
|
|
||||||
if (participants.length === 0) {
|
|
||||||
return { uniqueParticipants: [], uniqueHandles: [] };
|
|
||||||
}
|
|
||||||
|
|
||||||
const uniqueHandles = uniq(
|
|
||||||
participants.map((participant) => participant.handle),
|
|
||||||
);
|
|
||||||
|
|
||||||
const uniqueParticipants = uniqBy(participants, 'handle');
|
|
||||||
|
|
||||||
return { uniqueParticipants, uniqueHandles };
|
|
||||||
}
|
|
29
yarn.lock
29
yarn.lock
@ -16644,6 +16644,24 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/lodash.uniq@npm:^4.5.9":
|
||||||
|
version: 4.5.9
|
||||||
|
resolution: "@types/lodash.uniq@npm:4.5.9"
|
||||||
|
dependencies:
|
||||||
|
"@types/lodash": "npm:*"
|
||||||
|
checksum: feb01dfd7b6e3d2b4d29bdb0d00cce8ffa685f1c219b34ae531f098811faab11696b7bdfc3654edcf65f6180ad0cda6471cec7da8a9e7688f294b5c8177cfea7
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/lodash.uniqby@npm:^4.7.9":
|
||||||
|
version: 4.7.9
|
||||||
|
resolution: "@types/lodash.uniqby@npm:4.7.9"
|
||||||
|
dependencies:
|
||||||
|
"@types/lodash": "npm:*"
|
||||||
|
checksum: b508927c8bd9a840c629169f8573d0f33a12c7232b4315785537f25914dec441323c54ba609f75fdefaa53f7ecd56f29bd6f598c3920ade45532b8eab76bf044
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@types/lodash.upperfirst@npm:^4.3.7":
|
"@types/lodash.upperfirst@npm:^4.3.7":
|
||||||
version: 4.3.9
|
version: 4.3.9
|
||||||
resolution: "@types/lodash.upperfirst@npm:4.3.9"
|
resolution: "@types/lodash.upperfirst@npm:4.3.9"
|
||||||
@ -33589,6 +33607,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"lodash.uniqby@npm:^4.7.0":
|
||||||
|
version: 4.7.0
|
||||||
|
resolution: "lodash.uniqby@npm:4.7.0"
|
||||||
|
checksum: c505c0de20ca759599a2ba38710e8fb95ff2d2028e24d86c901ef2c74be8056518571b9b754bfb75053b2818d30dd02243e4a4621a6940c206bbb3f7626db656
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"lodash.upperfirst@npm:^4.3.1":
|
"lodash.upperfirst@npm:^4.3.1":
|
||||||
version: 4.3.1
|
version: 4.3.1
|
||||||
resolution: "lodash.upperfirst@npm:4.3.1"
|
resolution: "lodash.upperfirst@npm:4.3.1"
|
||||||
@ -45770,6 +45795,8 @@ __metadata:
|
|||||||
"@types/lodash.isobject": "npm:^3.0.7"
|
"@types/lodash.isobject": "npm:^3.0.7"
|
||||||
"@types/lodash.omit": "npm:^4.5.9"
|
"@types/lodash.omit": "npm:^4.5.9"
|
||||||
"@types/lodash.snakecase": "npm:^4.1.7"
|
"@types/lodash.snakecase": "npm:^4.1.7"
|
||||||
|
"@types/lodash.uniq": "npm:^4.5.9"
|
||||||
|
"@types/lodash.uniqby": "npm:^4.7.9"
|
||||||
"@types/lodash.upperfirst": "npm:^4.3.7"
|
"@types/lodash.upperfirst": "npm:^4.3.7"
|
||||||
"@types/react": "npm:^18.2.39"
|
"@types/react": "npm:^18.2.39"
|
||||||
cache-manager: "npm:^5.4.0"
|
cache-manager: "npm:^5.4.0"
|
||||||
@ -45778,6 +45805,8 @@ __metadata:
|
|||||||
graphql-middleware: "npm:^6.1.35"
|
graphql-middleware: "npm:^6.1.35"
|
||||||
jwt-decode: "npm:^4.0.0"
|
jwt-decode: "npm:^4.0.0"
|
||||||
lodash.differencewith: "npm:^4.5.0"
|
lodash.differencewith: "npm:^4.5.0"
|
||||||
|
lodash.uniq: "npm:^4.5.0"
|
||||||
|
lodash.uniqby: "npm:^4.7.0"
|
||||||
passport: "npm:^0.7.0"
|
passport: "npm:^0.7.0"
|
||||||
psl: "npm:^1.9.0"
|
psl: "npm:^1.9.0"
|
||||||
rimraf: "npm:^5.0.5"
|
rimraf: "npm:^5.0.5"
|
||||||
|
Loading…
Reference in New Issue
Block a user