5748 Create contacts for emails sent and received by email aliases (#5855)
Closes #5748 - Create feature flag - Add scope `https://www.googleapis.com/auth/profile.emails.read` when connecting an account - Get email aliases with google people API, store them in connectedAccount and refresh them before each message-import - Update the contact creation logic accordingly - Refactor --------- Co-authored-by: Charles Bochet <charles@twenty.com>
This commit is contained in:
@ -45,6 +45,11 @@ export const seedFeatureFlags = async (
|
|||||||
workspaceId: workspaceId,
|
workspaceId: workspaceId,
|
||||||
value: true,
|
value: true,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: FeatureFlagKeys.IsMessagingAliasFetchingEnabled,
|
||||||
|
workspaceId: workspaceId,
|
||||||
|
value: true,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: FeatureFlagKeys.IsGoogleCalendarSyncV2Enabled,
|
key: FeatureFlagKeys.IsGoogleCalendarSyncV2Enabled,
|
||||||
workspaceId: workspaceId,
|
workspaceId: workspaceId,
|
||||||
|
|||||||
@ -9,14 +9,14 @@ import {
|
|||||||
|
|
||||||
import { Response } from 'express';
|
import { Response } from 'express';
|
||||||
|
|
||||||
import { GoogleAPIsProviderEnabledGuard } from 'src/engine/core-modules/auth/guards/google-apis-provider-enabled.guard';
|
import { GoogleAPIsOauthRequestCodeGuard } from 'src/engine/core-modules/auth/guards/google-apis-oauth-request-code.guard';
|
||||||
import { GoogleAPIsOauthGuard } from 'src/engine/core-modules/auth/guards/google-apis-oauth.guard';
|
|
||||||
import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/strategies/google-apis.auth.strategy';
|
|
||||||
import { GoogleAPIsService } from 'src/engine/core-modules/auth/services/google-apis.service';
|
import { GoogleAPIsService } from 'src/engine/core-modules/auth/services/google-apis.service';
|
||||||
import { TokenService } from 'src/engine/core-modules/auth/services/token.service';
|
import { TokenService } from 'src/engine/core-modules/auth/services/token.service';
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
import { OnboardingService } from 'src/engine/core-modules/onboarding/onboarding.service';
|
import { OnboardingService } from 'src/engine/core-modules/onboarding/onboarding.service';
|
||||||
import { LoadServiceWithWorkspaceContext } from 'src/engine/twenty-orm/context/load-service-with-workspace.context';
|
import { LoadServiceWithWorkspaceContext } from 'src/engine/twenty-orm/context/load-service-with-workspace.context';
|
||||||
|
import { GoogleAPIsOauthExchangeCodeForTokenGuard } from 'src/engine/core-modules/auth/guards/google-apis-oauth-exchange-code-for-token.guard';
|
||||||
|
import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type';
|
||||||
|
|
||||||
@Controller('auth/google-apis')
|
@Controller('auth/google-apis')
|
||||||
export class GoogleAPIsAuthController {
|
export class GoogleAPIsAuthController {
|
||||||
@ -29,14 +29,14 @@ export class GoogleAPIsAuthController {
|
|||||||
) {}
|
) {}
|
||||||
|
|
||||||
@Get()
|
@Get()
|
||||||
@UseGuards(GoogleAPIsProviderEnabledGuard, GoogleAPIsOauthGuard)
|
@UseGuards(GoogleAPIsOauthRequestCodeGuard)
|
||||||
async googleAuth() {
|
async googleAuth() {
|
||||||
// As this method is protected by Google Auth guard, it will trigger Google SSO flow
|
// As this method is protected by Google Auth guard, it will trigger Google SSO flow
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('get-access-token')
|
@Get('get-access-token')
|
||||||
@UseGuards(GoogleAPIsProviderEnabledGuard, GoogleAPIsOauthGuard)
|
@UseGuards(GoogleAPIsOauthExchangeCodeForTokenGuard)
|
||||||
async googleAuthGetAccessToken(
|
async googleAuthGetAccessToken(
|
||||||
@Req() req: GoogleAPIsRequest,
|
@Req() req: GoogleAPIsRequest,
|
||||||
@Res() res: Response,
|
@Res() res: Response,
|
||||||
@ -44,7 +44,7 @@ export class GoogleAPIsAuthController {
|
|||||||
const { user } = req;
|
const { user } = req;
|
||||||
|
|
||||||
const {
|
const {
|
||||||
email,
|
emails,
|
||||||
accessToken,
|
accessToken,
|
||||||
refreshToken,
|
refreshToken,
|
||||||
transientToken,
|
transientToken,
|
||||||
@ -68,6 +68,8 @@ export class GoogleAPIsAuthController {
|
|||||||
throw new Error('Workspace not found');
|
throw new Error('Workspace not found');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const handle = emails[0].value;
|
||||||
|
|
||||||
const googleAPIsServiceInstance =
|
const googleAPIsServiceInstance =
|
||||||
await this.loadServiceWithWorkspaceContext.load(
|
await this.loadServiceWithWorkspaceContext.load(
|
||||||
this.googleAPIsService,
|
this.googleAPIsService,
|
||||||
@ -75,7 +77,7 @@ export class GoogleAPIsAuthController {
|
|||||||
);
|
);
|
||||||
|
|
||||||
await googleAPIsServiceInstance.refreshGoogleRefreshToken({
|
await googleAPIsServiceInstance.refreshGoogleRefreshToken({
|
||||||
handle: email,
|
handle,
|
||||||
workspaceMemberId: workspaceMemberId,
|
workspaceMemberId: workspaceMemberId,
|
||||||
workspaceId: workspaceId,
|
workspaceId: workspaceId,
|
||||||
accessToken,
|
accessToken,
|
||||||
|
|||||||
@ -0,0 +1,74 @@
|
|||||||
|
import {
|
||||||
|
ExecutionContext,
|
||||||
|
Injectable,
|
||||||
|
NotFoundException,
|
||||||
|
} from '@nestjs/common';
|
||||||
|
import { AuthGuard } from '@nestjs/passport';
|
||||||
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
|
import { Repository } from 'typeorm';
|
||||||
|
|
||||||
|
import { TokenService } from 'src/engine/core-modules/auth/services/token.service';
|
||||||
|
import {
|
||||||
|
GoogleAPIScopeConfig,
|
||||||
|
GoogleAPIsOauthExchangeCodeForTokenStrategy,
|
||||||
|
} from 'src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy';
|
||||||
|
import {
|
||||||
|
FeatureFlagEntity,
|
||||||
|
FeatureFlagKeys,
|
||||||
|
} from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
import { setRequestExtraParams } from 'src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleAPIsOauthExchangeCodeForTokenGuard extends AuthGuard(
|
||||||
|
'google-apis',
|
||||||
|
) {
|
||||||
|
constructor(
|
||||||
|
private readonly environmentService: EnvironmentService,
|
||||||
|
private readonly tokenService: TokenService,
|
||||||
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
|
) {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
async canActivate(context: ExecutionContext) {
|
||||||
|
const request = context.switchToHttp().getRequest();
|
||||||
|
const state = JSON.parse(request.query.state);
|
||||||
|
|
||||||
|
if (
|
||||||
|
!this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED') &&
|
||||||
|
!this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED')
|
||||||
|
) {
|
||||||
|
throw new NotFoundException('Google apis auth is not enabled');
|
||||||
|
}
|
||||||
|
|
||||||
|
const { workspaceId } = await this.tokenService.verifyTransientToken(
|
||||||
|
state.transientToken,
|
||||||
|
);
|
||||||
|
|
||||||
|
const scopeConfig: GoogleAPIScopeConfig = {
|
||||||
|
isMessagingAliasFetchingEnabled:
|
||||||
|
!!(await this.featureFlagRepository.findOneBy({
|
||||||
|
workspaceId,
|
||||||
|
key: FeatureFlagKeys.IsMessagingAliasFetchingEnabled,
|
||||||
|
value: true,
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
|
||||||
|
new GoogleAPIsOauthExchangeCodeForTokenStrategy(
|
||||||
|
this.environmentService,
|
||||||
|
scopeConfig,
|
||||||
|
);
|
||||||
|
|
||||||
|
setRequestExtraParams(request, {
|
||||||
|
transientToken: state.transientToken,
|
||||||
|
redirectLocation: state.redirectLocation,
|
||||||
|
calendarVisibility: state.calendarVisibility,
|
||||||
|
messageVisibility: state.messageVisibility,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (await super.canActivate(context)) as boolean;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,72 @@
|
|||||||
|
import {
|
||||||
|
ExecutionContext,
|
||||||
|
Injectable,
|
||||||
|
NotFoundException,
|
||||||
|
} from '@nestjs/common';
|
||||||
|
import { AuthGuard } from '@nestjs/passport';
|
||||||
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
|
import { Repository } from 'typeorm';
|
||||||
|
|
||||||
|
import { TokenService } from 'src/engine/core-modules/auth/services/token.service';
|
||||||
|
import { GoogleAPIScopeConfig } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy';
|
||||||
|
import {
|
||||||
|
FeatureFlagEntity,
|
||||||
|
FeatureFlagKeys,
|
||||||
|
} from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
import { GoogleAPIsOauthRequestCodeStrategy } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-request-code.auth.strategy';
|
||||||
|
import { setRequestExtraParams } from 'src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleAPIsOauthRequestCodeGuard extends AuthGuard('google-apis') {
|
||||||
|
constructor(
|
||||||
|
private readonly environmentService: EnvironmentService,
|
||||||
|
private readonly tokenService: TokenService,
|
||||||
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
|
) {
|
||||||
|
super({
|
||||||
|
prompt: 'select_account',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async canActivate(context: ExecutionContext) {
|
||||||
|
const request = context.switchToHttp().getRequest();
|
||||||
|
|
||||||
|
if (
|
||||||
|
!this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED') &&
|
||||||
|
!this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED')
|
||||||
|
) {
|
||||||
|
throw new NotFoundException('Google apis auth is not enabled');
|
||||||
|
}
|
||||||
|
|
||||||
|
const { workspaceId } = await this.tokenService.verifyTransientToken(
|
||||||
|
request.query.transientToken,
|
||||||
|
);
|
||||||
|
|
||||||
|
const scopeConfig: GoogleAPIScopeConfig = {
|
||||||
|
isMessagingAliasFetchingEnabled:
|
||||||
|
!!(await this.featureFlagRepository.findOneBy({
|
||||||
|
workspaceId,
|
||||||
|
key: FeatureFlagKeys.IsMessagingAliasFetchingEnabled,
|
||||||
|
value: true,
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
|
||||||
|
new GoogleAPIsOauthRequestCodeStrategy(
|
||||||
|
this.environmentService,
|
||||||
|
scopeConfig,
|
||||||
|
);
|
||||||
|
setRequestExtraParams(request, {
|
||||||
|
transientToken: request.query.transientToken,
|
||||||
|
redirectLocation: request.query.redirectLocation,
|
||||||
|
calendarVisibility: request.query.calendarVisibility,
|
||||||
|
messageVisibility: request.query.messageVisibility,
|
||||||
|
});
|
||||||
|
|
||||||
|
const activate = (await super.canActivate(context)) as boolean;
|
||||||
|
|
||||||
|
return activate;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,39 +0,0 @@
|
|||||||
import { ExecutionContext, Injectable } from '@nestjs/common';
|
|
||||||
import { AuthGuard } from '@nestjs/passport';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class GoogleAPIsOauthGuard extends AuthGuard('google-apis') {
|
|
||||||
constructor() {
|
|
||||||
super({
|
|
||||||
prompt: 'select_account',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async canActivate(context: ExecutionContext) {
|
|
||||||
const request = context.switchToHttp().getRequest();
|
|
||||||
const transientToken = request.query.transientToken;
|
|
||||||
const redirectLocation = request.query.redirectLocation;
|
|
||||||
const calendarVisibility = request.query.calendarVisibility;
|
|
||||||
const messageVisibility = request.query.messageVisibility;
|
|
||||||
|
|
||||||
if (transientToken && typeof transientToken === 'string') {
|
|
||||||
request.params.transientToken = transientToken;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (redirectLocation && typeof redirectLocation === 'string') {
|
|
||||||
request.params.redirectLocation = redirectLocation;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (calendarVisibility && typeof calendarVisibility === 'string') {
|
|
||||||
request.params.calendarVisibility = calendarVisibility;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (messageVisibility && typeof messageVisibility === 'string') {
|
|
||||||
request.params.messageVisibility = messageVisibility;
|
|
||||||
}
|
|
||||||
|
|
||||||
const activate = (await super.canActivate(context)) as boolean;
|
|
||||||
|
|
||||||
return activate;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -1,41 +0,0 @@
|
|||||||
import { Injectable, CanActivate, NotFoundException } from '@nestjs/common';
|
|
||||||
import { InjectRepository } from '@nestjs/typeorm';
|
|
||||||
|
|
||||||
import { Repository } from 'typeorm';
|
|
||||||
|
|
||||||
import { TokenService } from 'src/engine/core-modules/auth/services/token.service';
|
|
||||||
import {
|
|
||||||
GoogleAPIScopeConfig,
|
|
||||||
GoogleAPIsStrategy,
|
|
||||||
} from 'src/engine/core-modules/auth/strategies/google-apis.auth.strategy';
|
|
||||||
import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class GoogleAPIsProviderEnabledGuard implements CanActivate {
|
|
||||||
constructor(
|
|
||||||
private readonly environmentService: EnvironmentService,
|
|
||||||
private readonly tokenService: TokenService,
|
|
||||||
@InjectRepository(FeatureFlagEntity, 'core')
|
|
||||||
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
|
||||||
) {}
|
|
||||||
|
|
||||||
async canActivate(): Promise<boolean> {
|
|
||||||
if (
|
|
||||||
!this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED') &&
|
|
||||||
!this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED')
|
|
||||||
) {
|
|
||||||
throw new NotFoundException('Google apis auth is not enabled');
|
|
||||||
}
|
|
||||||
|
|
||||||
const scopeConfig: GoogleAPIScopeConfig = {
|
|
||||||
isCalendarEnabled: !!this.environmentService.get(
|
|
||||||
'MESSAGING_PROVIDER_GMAIL_ENABLED',
|
|
||||||
),
|
|
||||||
};
|
|
||||||
|
|
||||||
new GoogleAPIsStrategy(this.environmentService, scopeConfig);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -0,0 +1,41 @@
|
|||||||
|
import { PassportStrategy } from '@nestjs/passport';
|
||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { Strategy } from 'passport-google-oauth20';
|
||||||
|
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
|
||||||
|
export type GoogleAPIScopeConfig = {
|
||||||
|
isCalendarEnabled?: boolean;
|
||||||
|
isMessagingAliasFetchingEnabled?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleAPIsOauthCommonStrategy extends PassportStrategy(
|
||||||
|
Strategy,
|
||||||
|
'google-apis',
|
||||||
|
) {
|
||||||
|
constructor(
|
||||||
|
environmentService: EnvironmentService,
|
||||||
|
scopeConfig: GoogleAPIScopeConfig,
|
||||||
|
) {
|
||||||
|
const scopes = [
|
||||||
|
'email',
|
||||||
|
'profile',
|
||||||
|
'https://www.googleapis.com/auth/gmail.readonly',
|
||||||
|
'https://www.googleapis.com/auth/calendar.events',
|
||||||
|
];
|
||||||
|
|
||||||
|
if (scopeConfig?.isMessagingAliasFetchingEnabled) {
|
||||||
|
scopes.push('https://www.googleapis.com/auth/profile.emails.read');
|
||||||
|
}
|
||||||
|
|
||||||
|
super({
|
||||||
|
clientID: environmentService.get('AUTH_GOOGLE_CLIENT_ID'),
|
||||||
|
clientSecret: environmentService.get('AUTH_GOOGLE_CLIENT_SECRET'),
|
||||||
|
callbackURL: environmentService.get('AUTH_GOOGLE_APIS_CALLBACK_URL'),
|
||||||
|
scope: scopes,
|
||||||
|
passReqToCallback: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,52 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { VerifyCallback } from 'passport-google-oauth20';
|
||||||
|
|
||||||
|
import { GoogleAPIsOauthCommonStrategy } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy';
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type';
|
||||||
|
|
||||||
|
export type GoogleAPIScopeConfig = {
|
||||||
|
isCalendarEnabled?: boolean;
|
||||||
|
isMessagingAliasFetchingEnabled?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleAPIsOauthExchangeCodeForTokenStrategy extends GoogleAPIsOauthCommonStrategy {
|
||||||
|
constructor(
|
||||||
|
environmentService: EnvironmentService,
|
||||||
|
scopeConfig: GoogleAPIScopeConfig,
|
||||||
|
) {
|
||||||
|
super(environmentService, scopeConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
async validate(
|
||||||
|
request: GoogleAPIsRequest,
|
||||||
|
accessToken: string,
|
||||||
|
refreshToken: string,
|
||||||
|
profile: any,
|
||||||
|
done: VerifyCallback,
|
||||||
|
): Promise<void> {
|
||||||
|
const { name, emails, photos } = profile;
|
||||||
|
|
||||||
|
const state =
|
||||||
|
typeof request.query.state === 'string'
|
||||||
|
? JSON.parse(request.query.state)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
const user: GoogleAPIsRequest['user'] = {
|
||||||
|
emails,
|
||||||
|
firstName: name.givenName,
|
||||||
|
lastName: name.familyName,
|
||||||
|
picture: photos?.[0]?.value,
|
||||||
|
accessToken,
|
||||||
|
refreshToken,
|
||||||
|
transientToken: state.transientToken,
|
||||||
|
redirectLocation: state.redirectLocation,
|
||||||
|
calendarVisibility: state.calendarVisibility,
|
||||||
|
messageVisibility: state.messageVisibility,
|
||||||
|
};
|
||||||
|
|
||||||
|
done(null, user);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,35 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { GoogleAPIsOauthCommonStrategy } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy';
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
|
||||||
|
export type GoogleAPIScopeConfig = {
|
||||||
|
isCalendarEnabled?: boolean;
|
||||||
|
isMessagingAliasFetchingEnabled?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleAPIsOauthRequestCodeStrategy extends GoogleAPIsOauthCommonStrategy {
|
||||||
|
constructor(
|
||||||
|
environmentService: EnvironmentService,
|
||||||
|
scopeConfig: GoogleAPIScopeConfig,
|
||||||
|
) {
|
||||||
|
super(environmentService, scopeConfig);
|
||||||
|
}
|
||||||
|
|
||||||
|
authenticate(req: any, options: any) {
|
||||||
|
options = {
|
||||||
|
...options,
|
||||||
|
accessType: 'offline',
|
||||||
|
prompt: 'consent',
|
||||||
|
state: JSON.stringify({
|
||||||
|
transientToken: req.params.transientToken,
|
||||||
|
redirectLocation: req.params.redirectLocation,
|
||||||
|
calendarVisibility: req.params.calendarVisibility,
|
||||||
|
messageVisibility: req.params.messageVisibility,
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
|
||||||
|
return super.authenticate(req, options);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,110 +0,0 @@
|
|||||||
import { PassportStrategy } from '@nestjs/passport';
|
|
||||||
import { Injectable } from '@nestjs/common';
|
|
||||||
|
|
||||||
import { Strategy, VerifyCallback } from 'passport-google-oauth20';
|
|
||||||
import { Request } from 'express';
|
|
||||||
|
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
|
||||||
import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity';
|
|
||||||
import { MessageChannelVisibility } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity';
|
|
||||||
|
|
||||||
export type GoogleAPIsRequest = Omit<
|
|
||||||
Request,
|
|
||||||
'user' | 'workspace' | 'cacheVersion'
|
|
||||||
> & {
|
|
||||||
user: {
|
|
||||||
firstName?: string | null;
|
|
||||||
lastName?: string | null;
|
|
||||||
email: string;
|
|
||||||
picture: string | null;
|
|
||||||
workspaceInviteHash?: string;
|
|
||||||
accessToken: string;
|
|
||||||
refreshToken: string;
|
|
||||||
transientToken: string;
|
|
||||||
redirectLocation?: string;
|
|
||||||
calendarVisibility?: CalendarChannelVisibility;
|
|
||||||
messageVisibility?: MessageChannelVisibility;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export type GoogleAPIScopeConfig = {
|
|
||||||
isCalendarEnabled?: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
@Injectable()
|
|
||||||
export class GoogleAPIsStrategy extends PassportStrategy(
|
|
||||||
Strategy,
|
|
||||||
'google-apis',
|
|
||||||
) {
|
|
||||||
constructor(
|
|
||||||
environmentService: EnvironmentService,
|
|
||||||
scopeConfig: GoogleAPIScopeConfig,
|
|
||||||
) {
|
|
||||||
const scope = ['email', 'profile'];
|
|
||||||
|
|
||||||
if (environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED')) {
|
|
||||||
scope.push('https://www.googleapis.com/auth/gmail.readonly');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') &&
|
|
||||||
scopeConfig?.isCalendarEnabled
|
|
||||||
) {
|
|
||||||
scope.push('https://www.googleapis.com/auth/calendar.events');
|
|
||||||
}
|
|
||||||
|
|
||||||
super({
|
|
||||||
clientID: environmentService.get('AUTH_GOOGLE_CLIENT_ID'),
|
|
||||||
clientSecret: environmentService.get('AUTH_GOOGLE_CLIENT_SECRET'),
|
|
||||||
callbackURL: environmentService.get('AUTH_GOOGLE_APIS_CALLBACK_URL'),
|
|
||||||
scope,
|
|
||||||
passReqToCallback: true,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
authenticate(req: any, options: any) {
|
|
||||||
options = {
|
|
||||||
...options,
|
|
||||||
accessType: 'offline',
|
|
||||||
prompt: 'consent',
|
|
||||||
state: JSON.stringify({
|
|
||||||
transientToken: req.params.transientToken,
|
|
||||||
redirectLocation: req.params.redirectLocation,
|
|
||||||
calendarVisibility: req.params.calendarVisibility,
|
|
||||||
messageVisibility: req.params.messageVisibility,
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
return super.authenticate(req, options);
|
|
||||||
}
|
|
||||||
|
|
||||||
async validate(
|
|
||||||
request: GoogleAPIsRequest,
|
|
||||||
accessToken: string,
|
|
||||||
refreshToken: string,
|
|
||||||
profile: any,
|
|
||||||
done: VerifyCallback,
|
|
||||||
): Promise<void> {
|
|
||||||
const { name, emails, photos } = profile;
|
|
||||||
|
|
||||||
const state =
|
|
||||||
typeof request.query.state === 'string'
|
|
||||||
? JSON.parse(request.query.state)
|
|
||||||
: undefined;
|
|
||||||
|
|
||||||
const user: GoogleAPIsRequest['user'] = {
|
|
||||||
email: emails[0].value,
|
|
||||||
firstName: name.givenName,
|
|
||||||
lastName: name.familyName,
|
|
||||||
picture: photos?.[0]?.value,
|
|
||||||
accessToken,
|
|
||||||
refreshToken,
|
|
||||||
transientToken: state.transientToken,
|
|
||||||
redirectLocation: state.redirectLocation,
|
|
||||||
calendarVisibility: state.calendarVisibility,
|
|
||||||
messageVisibility: state.messageVisibility,
|
|
||||||
};
|
|
||||||
|
|
||||||
done(null, user);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -0,0 +1,23 @@
|
|||||||
|
import { Request } from 'express';
|
||||||
|
|
||||||
|
import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity';
|
||||||
|
import { MessageChannelVisibility } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity';
|
||||||
|
|
||||||
|
export type GoogleAPIsRequest = Omit<
|
||||||
|
Request,
|
||||||
|
'user' | 'workspace' | 'cacheVersion'
|
||||||
|
> & {
|
||||||
|
user: {
|
||||||
|
firstName?: string | null;
|
||||||
|
lastName?: string | null;
|
||||||
|
emails: { value: string }[];
|
||||||
|
picture: string | null;
|
||||||
|
workspaceInviteHash?: string;
|
||||||
|
accessToken: string;
|
||||||
|
refreshToken: string;
|
||||||
|
transientToken: string;
|
||||||
|
redirectLocation?: string;
|
||||||
|
calendarVisibility?: CalendarChannelVisibility;
|
||||||
|
messageVisibility?: MessageChannelVisibility;
|
||||||
|
};
|
||||||
|
};
|
||||||
@ -0,0 +1,40 @@
|
|||||||
|
import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type';
|
||||||
|
import { setRequestExtraParams } from 'src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util';
|
||||||
|
import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity';
|
||||||
|
import { MessageChannelVisibility } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity';
|
||||||
|
|
||||||
|
describe('googleApisSetRequestExtraParams', () => {
|
||||||
|
it('should set request extra params', () => {
|
||||||
|
const request = {
|
||||||
|
params: {},
|
||||||
|
} as GoogleAPIsRequest;
|
||||||
|
|
||||||
|
setRequestExtraParams(request, {
|
||||||
|
transientToken: 'abc',
|
||||||
|
redirectLocation: '/test',
|
||||||
|
calendarVisibility: CalendarChannelVisibility.SHARE_EVERYTHING,
|
||||||
|
messageVisibility: MessageChannelVisibility.SHARE_EVERYTHING,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(request.params).toEqual({
|
||||||
|
transientToken: 'abc',
|
||||||
|
redirectLocation: '/test',
|
||||||
|
calendarVisibility: CalendarChannelVisibility.SHARE_EVERYTHING,
|
||||||
|
messageVisibility: MessageChannelVisibility.SHARE_EVERYTHING,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should throw error if transientToken is not provided', () => {
|
||||||
|
const request = {
|
||||||
|
params: {},
|
||||||
|
} as GoogleAPIsRequest;
|
||||||
|
|
||||||
|
expect(() => {
|
||||||
|
setRequestExtraParams(request, {
|
||||||
|
redirectLocation: '/test',
|
||||||
|
calendarVisibility: CalendarChannelVisibility.SHARE_EVERYTHING,
|
||||||
|
messageVisibility: MessageChannelVisibility.SHARE_EVERYTHING,
|
||||||
|
});
|
||||||
|
}).toThrow('transientToken is required');
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,38 @@
|
|||||||
|
import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type';
|
||||||
|
|
||||||
|
type GoogleAPIsRequestExtraParams = {
|
||||||
|
transientToken?: string;
|
||||||
|
redirectLocation?: string;
|
||||||
|
calendarVisibility?: string;
|
||||||
|
messageVisibility?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setRequestExtraParams = (
|
||||||
|
request: GoogleAPIsRequest,
|
||||||
|
params: GoogleAPIsRequestExtraParams,
|
||||||
|
): void => {
|
||||||
|
const {
|
||||||
|
transientToken,
|
||||||
|
redirectLocation,
|
||||||
|
calendarVisibility,
|
||||||
|
messageVisibility,
|
||||||
|
} = params;
|
||||||
|
|
||||||
|
if (!transientToken) {
|
||||||
|
throw new Error('transientToken is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
request.params.transientToken = transientToken;
|
||||||
|
|
||||||
|
if (redirectLocation) {
|
||||||
|
request.params.redirectLocation = redirectLocation;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (calendarVisibility) {
|
||||||
|
request.params.calendarVisibility = calendarVisibility;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (messageVisibility) {
|
||||||
|
request.params.messageVisibility = messageVisibility;
|
||||||
|
}
|
||||||
|
};
|
||||||
@ -11,9 +11,9 @@ import {
|
|||||||
import { Response } from 'express';
|
import { Response } from 'express';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
BillingService,
|
BillingWorkspaceService,
|
||||||
WebhookEvent,
|
WebhookEvent,
|
||||||
} from 'src/engine/core-modules/billing/billing.service';
|
} from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service';
|
import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service';
|
||||||
|
|
||||||
@Controller('billing')
|
@Controller('billing')
|
||||||
@ -22,7 +22,7 @@ export class BillingController {
|
|||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private readonly stripeService: StripeService,
|
private readonly stripeService: StripeService,
|
||||||
private readonly billingService: BillingService,
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@Post('/webhooks')
|
@Post('/webhooks')
|
||||||
@ -42,7 +42,7 @@ export class BillingController {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (event.type === WebhookEvent.SETUP_INTENT_SUCCEEDED) {
|
if (event.type === WebhookEvent.SETUP_INTENT_SUCCEEDED) {
|
||||||
await this.billingService.handleUnpaidInvoices(event.data);
|
await this.billingWorkspaceService.handleUnpaidInvoices(event.data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
@ -58,7 +58,7 @@ export class BillingController {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.billingService.upsertBillingSubscription(
|
await this.billingWorkspaceService.upsertBillingSubscription(
|
||||||
workspaceId,
|
workspaceId,
|
||||||
event.data,
|
event.data,
|
||||||
);
|
);
|
||||||
|
|||||||
@ -11,6 +11,7 @@ import { BillingResolver } from 'src/engine/core-modules/billing/billing.resolve
|
|||||||
import { BillingWorkspaceMemberListener } from 'src/engine/core-modules/billing/listeners/billing-workspace-member.listener';
|
import { BillingWorkspaceMemberListener } from 'src/engine/core-modules/billing/listeners/billing-workspace-member.listener';
|
||||||
import { UserWorkspaceModule } from 'src/engine/core-modules/user-workspace/user-workspace.module';
|
import { UserWorkspaceModule } from 'src/engine/core-modules/user-workspace/user-workspace.module';
|
||||||
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 { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [
|
imports: [
|
||||||
@ -27,7 +28,12 @@ import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-
|
|||||||
),
|
),
|
||||||
],
|
],
|
||||||
controllers: [BillingController],
|
controllers: [BillingController],
|
||||||
providers: [BillingService, BillingResolver, BillingWorkspaceMemberListener],
|
providers: [
|
||||||
exports: [BillingService],
|
BillingService,
|
||||||
|
BillingWorkspaceService,
|
||||||
|
BillingResolver,
|
||||||
|
BillingWorkspaceMemberListener,
|
||||||
|
],
|
||||||
|
exports: [BillingService, BillingWorkspaceService],
|
||||||
})
|
})
|
||||||
export class BillingModule {}
|
export class BillingModule {}
|
||||||
|
|||||||
@ -3,8 +3,8 @@ import { UseGuards } from '@nestjs/common';
|
|||||||
|
|
||||||
import {
|
import {
|
||||||
AvailableProduct,
|
AvailableProduct,
|
||||||
BillingService,
|
BillingWorkspaceService,
|
||||||
} from 'src/engine/core-modules/billing/billing.service';
|
} from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
import { ProductInput } from 'src/engine/core-modules/billing/dto/product.input';
|
import { ProductInput } from 'src/engine/core-modules/billing/dto/product.input';
|
||||||
import { assert } from 'src/utils/assert';
|
import { assert } from 'src/utils/assert';
|
||||||
import { ProductPricesEntity } from 'src/engine/core-modules/billing/dto/product-prices.entity';
|
import { ProductPricesEntity } from 'src/engine/core-modules/billing/dto/product-prices.entity';
|
||||||
@ -18,11 +18,14 @@ import { UpdateBillingEntity } from 'src/engine/core-modules/billing/dto/update-
|
|||||||
|
|
||||||
@Resolver()
|
@Resolver()
|
||||||
export class BillingResolver {
|
export class BillingResolver {
|
||||||
constructor(private readonly billingService: BillingService) {}
|
constructor(
|
||||||
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
|
) {}
|
||||||
|
|
||||||
@Query(() => ProductPricesEntity)
|
@Query(() => ProductPricesEntity)
|
||||||
async getProductPrices(@Args() { product }: ProductInput) {
|
async getProductPrices(@Args() { product }: ProductInput) {
|
||||||
const stripeProductId = this.billingService.getProductStripeId(product);
|
const stripeProductId =
|
||||||
|
this.billingWorkspaceService.getProductStripeId(product);
|
||||||
|
|
||||||
assert(
|
assert(
|
||||||
stripeProductId,
|
stripeProductId,
|
||||||
@ -32,7 +35,7 @@ export class BillingResolver {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const productPrices =
|
const productPrices =
|
||||||
await this.billingService.getProductPrices(stripeProductId);
|
await this.billingWorkspaceService.getProductPrices(stripeProductId);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
totalNumberOfPrices: productPrices.length,
|
totalNumberOfPrices: productPrices.length,
|
||||||
@ -47,7 +50,7 @@ export class BillingResolver {
|
|||||||
@Args() { returnUrlPath }: BillingSessionInput,
|
@Args() { returnUrlPath }: BillingSessionInput,
|
||||||
) {
|
) {
|
||||||
return {
|
return {
|
||||||
url: await this.billingService.computeBillingPortalSessionURL(
|
url: await this.billingWorkspaceService.computeBillingPortalSessionURL(
|
||||||
user.defaultWorkspaceId,
|
user.defaultWorkspaceId,
|
||||||
returnUrlPath,
|
returnUrlPath,
|
||||||
),
|
),
|
||||||
@ -60,7 +63,7 @@ export class BillingResolver {
|
|||||||
@AuthUser() user: User,
|
@AuthUser() user: User,
|
||||||
@Args() { recurringInterval, successUrlPath }: CheckoutSessionInput,
|
@Args() { recurringInterval, successUrlPath }: CheckoutSessionInput,
|
||||||
) {
|
) {
|
||||||
const stripeProductId = this.billingService.getProductStripeId(
|
const stripeProductId = this.billingWorkspaceService.getProductStripeId(
|
||||||
AvailableProduct.BasePlan,
|
AvailableProduct.BasePlan,
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -70,7 +73,7 @@ export class BillingResolver {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const productPrices =
|
const productPrices =
|
||||||
await this.billingService.getProductPrices(stripeProductId);
|
await this.billingWorkspaceService.getProductPrices(stripeProductId);
|
||||||
|
|
||||||
const stripePriceId = productPrices.filter(
|
const stripePriceId = productPrices.filter(
|
||||||
(price) => price.recurringInterval === recurringInterval,
|
(price) => price.recurringInterval === recurringInterval,
|
||||||
@ -82,7 +85,7 @@ export class BillingResolver {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
url: await this.billingService.computeCheckoutSessionURL(
|
url: await this.billingWorkspaceService.computeCheckoutSessionURL(
|
||||||
user,
|
user,
|
||||||
stripePriceId,
|
stripePriceId,
|
||||||
successUrlPath,
|
successUrlPath,
|
||||||
@ -93,7 +96,7 @@ export class BillingResolver {
|
|||||||
@Mutation(() => UpdateBillingEntity)
|
@Mutation(() => UpdateBillingEntity)
|
||||||
@UseGuards(JwtAuthGuard)
|
@UseGuards(JwtAuthGuard)
|
||||||
async updateBillingSubscription(@AuthUser() user: User) {
|
async updateBillingSubscription(@AuthUser() user: User) {
|
||||||
await this.billingService.updateBillingSubscription(user);
|
await this.billingWorkspaceService.updateBillingSubscription(user);
|
||||||
|
|
||||||
return { success: true };
|
return { success: true };
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,51 +1,21 @@
|
|||||||
import { Injectable, Logger } from '@nestjs/common';
|
import { Injectable, Logger } from '@nestjs/common';
|
||||||
import { InjectRepository } from '@nestjs/typeorm';
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
import Stripe from 'stripe';
|
import { In, Repository } from 'typeorm';
|
||||||
import { In, Not, Repository } from 'typeorm';
|
|
||||||
|
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service';
|
|
||||||
import {
|
import {
|
||||||
BillingSubscription,
|
BillingSubscription,
|
||||||
SubscriptionInterval,
|
|
||||||
SubscriptionStatus,
|
SubscriptionStatus,
|
||||||
} from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
} from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
||||||
import { BillingSubscriptionItem } from 'src/engine/core-modules/billing/entities/billing-subscription-item.entity';
|
|
||||||
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
|
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
|
||||||
import { ProductPriceEntity } from 'src/engine/core-modules/billing/dto/product-price.entity';
|
|
||||||
import { User } from 'src/engine/core-modules/user/user.entity';
|
|
||||||
import { assert } from 'src/utils/assert';
|
|
||||||
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
|
||||||
import {
|
|
||||||
FeatureFlagEntity,
|
|
||||||
FeatureFlagKeys,
|
|
||||||
} from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
|
||||||
|
|
||||||
export enum AvailableProduct {
|
|
||||||
BasePlan = 'base-plan',
|
|
||||||
}
|
|
||||||
|
|
||||||
export enum WebhookEvent {
|
|
||||||
CUSTOMER_SUBSCRIPTION_CREATED = 'customer.subscription.created',
|
|
||||||
CUSTOMER_SUBSCRIPTION_UPDATED = 'customer.subscription.updated',
|
|
||||||
CUSTOMER_SUBSCRIPTION_DELETED = 'customer.subscription.deleted',
|
|
||||||
SETUP_INTENT_SUCCEEDED = 'setup_intent.succeeded',
|
|
||||||
}
|
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class BillingService {
|
export class BillingService {
|
||||||
protected readonly logger = new Logger(BillingService.name);
|
protected readonly logger = new Logger(BillingService.name);
|
||||||
constructor(
|
constructor(
|
||||||
private readonly stripeService: StripeService,
|
|
||||||
private readonly userWorkspaceService: UserWorkspaceService,
|
|
||||||
private readonly environmentService: EnvironmentService,
|
private readonly environmentService: EnvironmentService,
|
||||||
@InjectRepository(BillingSubscription, 'core')
|
@InjectRepository(BillingSubscription, 'core')
|
||||||
private readonly billingSubscriptionRepository: Repository<BillingSubscription>,
|
|
||||||
@InjectRepository(FeatureFlagEntity, 'core')
|
|
||||||
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
|
||||||
@InjectRepository(BillingSubscriptionItem, 'core')
|
|
||||||
private readonly billingSubscriptionItemRepository: Repository<BillingSubscriptionItem>,
|
|
||||||
@InjectRepository(Workspace, 'core')
|
@InjectRepository(Workspace, 'core')
|
||||||
private readonly workspaceRepository: Repository<Workspace>,
|
private readonly workspaceRepository: Repository<Workspace>,
|
||||||
) {}
|
) {}
|
||||||
@ -68,283 +38,4 @@ export class BillingService {
|
|||||||
})
|
})
|
||||||
).map((workspace) => workspace.id);
|
).map((workspace) => workspace.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
async isBillingEnabledForWorkspace(workspaceId: string) {
|
|
||||||
const isFreeAccessEnabled = await this.featureFlagRepository.findOneBy({
|
|
||||||
workspaceId,
|
|
||||||
key: FeatureFlagKeys.IsFreeAccessEnabled,
|
|
||||||
value: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
|
||||||
!isFreeAccessEnabled && this.environmentService.get('IS_BILLING_ENABLED')
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
getProductStripeId(product: AvailableProduct) {
|
|
||||||
if (product === AvailableProduct.BasePlan) {
|
|
||||||
return this.environmentService.get('BILLING_STRIPE_BASE_PLAN_PRODUCT_ID');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getProductPrices(stripeProductId: string) {
|
|
||||||
const productPrices =
|
|
||||||
await this.stripeService.getProductPrices(stripeProductId);
|
|
||||||
|
|
||||||
return this.formatProductPrices(productPrices.data);
|
|
||||||
}
|
|
||||||
|
|
||||||
formatProductPrices(prices: Stripe.Price[]) {
|
|
||||||
const result: Record<string, ProductPriceEntity> = {};
|
|
||||||
|
|
||||||
prices.forEach((item) => {
|
|
||||||
const interval = item.recurring?.interval;
|
|
||||||
|
|
||||||
if (!interval || !item.unit_amount) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (
|
|
||||||
!result[interval] ||
|
|
||||||
item.created > (result[interval]?.created || 0)
|
|
||||||
) {
|
|
||||||
result[interval] = {
|
|
||||||
unitAmount: item.unit_amount,
|
|
||||||
recurringInterval: interval,
|
|
||||||
created: item.created,
|
|
||||||
stripePriceId: item.id,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return Object.values(result).sort((a, b) => a.unitAmount - b.unitAmount);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getCurrentBillingSubscription(criteria: {
|
|
||||||
workspaceId?: string;
|
|
||||||
stripeCustomerId?: string;
|
|
||||||
}) {
|
|
||||||
const notCanceledSubscriptions =
|
|
||||||
await this.billingSubscriptionRepository.find({
|
|
||||||
where: { ...criteria, status: Not(SubscriptionStatus.Canceled) },
|
|
||||||
relations: ['billingSubscriptionItems'],
|
|
||||||
});
|
|
||||||
|
|
||||||
assert(
|
|
||||||
notCanceledSubscriptions.length <= 1,
|
|
||||||
`More than one not canceled subscription for workspace ${criteria.workspaceId}`,
|
|
||||||
);
|
|
||||||
|
|
||||||
return notCanceledSubscriptions?.[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
async getBillingSubscription(stripeSubscriptionId: string) {
|
|
||||||
return this.billingSubscriptionRepository.findOneOrFail({
|
|
||||||
where: { stripeSubscriptionId },
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async getStripeCustomerId(workspaceId: string) {
|
|
||||||
const subscriptions = await this.billingSubscriptionRepository.find({
|
|
||||||
where: { workspaceId },
|
|
||||||
});
|
|
||||||
|
|
||||||
return subscriptions?.[0]?.stripeCustomerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getBillingSubscriptionItem(
|
|
||||||
workspaceId: string,
|
|
||||||
stripeProductId = this.environmentService.get(
|
|
||||||
'BILLING_STRIPE_BASE_PLAN_PRODUCT_ID',
|
|
||||||
),
|
|
||||||
) {
|
|
||||||
const billingSubscription = await this.getCurrentBillingSubscription({
|
|
||||||
workspaceId,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!billingSubscription) {
|
|
||||||
throw new Error(
|
|
||||||
`Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const billingSubscriptionItem =
|
|
||||||
billingSubscription.billingSubscriptionItems.filter(
|
|
||||||
(billingSubscriptionItem) =>
|
|
||||||
billingSubscriptionItem.stripeProductId === stripeProductId,
|
|
||||||
)?.[0];
|
|
||||||
|
|
||||||
if (!billingSubscriptionItem) {
|
|
||||||
throw new Error(
|
|
||||||
`Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return billingSubscriptionItem;
|
|
||||||
}
|
|
||||||
|
|
||||||
async computeBillingPortalSessionURL(
|
|
||||||
workspaceId: string,
|
|
||||||
returnUrlPath?: string,
|
|
||||||
) {
|
|
||||||
const stripeCustomerId = await this.getStripeCustomerId(workspaceId);
|
|
||||||
|
|
||||||
if (!stripeCustomerId) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL');
|
|
||||||
const returnUrl = returnUrlPath
|
|
||||||
? frontBaseUrl + returnUrlPath
|
|
||||||
: frontBaseUrl;
|
|
||||||
|
|
||||||
const session = await this.stripeService.createBillingPortalSession(
|
|
||||||
stripeCustomerId,
|
|
||||||
returnUrl,
|
|
||||||
);
|
|
||||||
|
|
||||||
assert(session.url, 'Error: missing billingPortal.session.url');
|
|
||||||
|
|
||||||
return session.url;
|
|
||||||
}
|
|
||||||
|
|
||||||
async updateBillingSubscription(user: User) {
|
|
||||||
const billingSubscription = await this.getCurrentBillingSubscription({
|
|
||||||
workspaceId: user.defaultWorkspaceId,
|
|
||||||
});
|
|
||||||
const newInterval =
|
|
||||||
billingSubscription?.interval === SubscriptionInterval.Year
|
|
||||||
? SubscriptionInterval.Month
|
|
||||||
: SubscriptionInterval.Year;
|
|
||||||
const billingSubscriptionItem = await this.getBillingSubscriptionItem(
|
|
||||||
user.defaultWorkspaceId,
|
|
||||||
);
|
|
||||||
const stripeProductId = this.getProductStripeId(AvailableProduct.BasePlan);
|
|
||||||
|
|
||||||
if (!stripeProductId) {
|
|
||||||
throw new Error('Stripe product id not found for basePlan');
|
|
||||||
}
|
|
||||||
const productPrices = await this.getProductPrices(stripeProductId);
|
|
||||||
|
|
||||||
const stripePriceId = productPrices.filter(
|
|
||||||
(price) => price.recurringInterval === newInterval,
|
|
||||||
)?.[0]?.stripePriceId;
|
|
||||||
|
|
||||||
await this.stripeService.updateBillingSubscriptionItem(
|
|
||||||
billingSubscriptionItem,
|
|
||||||
stripePriceId,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
async computeCheckoutSessionURL(
|
|
||||||
user: User,
|
|
||||||
priceId: string,
|
|
||||||
successUrlPath?: string,
|
|
||||||
): Promise<string> {
|
|
||||||
const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL');
|
|
||||||
const successUrl = successUrlPath
|
|
||||||
? frontBaseUrl + successUrlPath
|
|
||||||
: frontBaseUrl;
|
|
||||||
|
|
||||||
const quantity =
|
|
||||||
(await this.userWorkspaceService.getWorkspaceMemberCount()) || 1;
|
|
||||||
|
|
||||||
const stripeCustomerId = (
|
|
||||||
await this.billingSubscriptionRepository.findOneBy({
|
|
||||||
workspaceId: user.defaultWorkspaceId,
|
|
||||||
})
|
|
||||||
)?.stripeCustomerId;
|
|
||||||
|
|
||||||
const session = await this.stripeService.createCheckoutSession(
|
|
||||||
user,
|
|
||||||
priceId,
|
|
||||||
quantity,
|
|
||||||
successUrl,
|
|
||||||
frontBaseUrl,
|
|
||||||
stripeCustomerId,
|
|
||||||
);
|
|
||||||
|
|
||||||
assert(session.url, 'Error: missing checkout.session.url');
|
|
||||||
|
|
||||||
return session.url;
|
|
||||||
}
|
|
||||||
|
|
||||||
async deleteSubscription(workspaceId: string) {
|
|
||||||
const subscriptionToCancel = await this.getCurrentBillingSubscription({
|
|
||||||
workspaceId,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (subscriptionToCancel) {
|
|
||||||
await this.stripeService.cancelSubscription(
|
|
||||||
subscriptionToCancel.stripeSubscriptionId,
|
|
||||||
);
|
|
||||||
await this.billingSubscriptionRepository.delete(subscriptionToCancel.id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async handleUnpaidInvoices(data: Stripe.SetupIntentSucceededEvent.Data) {
|
|
||||||
const billingSubscription = await this.getCurrentBillingSubscription({
|
|
||||||
stripeCustomerId: data.object.customer as string,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (billingSubscription?.status === 'unpaid') {
|
|
||||||
await this.stripeService.collectLastInvoice(
|
|
||||||
billingSubscription.stripeSubscriptionId,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async upsertBillingSubscription(
|
|
||||||
workspaceId: string,
|
|
||||||
data:
|
|
||||||
| Stripe.CustomerSubscriptionUpdatedEvent.Data
|
|
||||||
| Stripe.CustomerSubscriptionCreatedEvent.Data
|
|
||||||
| Stripe.CustomerSubscriptionDeletedEvent.Data,
|
|
||||||
) {
|
|
||||||
const workspace = this.workspaceRepository.find({
|
|
||||||
where: { id: workspaceId },
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!workspace) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.billingSubscriptionRepository.upsert(
|
|
||||||
{
|
|
||||||
workspaceId: workspaceId,
|
|
||||||
stripeCustomerId: data.object.customer as string,
|
|
||||||
stripeSubscriptionId: data.object.id,
|
|
||||||
status: data.object.status,
|
|
||||||
interval: data.object.items.data[0].plan.interval,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
conflictPaths: ['stripeSubscriptionId'],
|
|
||||||
skipUpdateIfNoValuesChanged: true,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
const billingSubscription = await this.getBillingSubscription(
|
|
||||||
data.object.id,
|
|
||||||
);
|
|
||||||
|
|
||||||
await this.billingSubscriptionItemRepository.upsert(
|
|
||||||
data.object.items.data.map((item) => {
|
|
||||||
return {
|
|
||||||
billingSubscriptionId: billingSubscription.id,
|
|
||||||
stripeProductId: item.price.product as string,
|
|
||||||
stripePriceId: item.price.id,
|
|
||||||
stripeSubscriptionItemId: item.id,
|
|
||||||
quantity: item.quantity,
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
{
|
|
||||||
conflictPaths: ['billingSubscriptionId', 'stripeProductId'],
|
|
||||||
skipUpdateIfNoValuesChanged: true,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
await this.featureFlagRepository.delete({
|
|
||||||
workspaceId,
|
|
||||||
key: FeatureFlagKeys.IsFreeAccessEnabled,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -0,0 +1,332 @@
|
|||||||
|
import { Injectable, Logger } from '@nestjs/common';
|
||||||
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
|
import Stripe from 'stripe';
|
||||||
|
import { Not, Repository } from 'typeorm';
|
||||||
|
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service';
|
||||||
|
import {
|
||||||
|
BillingSubscription,
|
||||||
|
SubscriptionInterval,
|
||||||
|
SubscriptionStatus,
|
||||||
|
} from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
||||||
|
import { BillingSubscriptionItem } from 'src/engine/core-modules/billing/entities/billing-subscription-item.entity';
|
||||||
|
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
|
||||||
|
import { ProductPriceEntity } from 'src/engine/core-modules/billing/dto/product-price.entity';
|
||||||
|
import { User } from 'src/engine/core-modules/user/user.entity';
|
||||||
|
import { assert } from 'src/utils/assert';
|
||||||
|
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
||||||
|
import {
|
||||||
|
FeatureFlagEntity,
|
||||||
|
FeatureFlagKeys,
|
||||||
|
} from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
||||||
|
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
||||||
|
|
||||||
|
export enum AvailableProduct {
|
||||||
|
BasePlan = 'base-plan',
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum WebhookEvent {
|
||||||
|
CUSTOMER_SUBSCRIPTION_CREATED = 'customer.subscription.created',
|
||||||
|
CUSTOMER_SUBSCRIPTION_UPDATED = 'customer.subscription.updated',
|
||||||
|
CUSTOMER_SUBSCRIPTION_DELETED = 'customer.subscription.deleted',
|
||||||
|
SETUP_INTENT_SUCCEEDED = 'setup_intent.succeeded',
|
||||||
|
}
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class BillingWorkspaceService {
|
||||||
|
protected readonly logger = new Logger(BillingService.name);
|
||||||
|
constructor(
|
||||||
|
private readonly stripeService: StripeService,
|
||||||
|
private readonly userWorkspaceService: UserWorkspaceService,
|
||||||
|
private readonly environmentService: EnvironmentService,
|
||||||
|
@InjectRepository(BillingSubscription, 'core')
|
||||||
|
private readonly billingSubscriptionRepository: Repository<BillingSubscription>,
|
||||||
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
|
@InjectRepository(BillingSubscriptionItem, 'core')
|
||||||
|
private readonly billingSubscriptionItemRepository: Repository<BillingSubscriptionItem>,
|
||||||
|
@InjectRepository(Workspace, 'core')
|
||||||
|
private readonly workspaceRepository: Repository<Workspace>,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
async isBillingEnabledForWorkspace(workspaceId: string) {
|
||||||
|
const isFreeAccessEnabled = await this.featureFlagRepository.findOneBy({
|
||||||
|
workspaceId,
|
||||||
|
key: FeatureFlagKeys.IsFreeAccessEnabled,
|
||||||
|
value: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
!isFreeAccessEnabled && this.environmentService.get('IS_BILLING_ENABLED')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
getProductStripeId(product: AvailableProduct) {
|
||||||
|
if (product === AvailableProduct.BasePlan) {
|
||||||
|
return this.environmentService.get('BILLING_STRIPE_BASE_PLAN_PRODUCT_ID');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async getProductPrices(stripeProductId: string) {
|
||||||
|
const productPrices =
|
||||||
|
await this.stripeService.getProductPrices(stripeProductId);
|
||||||
|
|
||||||
|
return this.formatProductPrices(productPrices.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
formatProductPrices(prices: Stripe.Price[]) {
|
||||||
|
const result: Record<string, ProductPriceEntity> = {};
|
||||||
|
|
||||||
|
prices.forEach((item) => {
|
||||||
|
const interval = item.recurring?.interval;
|
||||||
|
|
||||||
|
if (!interval || !item.unit_amount) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
!result[interval] ||
|
||||||
|
item.created > (result[interval]?.created || 0)
|
||||||
|
) {
|
||||||
|
result[interval] = {
|
||||||
|
unitAmount: item.unit_amount,
|
||||||
|
recurringInterval: interval,
|
||||||
|
created: item.created,
|
||||||
|
stripePriceId: item.id,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return Object.values(result).sort((a, b) => a.unitAmount - b.unitAmount);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getCurrentBillingSubscription(criteria: {
|
||||||
|
workspaceId?: string;
|
||||||
|
stripeCustomerId?: string;
|
||||||
|
}) {
|
||||||
|
const notCanceledSubscriptions =
|
||||||
|
await this.billingSubscriptionRepository.find({
|
||||||
|
where: { ...criteria, status: Not(SubscriptionStatus.Canceled) },
|
||||||
|
relations: ['billingSubscriptionItems'],
|
||||||
|
});
|
||||||
|
|
||||||
|
assert(
|
||||||
|
notCanceledSubscriptions.length <= 1,
|
||||||
|
`More than one not canceled subscription for workspace ${criteria.workspaceId}`,
|
||||||
|
);
|
||||||
|
|
||||||
|
return notCanceledSubscriptions?.[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
async getBillingSubscription(stripeSubscriptionId: string) {
|
||||||
|
return this.billingSubscriptionRepository.findOneOrFail({
|
||||||
|
where: { stripeSubscriptionId },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async getStripeCustomerId(workspaceId: string) {
|
||||||
|
const subscriptions = await this.billingSubscriptionRepository.find({
|
||||||
|
where: { workspaceId },
|
||||||
|
});
|
||||||
|
|
||||||
|
return subscriptions?.[0]?.stripeCustomerId;
|
||||||
|
}
|
||||||
|
|
||||||
|
async getBillingSubscriptionItem(
|
||||||
|
workspaceId: string,
|
||||||
|
stripeProductId = this.environmentService.get(
|
||||||
|
'BILLING_STRIPE_BASE_PLAN_PRODUCT_ID',
|
||||||
|
),
|
||||||
|
) {
|
||||||
|
const billingSubscription = await this.getCurrentBillingSubscription({
|
||||||
|
workspaceId,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!billingSubscription) {
|
||||||
|
throw new Error(
|
||||||
|
`Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const billingSubscriptionItem =
|
||||||
|
billingSubscription.billingSubscriptionItems.filter(
|
||||||
|
(billingSubscriptionItem) =>
|
||||||
|
billingSubscriptionItem.stripeProductId === stripeProductId,
|
||||||
|
)?.[0];
|
||||||
|
|
||||||
|
if (!billingSubscriptionItem) {
|
||||||
|
throw new Error(
|
||||||
|
`Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return billingSubscriptionItem;
|
||||||
|
}
|
||||||
|
|
||||||
|
async computeBillingPortalSessionURL(
|
||||||
|
workspaceId: string,
|
||||||
|
returnUrlPath?: string,
|
||||||
|
) {
|
||||||
|
const stripeCustomerId = await this.getStripeCustomerId(workspaceId);
|
||||||
|
|
||||||
|
if (!stripeCustomerId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL');
|
||||||
|
const returnUrl = returnUrlPath
|
||||||
|
? frontBaseUrl + returnUrlPath
|
||||||
|
: frontBaseUrl;
|
||||||
|
|
||||||
|
const session = await this.stripeService.createBillingPortalSession(
|
||||||
|
stripeCustomerId,
|
||||||
|
returnUrl,
|
||||||
|
);
|
||||||
|
|
||||||
|
assert(session.url, 'Error: missing billingPortal.session.url');
|
||||||
|
|
||||||
|
return session.url;
|
||||||
|
}
|
||||||
|
|
||||||
|
async updateBillingSubscription(user: User) {
|
||||||
|
const billingSubscription = await this.getCurrentBillingSubscription({
|
||||||
|
workspaceId: user.defaultWorkspaceId,
|
||||||
|
});
|
||||||
|
const newInterval =
|
||||||
|
billingSubscription?.interval === SubscriptionInterval.Year
|
||||||
|
? SubscriptionInterval.Month
|
||||||
|
: SubscriptionInterval.Year;
|
||||||
|
const billingSubscriptionItem = await this.getBillingSubscriptionItem(
|
||||||
|
user.defaultWorkspaceId,
|
||||||
|
);
|
||||||
|
const stripeProductId = this.getProductStripeId(AvailableProduct.BasePlan);
|
||||||
|
|
||||||
|
if (!stripeProductId) {
|
||||||
|
throw new Error('Stripe product id not found for basePlan');
|
||||||
|
}
|
||||||
|
const productPrices = await this.getProductPrices(stripeProductId);
|
||||||
|
|
||||||
|
const stripePriceId = productPrices.filter(
|
||||||
|
(price) => price.recurringInterval === newInterval,
|
||||||
|
)?.[0]?.stripePriceId;
|
||||||
|
|
||||||
|
await this.stripeService.updateBillingSubscriptionItem(
|
||||||
|
billingSubscriptionItem,
|
||||||
|
stripePriceId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async computeCheckoutSessionURL(
|
||||||
|
user: User,
|
||||||
|
priceId: string,
|
||||||
|
successUrlPath?: string,
|
||||||
|
): Promise<string> {
|
||||||
|
const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL');
|
||||||
|
const successUrl = successUrlPath
|
||||||
|
? frontBaseUrl + successUrlPath
|
||||||
|
: frontBaseUrl;
|
||||||
|
|
||||||
|
const quantity =
|
||||||
|
(await this.userWorkspaceService.getWorkspaceMemberCount()) || 1;
|
||||||
|
|
||||||
|
const stripeCustomerId = (
|
||||||
|
await this.billingSubscriptionRepository.findOneBy({
|
||||||
|
workspaceId: user.defaultWorkspaceId,
|
||||||
|
})
|
||||||
|
)?.stripeCustomerId;
|
||||||
|
|
||||||
|
const session = await this.stripeService.createCheckoutSession(
|
||||||
|
user,
|
||||||
|
priceId,
|
||||||
|
quantity,
|
||||||
|
successUrl,
|
||||||
|
frontBaseUrl,
|
||||||
|
stripeCustomerId,
|
||||||
|
);
|
||||||
|
|
||||||
|
assert(session.url, 'Error: missing checkout.session.url');
|
||||||
|
|
||||||
|
return session.url;
|
||||||
|
}
|
||||||
|
|
||||||
|
async deleteSubscription(workspaceId: string) {
|
||||||
|
const subscriptionToCancel = await this.getCurrentBillingSubscription({
|
||||||
|
workspaceId,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (subscriptionToCancel) {
|
||||||
|
await this.stripeService.cancelSubscription(
|
||||||
|
subscriptionToCancel.stripeSubscriptionId,
|
||||||
|
);
|
||||||
|
await this.billingSubscriptionRepository.delete(subscriptionToCancel.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async handleUnpaidInvoices(data: Stripe.SetupIntentSucceededEvent.Data) {
|
||||||
|
const billingSubscription = await this.getCurrentBillingSubscription({
|
||||||
|
stripeCustomerId: data.object.customer as string,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (billingSubscription?.status === 'unpaid') {
|
||||||
|
await this.stripeService.collectLastInvoice(
|
||||||
|
billingSubscription.stripeSubscriptionId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async upsertBillingSubscription(
|
||||||
|
workspaceId: string,
|
||||||
|
data:
|
||||||
|
| Stripe.CustomerSubscriptionUpdatedEvent.Data
|
||||||
|
| Stripe.CustomerSubscriptionCreatedEvent.Data
|
||||||
|
| Stripe.CustomerSubscriptionDeletedEvent.Data,
|
||||||
|
) {
|
||||||
|
const workspace = this.workspaceRepository.find({
|
||||||
|
where: { id: workspaceId },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!workspace) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.billingSubscriptionRepository.upsert(
|
||||||
|
{
|
||||||
|
workspaceId: workspaceId,
|
||||||
|
stripeCustomerId: data.object.customer as string,
|
||||||
|
stripeSubscriptionId: data.object.id,
|
||||||
|
status: data.object.status,
|
||||||
|
interval: data.object.items.data[0].plan.interval,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
conflictPaths: ['stripeSubscriptionId'],
|
||||||
|
skipUpdateIfNoValuesChanged: true,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
const billingSubscription = await this.getBillingSubscription(
|
||||||
|
data.object.id,
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.billingSubscriptionItemRepository.upsert(
|
||||||
|
data.object.items.data.map((item) => {
|
||||||
|
return {
|
||||||
|
billingSubscriptionId: billingSubscription.id,
|
||||||
|
stripeProductId: item.price.product as string,
|
||||||
|
stripePriceId: item.price.id,
|
||||||
|
stripeSubscriptionItemId: item.id,
|
||||||
|
quantity: item.quantity,
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
conflictPaths: ['billingSubscriptionId', 'stripeProductId'],
|
||||||
|
skipUpdateIfNoValuesChanged: true,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.featureFlagRepository.delete({
|
||||||
|
workspaceId,
|
||||||
|
key: FeatureFlagKeys.IsFreeAccessEnabled,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -2,7 +2,7 @@ import { ArgsType, Field } from '@nestjs/graphql';
|
|||||||
|
|
||||||
import { IsNotEmpty, IsString } from 'class-validator';
|
import { IsNotEmpty, IsString } from 'class-validator';
|
||||||
|
|
||||||
import { AvailableProduct } from 'src/engine/core-modules/billing/billing.service';
|
import { AvailableProduct } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
|
|
||||||
@ArgsType()
|
@ArgsType()
|
||||||
export class ProductInput {
|
export class ProductInput {
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import { Logger, Scope } from '@nestjs/common';
|
import { Logger, Scope } from '@nestjs/common';
|
||||||
|
|
||||||
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
||||||
import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service';
|
import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service';
|
||||||
import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator';
|
import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator';
|
||||||
@ -16,7 +16,7 @@ export class UpdateSubscriptionJob {
|
|||||||
protected readonly logger = new Logger(UpdateSubscriptionJob.name);
|
protected readonly logger = new Logger(UpdateSubscriptionJob.name);
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private readonly billingService: BillingService,
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
private readonly userWorkspaceService: UserWorkspaceService,
|
private readonly userWorkspaceService: UserWorkspaceService,
|
||||||
private readonly stripeService: StripeService,
|
private readonly stripeService: StripeService,
|
||||||
) {}
|
) {}
|
||||||
@ -32,7 +32,9 @@ export class UpdateSubscriptionJob {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const billingSubscriptionItem =
|
const billingSubscriptionItem =
|
||||||
await this.billingService.getBillingSubscriptionItem(data.workspaceId);
|
await this.billingWorkspaceService.getBillingSubscriptionItem(
|
||||||
|
data.workspaceId,
|
||||||
|
);
|
||||||
|
|
||||||
await this.stripeService.updateSubscriptionItem(
|
await this.stripeService.updateSubscriptionItem(
|
||||||
billingSubscriptionItem.stripeSubscriptionItemId,
|
billingSubscriptionItem.stripeSubscriptionItemId,
|
||||||
|
|||||||
@ -10,14 +10,14 @@ import {
|
|||||||
UpdateSubscriptionJobData,
|
UpdateSubscriptionJobData,
|
||||||
} from 'src/engine/core-modules/billing/jobs/update-subscription.job';
|
} from 'src/engine/core-modules/billing/jobs/update-subscription.job';
|
||||||
import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator';
|
import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator';
|
||||||
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class BillingWorkspaceMemberListener {
|
export class BillingWorkspaceMemberListener {
|
||||||
constructor(
|
constructor(
|
||||||
@InjectMessageQueue(MessageQueue.billingQueue)
|
@InjectMessageQueue(MessageQueue.billingQueue)
|
||||||
private readonly messageQueueService: MessageQueueService,
|
private readonly messageQueueService: MessageQueueService,
|
||||||
private readonly billingService: BillingService,
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@OnEvent('workspaceMember.created')
|
@OnEvent('workspaceMember.created')
|
||||||
@ -26,7 +26,7 @@ export class BillingWorkspaceMemberListener {
|
|||||||
payload: ObjectRecordCreateEvent<WorkspaceMemberWorkspaceEntity>,
|
payload: ObjectRecordCreateEvent<WorkspaceMemberWorkspaceEntity>,
|
||||||
) {
|
) {
|
||||||
const isBillingEnabledForWorkspace =
|
const isBillingEnabledForWorkspace =
|
||||||
await this.billingService.isBillingEnabledForWorkspace(
|
await this.billingWorkspaceService.isBillingEnabledForWorkspace(
|
||||||
payload.workspaceId,
|
payload.workspaceId,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@ -22,6 +22,7 @@ export enum FeatureFlagKeys {
|
|||||||
IsPostgreSQLIntegrationEnabled = 'IS_POSTGRESQL_INTEGRATION_ENABLED',
|
IsPostgreSQLIntegrationEnabled = 'IS_POSTGRESQL_INTEGRATION_ENABLED',
|
||||||
IsStripeIntegrationEnabled = 'IS_STRIPE_INTEGRATION_ENABLED',
|
IsStripeIntegrationEnabled = 'IS_STRIPE_INTEGRATION_ENABLED',
|
||||||
IsContactCreationForSentAndReceivedEmailsEnabled = 'IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED',
|
IsContactCreationForSentAndReceivedEmailsEnabled = 'IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED',
|
||||||
|
IsMessagingAliasFetchingEnabled = 'IS_MESSAGING_ALIAS_FETCHING_ENABLED',
|
||||||
IsGoogleCalendarSyncV2Enabled = 'IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED',
|
IsGoogleCalendarSyncV2Enabled = 'IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED',
|
||||||
IsFreeAccessEnabled = 'IS_FREE_ACCESS_ENABLED',
|
IsFreeAccessEnabled = 'IS_FREE_ACCESS_ENABLED',
|
||||||
}
|
}
|
||||||
|
|||||||
@ -5,6 +5,7 @@ import { NestjsQueryTypeOrmModule } from '@ptc-org/nestjs-query-typeorm';
|
|||||||
|
|
||||||
import { TypeORMModule } from 'src/database/typeorm/typeorm.module';
|
import { TypeORMModule } from 'src/database/typeorm/typeorm.module';
|
||||||
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 { IsFeatureEnabledService } from 'src/engine/core-modules/feature-flag/services/is-feature-enabled.service';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [
|
imports: [
|
||||||
@ -17,7 +18,7 @@ import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-
|
|||||||
resolvers: [],
|
resolvers: [],
|
||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
exports: [],
|
exports: [IsFeatureEnabledService],
|
||||||
providers: [],
|
providers: [IsFeatureEnabledService],
|
||||||
})
|
})
|
||||||
export class FeatureFlagModule {}
|
export class FeatureFlagModule {}
|
||||||
|
|||||||
@ -0,0 +1,30 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
|
import { Repository } from 'typeorm';
|
||||||
|
|
||||||
|
import {
|
||||||
|
FeatureFlagEntity,
|
||||||
|
FeatureFlagKeys,
|
||||||
|
} from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class IsFeatureEnabledService {
|
||||||
|
constructor(
|
||||||
|
@InjectRepository(FeatureFlagEntity, 'core')
|
||||||
|
private readonly featureFlagRepository: Repository<FeatureFlagEntity>,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async isFeatureEnabled(
|
||||||
|
key: FeatureFlagKeys,
|
||||||
|
workspaceId: string,
|
||||||
|
): Promise<boolean> {
|
||||||
|
const featureFlag = await this.featureFlagRepository.findOneBy({
|
||||||
|
workspaceId,
|
||||||
|
key,
|
||||||
|
value: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
return !!featureFlag?.value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -14,7 +14,7 @@ import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inje
|
|||||||
import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository';
|
import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository';
|
||||||
import { SubscriptionStatus } from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
import { SubscriptionStatus } from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
||||||
import { isDefined } from 'src/utils/is-defined';
|
import { isDefined } from 'src/utils/is-defined';
|
||||||
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
|
|
||||||
enum OnboardingStepValues {
|
enum OnboardingStepValues {
|
||||||
SKIPPED = 'SKIPPED',
|
SKIPPED = 'SKIPPED',
|
||||||
@ -33,7 +33,7 @@ type OnboardingKeyValueType = {
|
|||||||
@Injectable()
|
@Injectable()
|
||||||
export class OnboardingService {
|
export class OnboardingService {
|
||||||
constructor(
|
constructor(
|
||||||
private readonly billingService: BillingService,
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
private readonly workspaceManagerService: WorkspaceManagerService,
|
private readonly workspaceManagerService: WorkspaceManagerService,
|
||||||
private readonly userWorkspaceService: UserWorkspaceService,
|
private readonly userWorkspaceService: UserWorkspaceService,
|
||||||
private readonly keyValuePairService: KeyValuePairService<OnboardingKeyValueType>,
|
private readonly keyValuePairService: KeyValuePairService<OnboardingKeyValueType>,
|
||||||
@ -45,7 +45,7 @@ export class OnboardingService {
|
|||||||
|
|
||||||
private async isSubscriptionIncompleteOnboardingStatus(user: User) {
|
private async isSubscriptionIncompleteOnboardingStatus(user: User) {
|
||||||
const isBillingEnabledForWorkspace =
|
const isBillingEnabledForWorkspace =
|
||||||
await this.billingService.isBillingEnabledForWorkspace(
|
await this.billingWorkspaceService.isBillingEnabledForWorkspace(
|
||||||
user.defaultWorkspaceId,
|
user.defaultWorkspaceId,
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -54,7 +54,7 @@ export class OnboardingService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const currentBillingSubscription =
|
const currentBillingSubscription =
|
||||||
await this.billingService.getCurrentBillingSubscription({
|
await this.billingWorkspaceService.getCurrentBillingSubscription({
|
||||||
workspaceId: user.defaultWorkspaceId,
|
workspaceId: user.defaultWorkspaceId,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@ -5,7 +5,7 @@ import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
|
|||||||
import { WorkspaceManagerService } from 'src/engine/workspace-manager/workspace-manager.service';
|
import { WorkspaceManagerService } from 'src/engine/workspace-manager/workspace-manager.service';
|
||||||
import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity';
|
import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity';
|
||||||
import { User } from 'src/engine/core-modules/user/user.entity';
|
import { User } from 'src/engine/core-modules/user/user.entity';
|
||||||
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
||||||
import { UserService } from 'src/engine/core-modules/user/services/user.service';
|
import { UserService } from 'src/engine/core-modules/user/services/user.service';
|
||||||
import { EmailService } from 'src/engine/integrations/email/email.service';
|
import { EmailService } from 'src/engine/integrations/email/email.service';
|
||||||
@ -46,7 +46,7 @@ describe('WorkspaceService', () => {
|
|||||||
useValue: {},
|
useValue: {},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
provide: BillingService,
|
provide: BillingWorkspaceService,
|
||||||
useValue: {},
|
useValue: {},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@ -14,7 +14,7 @@ import { User } from 'src/engine/core-modules/user/user.entity';
|
|||||||
import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input';
|
import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input';
|
||||||
import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity';
|
import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity';
|
||||||
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service';
|
||||||
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
import { SendInviteLink } from 'src/engine/core-modules/workspace/dtos/send-invite-link.entity';
|
import { SendInviteLink } from 'src/engine/core-modules/workspace/dtos/send-invite-link.entity';
|
||||||
import { EmailService } from 'src/engine/integrations/email/email.service';
|
import { EmailService } from 'src/engine/integrations/email/email.service';
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
@ -30,7 +30,7 @@ export class WorkspaceService extends TypeOrmQueryService<Workspace> {
|
|||||||
private readonly userWorkspaceRepository: Repository<UserWorkspace>,
|
private readonly userWorkspaceRepository: Repository<UserWorkspace>,
|
||||||
private readonly workspaceManagerService: WorkspaceManagerService,
|
private readonly workspaceManagerService: WorkspaceManagerService,
|
||||||
private readonly userWorkspaceService: UserWorkspaceService,
|
private readonly userWorkspaceService: UserWorkspaceService,
|
||||||
private readonly billingService: BillingService,
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
private readonly environmentService: EnvironmentService,
|
private readonly environmentService: EnvironmentService,
|
||||||
private readonly emailService: EmailService,
|
private readonly emailService: EmailService,
|
||||||
private readonly onboardingService: OnboardingService,
|
private readonly onboardingService: OnboardingService,
|
||||||
@ -64,7 +64,7 @@ export class WorkspaceService extends TypeOrmQueryService<Workspace> {
|
|||||||
assert(workspace, 'Workspace not found');
|
assert(workspace, 'Workspace not found');
|
||||||
|
|
||||||
await this.userWorkspaceRepository.delete({ workspaceId: id });
|
await this.userWorkspaceRepository.delete({ workspaceId: id });
|
||||||
await this.billingService.deleteSubscription(workspace.id);
|
await this.billingWorkspaceService.deleteSubscription(workspace.id);
|
||||||
|
|
||||||
await this.workspaceManagerService.delete(id);
|
await this.workspaceManagerService.delete(id);
|
||||||
|
|
||||||
|
|||||||
@ -22,7 +22,7 @@ import { User } from 'src/engine/core-modules/user/user.entity';
|
|||||||
import { AuthUser } from 'src/engine/decorators/auth/auth-user.decorator';
|
import { AuthUser } from 'src/engine/decorators/auth/auth-user.decorator';
|
||||||
import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input';
|
import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input';
|
||||||
import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
||||||
import { BillingService } from 'src/engine/core-modules/billing/billing.service';
|
import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service';
|
||||||
import { DemoEnvGuard } from 'src/engine/guards/demo.env.guard';
|
import { DemoEnvGuard } from 'src/engine/guards/demo.env.guard';
|
||||||
import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service';
|
import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service';
|
||||||
import { SendInviteLink } from 'src/engine/core-modules/workspace/dtos/send-invite-link.entity';
|
import { SendInviteLink } from 'src/engine/core-modules/workspace/dtos/send-invite-link.entity';
|
||||||
@ -41,7 +41,7 @@ export class WorkspaceResolver {
|
|||||||
private readonly workspaceCacheVersionService: WorkspaceCacheVersionService,
|
private readonly workspaceCacheVersionService: WorkspaceCacheVersionService,
|
||||||
private readonly userWorkspaceService: UserWorkspaceService,
|
private readonly userWorkspaceService: UserWorkspaceService,
|
||||||
private readonly fileUploadService: FileUploadService,
|
private readonly fileUploadService: FileUploadService,
|
||||||
private readonly billingService: BillingService,
|
private readonly billingWorkspaceService: BillingWorkspaceService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@Query(() => Workspace)
|
@Query(() => Workspace)
|
||||||
@ -122,7 +122,7 @@ export class WorkspaceResolver {
|
|||||||
async currentBillingSubscription(
|
async currentBillingSubscription(
|
||||||
@Parent() workspace: Workspace,
|
@Parent() workspace: Workspace,
|
||||||
): Promise<BillingSubscription | null> {
|
): Promise<BillingSubscription | null> {
|
||||||
return this.billingService.getCurrentBillingSubscription({
|
return this.billingWorkspaceService.getCurrentBillingSubscription({
|
||||||
workspaceId: workspace.id,
|
workspaceId: workspace.id,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@ -59,6 +59,7 @@ export class AddStandardIdCommand extends CommandRunner {
|
|||||||
IS_POSTGRESQL_INTEGRATION_ENABLED: true,
|
IS_POSTGRESQL_INTEGRATION_ENABLED: true,
|
||||||
IS_STRIPE_INTEGRATION_ENABLED: false,
|
IS_STRIPE_INTEGRATION_ENABLED: false,
|
||||||
IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED: true,
|
IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED: true,
|
||||||
|
IS_MESSAGING_ALIAS_FETCHING_ENABLED: true,
|
||||||
IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED: true,
|
IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED: true,
|
||||||
IS_FREE_ACCESS_ENABLED: false,
|
IS_FREE_ACCESS_ENABLED: false,
|
||||||
},
|
},
|
||||||
@ -76,6 +77,7 @@ export class AddStandardIdCommand extends CommandRunner {
|
|||||||
IS_POSTGRESQL_INTEGRATION_ENABLED: true,
|
IS_POSTGRESQL_INTEGRATION_ENABLED: true,
|
||||||
IS_STRIPE_INTEGRATION_ENABLED: false,
|
IS_STRIPE_INTEGRATION_ENABLED: false,
|
||||||
IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED: true,
|
IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED: true,
|
||||||
|
IS_MESSAGING_ALIAS_FETCHING_ENABLED: true,
|
||||||
IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED: true,
|
IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED: true,
|
||||||
IS_FREE_ACCESS_ENABLED: false,
|
IS_FREE_ACCESS_ENABLED: false,
|
||||||
},
|
},
|
||||||
|
|||||||
@ -140,6 +140,7 @@ export const CONNECTED_ACCOUNT_STANDARD_FIELD_IDS = {
|
|||||||
authFailedAt: '20202020-d268-4c6b-baff-400d402b430a',
|
authFailedAt: '20202020-d268-4c6b-baff-400d402b430a',
|
||||||
messageChannels: '20202020-24f7-4362-8468-042204d1e445',
|
messageChannels: '20202020-24f7-4362-8468-042204d1e445',
|
||||||
calendarChannels: '20202020-af4a-47bb-99ec-51911c1d3977',
|
calendarChannels: '20202020-af4a-47bb-99ec-51911c1d3977',
|
||||||
|
emailAliases: '20202020-8a3d-46be-814f-6228af16c47b',
|
||||||
};
|
};
|
||||||
|
|
||||||
export const EVENT_STANDARD_FIELD_IDS = {
|
export const EVENT_STANDARD_FIELD_IDS = {
|
||||||
|
|||||||
@ -2,6 +2,9 @@ import { Logger, Scope } from '@nestjs/common';
|
|||||||
|
|
||||||
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 { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service';
|
import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service';
|
||||||
|
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
||||||
|
import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator';
|
import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator';
|
||||||
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
|
import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants';
|
||||||
import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator';
|
import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator';
|
||||||
@ -21,6 +24,8 @@ export class GoogleCalendarSyncJob {
|
|||||||
constructor(
|
constructor(
|
||||||
private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService,
|
private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService,
|
||||||
private readonly googleCalendarSyncService: GoogleCalendarSyncService,
|
private readonly googleCalendarSyncService: GoogleCalendarSyncService,
|
||||||
|
@InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity)
|
||||||
|
private readonly connectedAccountRepository: ConnectedAccountRepository,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@Process(GoogleCalendarSyncJob.name)
|
@Process(GoogleCalendarSyncJob.name)
|
||||||
@ -29,9 +34,22 @@ export class GoogleCalendarSyncJob {
|
|||||||
`google calendar sync for workspace ${data.workspaceId} and account ${data.connectedAccountId}`,
|
`google calendar sync for workspace ${data.workspaceId} and account ${data.connectedAccountId}`,
|
||||||
);
|
);
|
||||||
try {
|
try {
|
||||||
|
const { connectedAccountId, workspaceId } = data;
|
||||||
|
|
||||||
|
const connectedAccount = await this.connectedAccountRepository.getById(
|
||||||
|
connectedAccountId,
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!connectedAccount) {
|
||||||
|
throw new Error(
|
||||||
|
`No connected account found for ${connectedAccountId} in workspace ${workspaceId}`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken(
|
await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken(
|
||||||
data.workspaceId,
|
connectedAccount,
|
||||||
data.connectedAccountId,
|
workspaceId,
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logger.error(
|
this.logger.error(
|
||||||
|
|||||||
@ -108,9 +108,8 @@ export class GoogleCalendarSyncService {
|
|||||||
const calendarChannelId = calendarChannel.id;
|
const calendarChannelId = calendarChannel.id;
|
||||||
|
|
||||||
const { events, nextSyncToken } = await this.getEventsFromGoogleCalendar(
|
const { events, nextSyncToken } = await this.getEventsFromGoogleCalendar(
|
||||||
refreshToken,
|
connectedAccount,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
connectedAccountId,
|
|
||||||
emailOrDomainToReimport,
|
emailOrDomainToReimport,
|
||||||
syncToken,
|
syncToken,
|
||||||
);
|
);
|
||||||
@ -321,9 +320,8 @@ export class GoogleCalendarSyncService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public async getEventsFromGoogleCalendar(
|
public async getEventsFromGoogleCalendar(
|
||||||
refreshToken: string,
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
workspaceId: string,
|
workspaceId: string,
|
||||||
connectedAccountId: string,
|
|
||||||
emailOrDomainToReimport?: string,
|
emailOrDomainToReimport?: string,
|
||||||
syncToken?: string,
|
syncToken?: string,
|
||||||
): Promise<{
|
): Promise<{
|
||||||
@ -332,7 +330,7 @@ export class GoogleCalendarSyncService {
|
|||||||
}> {
|
}> {
|
||||||
const googleCalendarClient =
|
const googleCalendarClient =
|
||||||
await this.googleCalendarClientProvider.getGoogleCalendarClient(
|
await this.googleCalendarClientProvider.getGoogleCalendarClient(
|
||||||
refreshToken,
|
connectedAccount,
|
||||||
);
|
);
|
||||||
|
|
||||||
const startTime = Date.now();
|
const startTime = Date.now();
|
||||||
@ -360,7 +358,7 @@ export class GoogleCalendarSyncService {
|
|||||||
|
|
||||||
await this.calendarChannelRepository.update(
|
await this.calendarChannelRepository.update(
|
||||||
{
|
{
|
||||||
id: connectedAccountId,
|
id: connectedAccount.id,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
syncCursor: '',
|
syncCursor: '',
|
||||||
@ -368,7 +366,7 @@ export class GoogleCalendarSyncService {
|
|||||||
);
|
);
|
||||||
|
|
||||||
this.logger.log(
|
this.logger.log(
|
||||||
`Sync token is no longer valid for connected account ${connectedAccountId} in workspace ${workspaceId}, resetting sync cursor.`,
|
`Sync token is no longer valid for connected account ${connectedAccount.id} in workspace ${workspaceId}, resetting sync cursor.`,
|
||||||
);
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@ -399,9 +397,9 @@ export class GoogleCalendarSyncService {
|
|||||||
const endTime = Date.now();
|
const endTime = Date.now();
|
||||||
|
|
||||||
this.logger.log(
|
this.logger.log(
|
||||||
`google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} getting events list in ${
|
`google calendar sync for workspace ${workspaceId} and account ${
|
||||||
endTime - startTime
|
connectedAccount.id
|
||||||
}ms.`,
|
} getting events list in ${endTime - startTime}ms.`,
|
||||||
);
|
);
|
||||||
|
|
||||||
return { events, nextSyncToken };
|
return { events, nextSyncToken };
|
||||||
|
|||||||
@ -2,9 +2,10 @@ import { Module } from '@nestjs/common';
|
|||||||
|
|
||||||
import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module';
|
import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module';
|
||||||
import { GoogleCalendarClientProvider } from 'src/modules/calendar/services/providers/google-calendar/google-calendar.provider';
|
import { GoogleCalendarClientProvider } from 'src/modules/calendar/services/providers/google-calendar/google-calendar.provider';
|
||||||
|
import { OAuth2ClientManagerModule } from 'src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module';
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [EnvironmentModule],
|
imports: [EnvironmentModule, OAuth2ClientManagerModule],
|
||||||
providers: [GoogleCalendarClientProvider],
|
providers: [GoogleCalendarClientProvider],
|
||||||
exports: [GoogleCalendarClientProvider],
|
exports: [GoogleCalendarClientProvider],
|
||||||
})
|
})
|
||||||
|
|||||||
@ -1,18 +1,21 @@
|
|||||||
import { Injectable } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
import { OAuth2Client } from 'google-auth-library';
|
|
||||||
import { calendar_v3 as calendarV3, google } from 'googleapis';
|
import { calendar_v3 as calendarV3, google } from 'googleapis';
|
||||||
|
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class GoogleCalendarClientProvider {
|
export class GoogleCalendarClientProvider {
|
||||||
constructor(private readonly environmentService: EnvironmentService) {}
|
constructor(
|
||||||
|
private readonly oAuth2ClientManagerService: OAuth2ClientManagerService,
|
||||||
|
) {}
|
||||||
|
|
||||||
public async getGoogleCalendarClient(
|
public async getGoogleCalendarClient(
|
||||||
refreshToken: string,
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
): Promise<calendarV3.Calendar> {
|
): Promise<calendarV3.Calendar> {
|
||||||
const oAuth2Client = await this.getOAuth2Client(refreshToken);
|
const oAuth2Client =
|
||||||
|
await this.oAuth2ClientManagerService.getOAuth2Client(connectedAccount);
|
||||||
|
|
||||||
const googleCalendarClient = google.calendar({
|
const googleCalendarClient = google.calendar({
|
||||||
version: 'v3',
|
version: 'v3',
|
||||||
@ -21,24 +24,4 @@ export class GoogleCalendarClientProvider {
|
|||||||
|
|
||||||
return googleCalendarClient;
|
return googleCalendarClient;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async getOAuth2Client(refreshToken: string): Promise<OAuth2Client> {
|
|
||||||
const googleCalendarClientId = this.environmentService.get(
|
|
||||||
'AUTH_GOOGLE_CLIENT_ID',
|
|
||||||
);
|
|
||||||
const googleCalendarClientSecret = this.environmentService.get(
|
|
||||||
'AUTH_GOOGLE_CLIENT_SECRET',
|
|
||||||
);
|
|
||||||
|
|
||||||
const oAuth2Client = new google.auth.OAuth2(
|
|
||||||
googleCalendarClientId,
|
|
||||||
googleCalendarClientSecret,
|
|
||||||
);
|
|
||||||
|
|
||||||
oAuth2Client.setCredentials({
|
|
||||||
refresh_token: refreshToken,
|
|
||||||
});
|
|
||||||
|
|
||||||
return oAuth2Client;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -16,7 +16,7 @@ import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/perso
|
|||||||
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
|
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
|
||||||
import { getUniqueContactsAndHandles } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util';
|
import { getUniqueContactsAndHandles } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util';
|
||||||
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 { filterOutContactsFromCompanyOrWorkspace } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util';
|
import { filterOutSelfAndContactsFromCompanyOrWorkspace } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util';
|
||||||
import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service';
|
import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service';
|
||||||
import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity';
|
import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity';
|
||||||
import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity';
|
import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity';
|
||||||
@ -43,7 +43,7 @@ export class CreateCompanyAndContactService {
|
|||||||
) {}
|
) {}
|
||||||
|
|
||||||
async createCompaniesAndPeople(
|
async createCompaniesAndPeople(
|
||||||
connectedAccountHandle: string,
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
contactsToCreate: Contact[],
|
contactsToCreate: Contact[],
|
||||||
workspaceId: string,
|
workspaceId: string,
|
||||||
transactionManager?: EntityManager,
|
transactionManager?: EntityManager,
|
||||||
@ -62,9 +62,9 @@ export class CreateCompanyAndContactService {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const contactsToCreateFromOtherCompanies =
|
const contactsToCreateFromOtherCompanies =
|
||||||
filterOutContactsFromCompanyOrWorkspace(
|
filterOutSelfAndContactsFromCompanyOrWorkspace(
|
||||||
contactsToCreate,
|
contactsToCreate,
|
||||||
connectedAccountHandle,
|
connectedAccount,
|
||||||
workspaceMembers,
|
workspaceMembers,
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -150,7 +150,7 @@ export class CreateCompanyAndContactService {
|
|||||||
await this.workspaceDataSource?.transaction(
|
await this.workspaceDataSource?.transaction(
|
||||||
async (transactionManager: EntityManager) => {
|
async (transactionManager: EntityManager) => {
|
||||||
const createdPeople = await this.createCompaniesAndPeople(
|
const createdPeople = await this.createCompaniesAndPeople(
|
||||||
connectedAccount.handle,
|
connectedAccount,
|
||||||
contactsBatch,
|
contactsBatch,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
transactionManager,
|
transactionManager,
|
||||||
|
|||||||
@ -1,13 +1,16 @@
|
|||||||
import { getDomainNameFromHandle } from 'src/modules/calendar-messaging-participant/utils/get-domain-name-from-handle.util';
|
import { getDomainNameFromHandle } from 'src/modules/calendar-messaging-participant/utils/get-domain-name-from-handle.util';
|
||||||
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
|
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
import { Contact } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type';
|
import { Contact } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type';
|
||||||
|
|
||||||
export function filterOutContactsFromCompanyOrWorkspace(
|
export function filterOutSelfAndContactsFromCompanyOrWorkspace(
|
||||||
contacts: Contact[],
|
contacts: Contact[],
|
||||||
selfHandle: string,
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
workspaceMembers: WorkspaceMemberWorkspaceEntity[],
|
workspaceMembers: WorkspaceMemberWorkspaceEntity[],
|
||||||
): Contact[] {
|
): Contact[] {
|
||||||
const selfDomainName = getDomainNameFromHandle(selfHandle);
|
const selfDomainName = getDomainNameFromHandle(connectedAccount.handle);
|
||||||
|
|
||||||
|
const emailAliases = connectedAccount.emailAliases?.split(',') || [];
|
||||||
|
|
||||||
const workspaceMembersMap = workspaceMembers.reduce(
|
const workspaceMembersMap = workspaceMembers.reduce(
|
||||||
(map, workspaceMember) => {
|
(map, workspaceMember) => {
|
||||||
@ -21,6 +24,7 @@ export function filterOutContactsFromCompanyOrWorkspace(
|
|||||||
return contacts.filter(
|
return contacts.filter(
|
||||||
(contact) =>
|
(contact) =>
|
||||||
getDomainNameFromHandle(contact.handle) !== selfDomainName &&
|
getDomainNameFromHandle(contact.handle) !== selfDomainName &&
|
||||||
!workspaceMembersMap[contact.handle],
|
!workspaceMembersMap[contact.handle] &&
|
||||||
|
!emailAliases.includes(contact.handle),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -0,0 +1,43 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { google } from 'googleapis';
|
||||||
|
|
||||||
|
import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleEmailAliasManagerService {
|
||||||
|
constructor(
|
||||||
|
private readonly oAuth2ClientManagerService: OAuth2ClientManagerService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async getEmailAliases(
|
||||||
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
|
) {
|
||||||
|
const oAuth2Client =
|
||||||
|
await this.oAuth2ClientManagerService.getOAuth2Client(connectedAccount);
|
||||||
|
|
||||||
|
const people = google.people({
|
||||||
|
version: 'v1',
|
||||||
|
auth: oAuth2Client,
|
||||||
|
});
|
||||||
|
|
||||||
|
const emailsResponse = await people.people.get({
|
||||||
|
resourceName: 'people/me',
|
||||||
|
personFields: 'emailAddresses',
|
||||||
|
});
|
||||||
|
|
||||||
|
const emailAddresses = emailsResponse.data.emailAddresses;
|
||||||
|
|
||||||
|
const emailAliases =
|
||||||
|
emailAddresses
|
||||||
|
?.filter((emailAddress) => {
|
||||||
|
return emailAddress.metadata?.primary !== true;
|
||||||
|
})
|
||||||
|
.map((emailAddress) => {
|
||||||
|
return emailAddress.value || '';
|
||||||
|
}) || [];
|
||||||
|
|
||||||
|
return emailAliases;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,19 @@
|
|||||||
|
import { Module } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
|
||||||
|
import { GoogleEmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service';
|
||||||
|
import { EmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/services/email-alias-manager.service';
|
||||||
|
import { OAuth2ClientManagerModule } from 'src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
|
|
||||||
|
@Module({
|
||||||
|
imports: [
|
||||||
|
ObjectMetadataRepositoryModule.forFeature([
|
||||||
|
ConnectedAccountWorkspaceEntity,
|
||||||
|
]),
|
||||||
|
OAuth2ClientManagerModule,
|
||||||
|
],
|
||||||
|
providers: [EmailAliasManagerService, GoogleEmailAliasManagerService],
|
||||||
|
exports: [EmailAliasManagerService],
|
||||||
|
})
|
||||||
|
export class EmailAliasManagerModule {}
|
||||||
@ -0,0 +1,41 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
|
||||||
|
import { GoogleEmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service';
|
||||||
|
import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class EmailAliasManagerService {
|
||||||
|
constructor(
|
||||||
|
@InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity)
|
||||||
|
private readonly connectedAccountRepository: ConnectedAccountRepository,
|
||||||
|
private readonly googleEmailAliasManagerService: GoogleEmailAliasManagerService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async refreshEmailAliases(
|
||||||
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
|
workspaceId: string,
|
||||||
|
) {
|
||||||
|
let emailAliases: string[];
|
||||||
|
|
||||||
|
switch (connectedAccount.provider) {
|
||||||
|
case 'google':
|
||||||
|
emailAliases =
|
||||||
|
await this.googleEmailAliasManagerService.getEmailAliases(
|
||||||
|
connectedAccount,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error(
|
||||||
|
`Email alias manager for provider ${connectedAccount.provider} is not implemented`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.connectedAccountRepository.updateEmailAliases(
|
||||||
|
emailAliases,
|
||||||
|
connectedAccount.id,
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,29 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { OAuth2Client } from 'google-auth-library';
|
||||||
|
import { google } from 'googleapis';
|
||||||
|
|
||||||
|
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class GoogleOAuth2ClientManagerService {
|
||||||
|
constructor(private readonly environmentService: EnvironmentService) {}
|
||||||
|
|
||||||
|
public async getOAuth2Client(refreshToken: string): Promise<OAuth2Client> {
|
||||||
|
const gmailClientId = this.environmentService.get('AUTH_GOOGLE_CLIENT_ID');
|
||||||
|
const gmailClientSecret = this.environmentService.get(
|
||||||
|
'AUTH_GOOGLE_CLIENT_SECRET',
|
||||||
|
);
|
||||||
|
|
||||||
|
const oAuth2Client = new google.auth.OAuth2(
|
||||||
|
gmailClientId,
|
||||||
|
gmailClientSecret,
|
||||||
|
);
|
||||||
|
|
||||||
|
oAuth2Client.setCredentials({
|
||||||
|
refresh_token: refreshToken,
|
||||||
|
});
|
||||||
|
|
||||||
|
return oAuth2Client;
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,11 @@
|
|||||||
|
import { Module } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { GoogleOAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service';
|
||||||
|
import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service';
|
||||||
|
|
||||||
|
@Module({
|
||||||
|
imports: [],
|
||||||
|
providers: [OAuth2ClientManagerService, GoogleOAuth2ClientManagerService],
|
||||||
|
exports: [OAuth2ClientManagerService],
|
||||||
|
})
|
||||||
|
export class OAuth2ClientManagerModule {}
|
||||||
@ -0,0 +1,30 @@
|
|||||||
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
|
import { OAuth2Client } from 'google-auth-library';
|
||||||
|
|
||||||
|
import { GoogleOAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class OAuth2ClientManagerService {
|
||||||
|
constructor(
|
||||||
|
private readonly googleOAuth2ClientManagerService: GoogleOAuth2ClientManagerService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public async getOAuth2Client(
|
||||||
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
|
): Promise<OAuth2Client> {
|
||||||
|
const { refreshToken } = connectedAccount;
|
||||||
|
|
||||||
|
switch (connectedAccount.provider) {
|
||||||
|
case 'google':
|
||||||
|
return this.googleOAuth2ClientManagerService.getOAuth2Client(
|
||||||
|
refreshToken,
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
throw new Error(
|
||||||
|
`OAuth2 client manager for provider ${connectedAccount.provider} is not implemented`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -306,4 +306,22 @@ export class ConnectedAccountRepository {
|
|||||||
|
|
||||||
return connectedAccount;
|
return connectedAccount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async updateEmailAliases(
|
||||||
|
emailAliases: string[],
|
||||||
|
connectedAccountId: string,
|
||||||
|
workspaceId: string,
|
||||||
|
transactionManager?: EntityManager,
|
||||||
|
) {
|
||||||
|
const dataSourceSchema =
|
||||||
|
this.workspaceDataSourceService.getSchemaName(workspaceId);
|
||||||
|
|
||||||
|
await this.workspaceDataSourceService.executeRawQuery(
|
||||||
|
`UPDATE ${dataSourceSchema}."connectedAccount" SET "emailAliases" = $1 WHERE "id" = $2`,
|
||||||
|
// TODO: modify emailAliases to be of fieldmetadatatype array
|
||||||
|
[emailAliases.join(','), connectedAccountId],
|
||||||
|
workspaceId,
|
||||||
|
transactionManager,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -16,33 +16,27 @@ export class GoogleAPIRefreshAccessTokenService {
|
|||||||
) {}
|
) {}
|
||||||
|
|
||||||
async refreshAndSaveAccessToken(
|
async refreshAndSaveAccessToken(
|
||||||
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
workspaceId: string,
|
workspaceId: string,
|
||||||
connectedAccountId: string,
|
|
||||||
): Promise<string> {
|
): Promise<string> {
|
||||||
const connectedAccount = await this.connectedAccountRepository.getById(
|
|
||||||
connectedAccountId,
|
|
||||||
workspaceId,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!connectedAccount) {
|
|
||||||
throw new Error(
|
|
||||||
`No connected account found for ${connectedAccountId} in workspace ${workspaceId}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const refreshToken = connectedAccount.refreshToken;
|
const refreshToken = connectedAccount.refreshToken;
|
||||||
|
|
||||||
if (!refreshToken) {
|
if (!refreshToken) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
`No refresh token found for connected account ${connectedAccountId} in workspace ${workspaceId}`,
|
`No refresh token found for connected account ${connectedAccount.id} in workspace ${workspaceId}`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const accessToken = await this.refreshAccessToken(refreshToken);
|
const accessToken = await this.refreshAccessToken(refreshToken);
|
||||||
|
|
||||||
await this.connectedAccountRepository.updateAccessToken(
|
await this.connectedAccountRepository.updateAccessToken(
|
||||||
accessToken,
|
accessToken,
|
||||||
connectedAccountId,
|
connectedAccount.id,
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.connectedAccountRepository.updateAccessToken(
|
||||||
|
accessToken,
|
||||||
|
connectedAccount.id,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@ -17,6 +17,8 @@ import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field
|
|||||||
import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator';
|
import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator';
|
||||||
import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator';
|
import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator';
|
||||||
import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity';
|
import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity';
|
||||||
|
import { FeatureFlagKeys } from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
||||||
|
import { WorkspaceGate } from 'src/engine/twenty-orm/decorators/workspace-gate.decorator';
|
||||||
import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator';
|
import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator';
|
||||||
|
|
||||||
export enum ConnectedAccountProvider {
|
export enum ConnectedAccountProvider {
|
||||||
@ -89,6 +91,18 @@ export class ConnectedAccountWorkspaceEntity extends BaseWorkspaceEntity {
|
|||||||
@WorkspaceIsNullable()
|
@WorkspaceIsNullable()
|
||||||
authFailedAt: Date | null;
|
authFailedAt: Date | null;
|
||||||
|
|
||||||
|
@WorkspaceField({
|
||||||
|
standardId: CONNECTED_ACCOUNT_STANDARD_FIELD_IDS.emailAliases,
|
||||||
|
type: FieldMetadataType.TEXT,
|
||||||
|
label: 'Email Aliases',
|
||||||
|
description: 'Email Aliases',
|
||||||
|
icon: 'IconMail',
|
||||||
|
})
|
||||||
|
@WorkspaceGate({
|
||||||
|
featureFlag: FeatureFlagKeys.IsMessagingAliasFetchingEnabled,
|
||||||
|
})
|
||||||
|
emailAliases: string;
|
||||||
|
|
||||||
@WorkspaceRelation({
|
@WorkspaceRelation({
|
||||||
standardId: CONNECTED_ACCOUNT_STANDARD_FIELD_IDS.accountOwner,
|
standardId: CONNECTED_ACCOUNT_STANDARD_FIELD_IDS.accountOwner,
|
||||||
type: RelationMetadataType.MANY_TO_ONE,
|
type: RelationMetadataType.MANY_TO_ONE,
|
||||||
|
|||||||
@ -1,17 +1,15 @@
|
|||||||
import { Injectable, Logger } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
import { DataSource, EntityManager } from 'typeorm';
|
import { EntityManager } from 'typeorm';
|
||||||
import { v4 } from 'uuid';
|
import { v4 } from 'uuid';
|
||||||
|
|
||||||
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
|
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
|
||||||
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 { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository';
|
import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository';
|
||||||
import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository';
|
|
||||||
import { MessageThreadRepository } from 'src/modules/messaging/common/repositories/message-thread.repository';
|
import { MessageThreadRepository } from 'src/modules/messaging/common/repositories/message-thread.repository';
|
||||||
import { MessageRepository } from 'src/modules/messaging/common/repositories/message.repository';
|
import { MessageRepository } from 'src/modules/messaging/common/repositories/message.repository';
|
||||||
import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity';
|
import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity';
|
||||||
import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity';
|
|
||||||
import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity';
|
import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity';
|
||||||
import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity';
|
import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity';
|
||||||
import { GmailMessage } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message';
|
import { GmailMessage } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message';
|
||||||
@ -19,8 +17,6 @@ import { MessagingMessageThreadService } from 'src/modules/messaging/common/serv
|
|||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class MessagingMessageService {
|
export class MessagingMessageService {
|
||||||
private readonly logger = new Logger(MessagingMessageService.name);
|
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
|
||||||
@InjectObjectMetadataRepository(
|
@InjectObjectMetadataRepository(
|
||||||
@ -29,8 +25,6 @@ export class MessagingMessageService {
|
|||||||
private readonly messageChannelMessageAssociationRepository: MessageChannelMessageAssociationRepository,
|
private readonly messageChannelMessageAssociationRepository: MessageChannelMessageAssociationRepository,
|
||||||
@InjectObjectMetadataRepository(MessageWorkspaceEntity)
|
@InjectObjectMetadataRepository(MessageWorkspaceEntity)
|
||||||
private readonly messageRepository: MessageRepository,
|
private readonly messageRepository: MessageRepository,
|
||||||
@InjectObjectMetadataRepository(MessageChannelWorkspaceEntity)
|
|
||||||
private readonly messageChannelRepository: MessageChannelRepository,
|
|
||||||
@InjectObjectMetadataRepository(MessageThreadWorkspaceEntity)
|
@InjectObjectMetadataRepository(MessageThreadWorkspaceEntity)
|
||||||
private readonly messageThreadRepository: MessageThreadRepository,
|
private readonly messageThreadRepository: MessageThreadRepository,
|
||||||
private readonly messageThreadService: MessagingMessageThreadService,
|
private readonly messageThreadService: MessagingMessageThreadService,
|
||||||
@ -101,104 +95,6 @@ export class MessagingMessageService {
|
|||||||
return messageExternalIdsAndIdsMap;
|
return messageExternalIdsAndIdsMap;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async saveMessages(
|
|
||||||
messages: GmailMessage[],
|
|
||||||
workspaceDataSource: DataSource,
|
|
||||||
connectedAccount: ConnectedAccountWorkspaceEntity,
|
|
||||||
gmailMessageChannelId: string,
|
|
||||||
workspaceId: string,
|
|
||||||
): Promise<Map<string, string>> {
|
|
||||||
const messageExternalIdsAndIdsMap = new Map<string, string>();
|
|
||||||
|
|
||||||
try {
|
|
||||||
let keepImporting = true;
|
|
||||||
|
|
||||||
for (const message of messages) {
|
|
||||||
if (!keepImporting) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
await workspaceDataSource?.transaction(
|
|
||||||
async (manager: EntityManager) => {
|
|
||||||
const gmailMessageChannel =
|
|
||||||
await this.messageChannelRepository.getByIds(
|
|
||||||
[gmailMessageChannelId],
|
|
||||||
workspaceId,
|
|
||||||
manager,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (gmailMessageChannel.length === 0) {
|
|
||||||
this.logger.error(
|
|
||||||
`No message channel found for connected account ${connectedAccount.id} in workspace ${workspaceId} in saveMessages`,
|
|
||||||
);
|
|
||||||
|
|
||||||
keepImporting = false;
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const existingMessageChannelMessageAssociationsCount =
|
|
||||||
await this.messageChannelMessageAssociationRepository.countByMessageExternalIdsAndMessageChannelId(
|
|
||||||
[message.externalId],
|
|
||||||
gmailMessageChannelId,
|
|
||||||
workspaceId,
|
|
||||||
manager,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (existingMessageChannelMessageAssociationsCount > 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: This does not handle all thread merging use cases and might create orphan threads.
|
|
||||||
const savedOrExistingMessageThreadId =
|
|
||||||
await this.messageThreadService.saveMessageThreadOrReturnExistingMessageThread(
|
|
||||||
message.headerMessageId,
|
|
||||||
message.messageThreadExternalId,
|
|
||||||
workspaceId,
|
|
||||||
manager,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!savedOrExistingMessageThreadId) {
|
|
||||||
throw new Error(
|
|
||||||
`No message thread found for message ${message.headerMessageId} in workspace ${workspaceId} in saveMessages`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const savedOrExistingMessageId =
|
|
||||||
await this.saveMessageOrReturnExistingMessage(
|
|
||||||
message,
|
|
||||||
savedOrExistingMessageThreadId,
|
|
||||||
connectedAccount,
|
|
||||||
workspaceId,
|
|
||||||
manager,
|
|
||||||
);
|
|
||||||
|
|
||||||
messageExternalIdsAndIdsMap.set(
|
|
||||||
message.externalId,
|
|
||||||
savedOrExistingMessageId,
|
|
||||||
);
|
|
||||||
|
|
||||||
await this.messageChannelMessageAssociationRepository.insert(
|
|
||||||
gmailMessageChannelId,
|
|
||||||
savedOrExistingMessageId,
|
|
||||||
message.externalId,
|
|
||||||
savedOrExistingMessageThreadId,
|
|
||||||
message.messageThreadExternalId,
|
|
||||||
workspaceId,
|
|
||||||
manager,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
throw new Error(
|
|
||||||
`Error saving connected account ${connectedAccount.id} messages to workspace ${workspaceId}: ${error.message}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return messageExternalIdsAndIdsMap;
|
|
||||||
}
|
|
||||||
|
|
||||||
private async saveMessageOrReturnExistingMessage(
|
private async saveMessageOrReturnExistingMessage(
|
||||||
message: GmailMessage,
|
message: GmailMessage,
|
||||||
messageThreadId: string,
|
messageThreadId: string,
|
||||||
@ -219,8 +115,11 @@ export class MessagingMessageService {
|
|||||||
|
|
||||||
const newMessageId = v4();
|
const newMessageId = v4();
|
||||||
|
|
||||||
const messageDirection =
|
const messageDirection = connectedAccount.emailAliases?.includes(
|
||||||
connectedAccount.handle === message.fromHandle ? 'outgoing' : 'incoming';
|
message.fromHandle,
|
||||||
|
)
|
||||||
|
? 'outgoing'
|
||||||
|
: 'incoming';
|
||||||
|
|
||||||
const receivedAt = new Date(parseInt(message.internalDate));
|
const receivedAt = new Date(parseInt(message.internalDate));
|
||||||
|
|
||||||
|
|||||||
@ -58,6 +58,8 @@ export class MessagingSaveMessagesAndEnqueueContactCreationService {
|
|||||||
value: true,
|
value: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const emailAliases = connectedAccount.emailAliases?.split(',') || [];
|
||||||
|
|
||||||
const isContactCreationForSentAndReceivedEmailsEnabled =
|
const isContactCreationForSentAndReceivedEmailsEnabled =
|
||||||
isContactCreationForSentAndReceivedEmailsEnabledFeatureFlag?.value;
|
isContactCreationForSentAndReceivedEmailsEnabledFeatureFlag?.value;
|
||||||
|
|
||||||
@ -80,15 +82,21 @@ export class MessagingSaveMessagesAndEnqueueContactCreationService {
|
|||||||
const messageId = messageExternalIdsAndIdsMap.get(message.externalId);
|
const messageId = messageExternalIdsAndIdsMap.get(message.externalId);
|
||||||
|
|
||||||
return messageId
|
return messageId
|
||||||
? message.participants.map((participant: Participant) => ({
|
? message.participants.map((participant: Participant) => {
|
||||||
...participant,
|
const fromHandle =
|
||||||
messageId,
|
message.participants.find((p) => p.role === 'from')?.handle ||
|
||||||
shouldCreateContact:
|
'';
|
||||||
messageChannel.isContactAutoCreationEnabled &&
|
|
||||||
(isContactCreationForSentAndReceivedEmailsEnabled ||
|
return {
|
||||||
message.participants.find((p) => p.role === 'from')
|
...participant,
|
||||||
?.handle === connectedAccount.handle),
|
messageId,
|
||||||
}))
|
shouldCreateContact:
|
||||||
|
messageChannel.isContactAutoCreationEnabled &&
|
||||||
|
(isContactCreationForSentAndReceivedEmailsEnabled ||
|
||||||
|
emailAliases.includes(fromHandle)) &&
|
||||||
|
!emailAliases.includes(participant.handle),
|
||||||
|
};
|
||||||
|
})
|
||||||
: [];
|
: [];
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@ -2,8 +2,11 @@ import { Module } from '@nestjs/common';
|
|||||||
import { TypeOrmModule } from '@nestjs/typeorm';
|
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 { FeatureFlagModule } from 'src/engine/core-modules/feature-flag/feature-flag.module';
|
||||||
import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module';
|
import { EnvironmentModule } from 'src/engine/integrations/environment/environment.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';
|
||||||
|
import { EmailAliasManagerModule } from 'src/modules/connected-account/email-alias-manager/email-alias-manager.module';
|
||||||
|
import { OAuth2ClientManagerModule } from 'src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.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 { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity';
|
import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity';
|
||||||
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
@ -30,6 +33,9 @@ import { MessagingGmailPartialMessageListFetchService } from 'src/modules/messag
|
|||||||
]),
|
]),
|
||||||
MessagingCommonModule,
|
MessagingCommonModule,
|
||||||
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
|
TypeOrmModule.forFeature([FeatureFlagEntity], 'core'),
|
||||||
|
OAuth2ClientManagerModule,
|
||||||
|
EmailAliasManagerModule,
|
||||||
|
FeatureFlagModule,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
MessagingGmailClientProvider,
|
MessagingGmailClientProvider,
|
||||||
|
|||||||
@ -1,16 +1,21 @@
|
|||||||
import { Injectable } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
import { OAuth2Client } from 'google-auth-library';
|
|
||||||
import { gmail_v1, google } from 'googleapis';
|
import { gmail_v1, google } from 'googleapis';
|
||||||
|
|
||||||
import { EnvironmentService } from 'src/engine/integrations/environment/environment.service';
|
import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service';
|
||||||
|
import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class MessagingGmailClientProvider {
|
export class MessagingGmailClientProvider {
|
||||||
constructor(private readonly environmentService: EnvironmentService) {}
|
constructor(
|
||||||
|
private readonly oAuth2ClientManagerService: OAuth2ClientManagerService,
|
||||||
|
) {}
|
||||||
|
|
||||||
public async getGmailClient(refreshToken: string): Promise<gmail_v1.Gmail> {
|
public async getGmailClient(
|
||||||
const oAuth2Client = await this.getOAuth2Client(refreshToken);
|
connectedAccount: ConnectedAccountWorkspaceEntity,
|
||||||
|
): Promise<gmail_v1.Gmail> {
|
||||||
|
const oAuth2Client =
|
||||||
|
await this.oAuth2ClientManagerService.getOAuth2Client(connectedAccount);
|
||||||
|
|
||||||
const gmailClient = google.gmail({
|
const gmailClient = google.gmail({
|
||||||
version: 'v1',
|
version: 'v1',
|
||||||
@ -19,22 +24,4 @@ export class MessagingGmailClientProvider {
|
|||||||
|
|
||||||
return gmailClient;
|
return gmailClient;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async getOAuth2Client(refreshToken: string): Promise<OAuth2Client> {
|
|
||||||
const gmailClientId = this.environmentService.get('AUTH_GOOGLE_CLIENT_ID');
|
|
||||||
const gmailClientSecret = this.environmentService.get(
|
|
||||||
'AUTH_GOOGLE_CLIENT_SECRET',
|
|
||||||
);
|
|
||||||
|
|
||||||
const oAuth2Client = new google.auth.OAuth2(
|
|
||||||
gmailClientId,
|
|
||||||
gmailClientSecret,
|
|
||||||
);
|
|
||||||
|
|
||||||
oAuth2Client.setCredentials({
|
|
||||||
refresh_token: refreshToken,
|
|
||||||
});
|
|
||||||
|
|
||||||
return oAuth2Client;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -54,9 +54,7 @@ export class MessagingGmailFullMessageListFetchService {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const gmailClient: gmail_v1.Gmail =
|
const gmailClient: gmail_v1.Gmail =
|
||||||
await this.gmailClientProvider.getGmailClient(
|
await this.gmailClientProvider.getGmailClient(connectedAccount);
|
||||||
connectedAccount.refreshToken,
|
|
||||||
);
|
|
||||||
|
|
||||||
const { error: gmailError } =
|
const { error: gmailError } =
|
||||||
await this.fetchAllMessageIdsFromGmailAndStoreInCache(
|
await this.fetchAllMessageIdsFromGmailAndStoreInCache(
|
||||||
|
|||||||
@ -20,6 +20,9 @@ import { MessagingGmailFetchMessagesByBatchesService } from 'src/modules/messagi
|
|||||||
import { MessagingErrorHandlingService } from 'src/modules/messaging/common/services/messaging-error-handling.service';
|
import { MessagingErrorHandlingService } from 'src/modules/messaging/common/services/messaging-error-handling.service';
|
||||||
import { MessagingSaveMessagesAndEnqueueContactCreationService } from 'src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service';
|
import { MessagingSaveMessagesAndEnqueueContactCreationService } from 'src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service';
|
||||||
import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository';
|
import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository';
|
||||||
|
import { EmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/services/email-alias-manager.service';
|
||||||
|
import { IsFeatureEnabledService } from 'src/engine/core-modules/feature-flag/services/is-feature-enabled.service';
|
||||||
|
import { FeatureFlagKeys } from 'src/engine/core-modules/feature-flag/feature-flag.entity';
|
||||||
import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository';
|
import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository';
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
@ -41,6 +44,8 @@ export class MessagingGmailMessagesImportService {
|
|||||||
private readonly blocklistRepository: BlocklistRepository,
|
private readonly blocklistRepository: BlocklistRepository,
|
||||||
@InjectObjectMetadataRepository(MessageChannelWorkspaceEntity)
|
@InjectObjectMetadataRepository(MessageChannelWorkspaceEntity)
|
||||||
private readonly messageChannelRepository: MessageChannelRepository,
|
private readonly messageChannelRepository: MessageChannelRepository,
|
||||||
|
private readonly emailAliasManagerService: EmailAliasManagerService,
|
||||||
|
private readonly isFeatureEnabledService: IsFeatureEnabledService,
|
||||||
@InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity)
|
@InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity)
|
||||||
private readonly connectedAccountRepository: ConnectedAccountRepository,
|
private readonly connectedAccountRepository: ConnectedAccountRepository,
|
||||||
) {}
|
) {}
|
||||||
@ -78,8 +83,8 @@ export class MessagingGmailMessagesImportService {
|
|||||||
try {
|
try {
|
||||||
accessToken =
|
accessToken =
|
||||||
await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken(
|
await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken(
|
||||||
|
connectedAccount,
|
||||||
workspaceId,
|
workspaceId,
|
||||||
connectedAccount.id,
|
|
||||||
);
|
);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
await this.messagingTelemetryService.track({
|
await this.messagingTelemetryService.track({
|
||||||
@ -103,6 +108,30 @@ export class MessagingGmailMessagesImportService {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
await this.isFeatureEnabledService.isFeatureEnabled(
|
||||||
|
FeatureFlagKeys.IsMessagingAliasFetchingEnabled,
|
||||||
|
workspaceId,
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
await this.emailAliasManagerService.refreshEmailAliases(
|
||||||
|
connectedAccount,
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
await this.gmailErrorHandlingService.handleGmailError(
|
||||||
|
{
|
||||||
|
code: error.code,
|
||||||
|
reason: error.message,
|
||||||
|
},
|
||||||
|
'messages-import',
|
||||||
|
messageChannel,
|
||||||
|
workspaceId,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const messageIdsToFetch =
|
const messageIdsToFetch =
|
||||||
(await this.cacheStorage.setPop(
|
(await this.cacheStorage.setPop(
|
||||||
`messages-to-import:${workspaceId}:gmail:${messageChannel.id}`,
|
`messages-to-import:${workspaceId}:gmail:${messageChannel.id}`,
|
||||||
|
|||||||
@ -52,9 +52,7 @@ export class MessagingGmailPartialMessageListFetchService {
|
|||||||
const lastSyncHistoryId = messageChannel.syncCursor;
|
const lastSyncHistoryId = messageChannel.syncCursor;
|
||||||
|
|
||||||
const gmailClient: gmail_v1.Gmail =
|
const gmailClient: gmail_v1.Gmail =
|
||||||
await this.gmailClientProvider.getGmailClient(
|
await this.gmailClientProvider.getGmailClient(connectedAccount);
|
||||||
connectedAccount.refreshToken,
|
|
||||||
);
|
|
||||||
|
|
||||||
const { history, historyId, error } =
|
const { history, historyId, error } =
|
||||||
await this.gmailGetHistoryService.getHistory(
|
await this.gmailGetHistoryService.getHistory(
|
||||||
|
|||||||
Reference in New Issue
Block a user