[permissions] Remove raw queries and restrict its usage (#12360)

Closes https://github.com/twentyhq/core-team-issues/issues/748

In the frame of the work on permissions we

- remove all raw queries possible to use repositories instead
- forbid usage workspaceDataSource.executeRawQueries()
- restrict usage of workspaceDataSource.query() to force developers to
pass on shouldBypassPermissionChecks to use it.

---------

Co-authored-by: greptile-apps[bot] <165735046+greptile-apps[bot]@users.noreply.github.com>
This commit is contained in:
Marie
2025-06-02 10:53:51 +02:00
committed by GitHub
parent 1ef7b7a474
commit 9706f0df13
49 changed files with 495 additions and 754 deletions

View File

@ -235,6 +235,11 @@ export class MigrateRichTextContentPatchCommand extends ActiveOrSuspendedWorkspa
const rows = await workspaceDataSource.query( const rows = await workspaceDataSource.query(
`SELECT id, "${richTextField.name}" FROM "${schemaName}"."${computeTableName(objectMetadata.nameSingular, objectMetadata.isCustom)}" WHERE "${richTextField.name}" IS NOT NULL`, `SELECT id, "${richTextField.name}" FROM "${schemaName}"."${computeTableName(objectMetadata.nameSingular, objectMetadata.isCustom)}" WHERE "${richTextField.name}" IS NOT NULL`,
undefined, // parameters
undefined, // queryRunner
{
shouldBypassPermissionChecks: true,
},
); );
this.logger.log(`Generating markdown for ${rows.length} records`); this.logger.log(`Generating markdown for ${rows.length} records`);
@ -251,6 +256,10 @@ export class MigrateRichTextContentPatchCommand extends ActiveOrSuspendedWorkspa
await workspaceDataSource.query( await workspaceDataSource.query(
`UPDATE "${schemaName}"."${computeTableName(objectMetadata.nameSingular, objectMetadata.isCustom)}" SET "${richTextField.name}V2Blocknote" = $1, "${richTextField.name}V2Markdown" = $2 WHERE id = $3`, `UPDATE "${schemaName}"."${computeTableName(objectMetadata.nameSingular, objectMetadata.isCustom)}" SET "${richTextField.name}V2Blocknote" = $1, "${richTextField.name}V2Markdown" = $2 WHERE id = $3`,
[blocknoteFieldValue, markdownFieldValue, row.id], [blocknoteFieldValue, markdownFieldValue, row.id],
undefined, // queryRunner
{
shouldBypassPermissionChecks: true,
},
); );
} catch (error) { } catch (error) {
this.logger.log( this.logger.log(

View File

@ -1,20 +1,20 @@
import { InjectRepository } from '@nestjs/typeorm'; import { InjectRepository } from '@nestjs/typeorm';
import { Command } from 'nest-commander'; import { Command } from 'nest-commander';
import { Repository } from 'typeorm';
import { FieldMetadataType } from 'twenty-shared/types'; import { FieldMetadataType } from 'twenty-shared/types';
import { Repository } from 'typeorm';
import { import {
ActiveOrSuspendedWorkspacesMigrationCommandRunner, ActiveOrSuspendedWorkspacesMigrationCommandRunner,
RunOnWorkspaceArgs, RunOnWorkspaceArgs,
} from 'src/database/commands/command-runners/active-or-suspended-workspaces-migration.command-runner'; } from 'src/database/commands/command-runners/active-or-suspended-workspaces-migration.command-runner';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { ActorMetadata } from 'src/engine/metadata-modules/field-metadata/composite-types/actor.composite-type'; import { ActorMetadata } from 'src/engine/metadata-modules/field-metadata/composite-types/actor.composite-type';
import { generateDefaultValue } from 'src/engine/metadata-modules/field-metadata/utils/generate-default-value'; import { generateDefaultValue } from 'src/engine/metadata-modules/field-metadata/utils/generate-default-value';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { computeTableName } from 'src/engine/utils/compute-table-name.util'; import { computeTableName } from 'src/engine/utils/compute-table-name.util';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
@Command({ @Command({
name: 'upgrade:0-54:0-54-created-by-default-value', name: 'upgrade:0-54:0-54-created-by-default-value',
@ -59,12 +59,19 @@ export class FixCreatedByDefaultValueCommand extends ActiveOrSuspendedWorkspaces
); );
const actualDefaultValue = ( const actualDefaultValue = (
await dataSource.query(` await dataSource.query(
`
SELECT column_default FROM information_schema.columns SELECT column_default FROM information_schema.columns
WHERE table_schema = '${schemaName}' WHERE table_schema = '${schemaName}'
AND table_name = '${tableName}' AND table_name = '${tableName}'
AND column_name = 'createdBySource'; AND column_name = 'createdBySource';
`) `,
undefined, // parameters
undefined, // queryRunner
{
shouldBypassPermissionChecks: true,
},
)
)?.[0]?.column_default; )?.[0]?.column_default;
if (actualDefaultValue !== null) { if (actualDefaultValue !== null) {
@ -75,12 +82,19 @@ export class FixCreatedByDefaultValueCommand extends ActiveOrSuspendedWorkspaces
FieldMetadataType.ACTOR, FieldMetadataType.ACTOR,
) as ActorMetadata; ) as ActorMetadata;
await dataSource.query(` await dataSource.query(
`
ALTER TABLE "${schemaName}"."${tableName}" ALTER TABLE "${schemaName}"."${tableName}"
ALTER COLUMN "createdBySource" SET DEFAULT ${createdByDefaultValues.source}, ALTER COLUMN "createdBySource" SET DEFAULT ${createdByDefaultValues.source},
ALTER COLUMN "createdByName" SET DEFAULT ${createdByDefaultValues.name}, ALTER COLUMN "createdByName" SET DEFAULT ${createdByDefaultValues.name},
ALTER COLUMN "createdByContext" SET DEFAULT '${JSON.stringify(createdByDefaultValues.context)}'; ALTER COLUMN "createdByContext" SET DEFAULT '${JSON.stringify(createdByDefaultValues.context)}';
`); `,
undefined, // parameters
undefined, // queryRunner
{
shouldBypassPermissionChecks: true,
},
);
} }
} }
} }

View File

@ -3,11 +3,11 @@ import { DataSource } from 'typeorm';
const tableName = 'billingSubscription'; const tableName = 'billingSubscription';
export const seedBillingSubscriptions = async ( export const seedBillingSubscriptions = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, [ .into(`${schemaName}.${tableName}`, [

View File

@ -3,11 +3,11 @@ import { DataSource } from 'typeorm';
const tableName = 'featureFlag'; const tableName = 'featureFlag';
export const deleteFeatureFlags = async ( export const deleteFeatureFlags = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.delete() .delete()
.from(`${schemaName}.${tableName}`) .from(`${schemaName}.${tableName}`)

View File

@ -11,11 +11,11 @@ export const DEV_SEED_USER_WORKSPACE_IDS = {
}; };
export const seedUserWorkspaces = async ( export const seedUserWorkspaces = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, ['id', 'userId', 'workspaceId']) .into(`${schemaName}.${tableName}`, ['id', 'userId', 'workspaceId'])
@ -41,11 +41,11 @@ export const seedUserWorkspaces = async (
}; };
export const deleteUserWorkspaces = async ( export const deleteUserWorkspaces = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.delete() .delete()
.from(`${schemaName}.${tableName}`) .from(`${schemaName}.${tableName}`)

View File

@ -10,11 +10,8 @@ export const DEMO_SEED_USER_IDS = {
TIM: '20202020-9e3b-46d4-a556-88b9ddc2b034', TIM: '20202020-9e3b-46d4-a556-88b9ddc2b034',
}; };
export const seedUsers = async ( export const seedUsers = async (dataSource: DataSource, schemaName: string) => {
workspaceDataSource: DataSource, await dataSource
schemaName: string,
) => {
await workspaceDataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, [ .into(`${schemaName}.${tableName}`, [

View File

@ -1,14 +1,14 @@
import { DataSource } from 'typeorm';
import { WorkspaceActivationStatus } from 'twenty-shared/workspace'; import { WorkspaceActivationStatus } from 'twenty-shared/workspace';
import { DataSource } from 'typeorm';
const tableName = 'workspace'; const tableName = 'workspace';
export const seedWorkspaces = async ( export const seedWorkspaces = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, [ .into(`${schemaName}.${tableName}`, [
@ -36,11 +36,11 @@ export const seedWorkspaces = async (
}; };
export const deleteWorkspaces = async ( export const deleteWorkspaces = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.delete() .delete()
.from(`${schemaName}.${tableName}`) .from(`${schemaName}.${tableName}`)

View File

@ -5,11 +5,11 @@ import { FeatureFlagKey } from 'src/engine/core-modules/feature-flag/enums/featu
const tableName = 'featureFlag'; const tableName = 'featureFlag';
export const seedFeatureFlags = async ( export const seedFeatureFlags = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, ['key', 'workspaceId', 'value']) .into(`${schemaName}.${tableName}`, ['key', 'workspaceId', 'value'])
@ -45,11 +45,11 @@ export const seedFeatureFlags = async (
}; };
export const deleteFeatureFlags = async ( export const deleteFeatureFlags = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.delete() .delete()
.from(`${schemaName}.${tableName}`) .from(`${schemaName}.${tableName}`)

View File

@ -17,7 +17,7 @@ export const DEV_SEED_USER_WORKSPACE_IDS = {
}; };
export const seedUserWorkspaces = async ( export const seedUserWorkspaces = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
@ -53,7 +53,7 @@ export const seedUserWorkspaces = async (
}, },
]; ];
} }
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, ['id', 'userId', 'workspaceId']) .into(`${schemaName}.${tableName}`, ['id', 'userId', 'workspaceId'])
@ -63,11 +63,11 @@ export const seedUserWorkspaces = async (
}; };
export const deleteUserWorkspaces = async ( export const deleteUserWorkspaces = async (
workspaceDataSource: DataSource, dataSource: DataSource,
schemaName: string, schemaName: string,
workspaceId: string, workspaceId: string,
) => { ) => {
await workspaceDataSource await dataSource
.createQueryBuilder() .createQueryBuilder()
.delete() .delete()
.from(`${schemaName}.${tableName}`) .from(`${schemaName}.${tableName}`)

View File

@ -8,11 +8,8 @@ export const DEV_SEED_USER_IDS = {
PHIL: '20202020-7169-42cf-bc47-1cfef15264b8', PHIL: '20202020-7169-42cf-bc47-1cfef15264b8',
}; };
export const seedUsers = async ( export const seedUsers = async (dataSource: DataSource, schemaName: string) => {
workspaceDataSource: DataSource, await dataSource
schemaName: string,
) => {
await workspaceDataSource
.createQueryBuilder() .createQueryBuilder()
.insert() .insert()
.into(`${schemaName}.${tableName}`, [ .into(`${schemaName}.${tableName}`, [

View File

@ -1,13 +1,9 @@
import { import { Brackets, NotBrackets, WhereExpressionBuilder } from 'typeorm';
Brackets,
NotBrackets,
SelectQueryBuilder,
WhereExpressionBuilder,
} from 'typeorm';
import { ObjectRecordFilter } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface'; import { ObjectRecordFilter } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface';
import { FieldMetadataMap } from 'src/engine/metadata-modules/types/field-metadata-map'; import { FieldMetadataMap } from 'src/engine/metadata-modules/types/field-metadata-map';
import { WorkspaceSelectQueryBuilder } from 'src/engine/twenty-orm/repository/workspace-select-query-builder';
import { GraphqlQueryFilterFieldParser } from './graphql-query-filter-field.parser'; import { GraphqlQueryFilterFieldParser } from './graphql-query-filter-field.parser';
@ -30,11 +26,11 @@ export class GraphqlQueryFilterConditionParser {
public parse( public parse(
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
queryBuilder: SelectQueryBuilder<any>, queryBuilder: WorkspaceSelectQueryBuilder<any>,
objectNameSingular: string, objectNameSingular: string,
filter: Partial<ObjectRecordFilter>, filter: Partial<ObjectRecordFilter>,
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
): SelectQueryBuilder<any> { ): WorkspaceSelectQueryBuilder<any> {
if (!filter || Object.keys(filter).length === 0) { if (!filter || Object.keys(filter).length === 0) {
return queryBuilder; return queryBuilder;
} }

View File

@ -1,9 +1,4 @@
import { import { FindOptionsWhere, ObjectLiteral, OrderByCondition } from 'typeorm';
FindOptionsWhere,
ObjectLiteral,
OrderByCondition,
SelectQueryBuilder,
} from 'typeorm';
import { import {
ObjectRecordFilter, ObjectRecordFilter,
@ -24,6 +19,7 @@ import { FieldMetadataMap } from 'src/engine/metadata-modules/types/field-metada
import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps'; import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps';
import { ObjectMetadataMaps } from 'src/engine/metadata-modules/types/object-metadata-maps'; import { ObjectMetadataMaps } from 'src/engine/metadata-modules/types/object-metadata-maps';
import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util'; import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util';
import { WorkspaceSelectQueryBuilder } from 'src/engine/twenty-orm/repository/workspace-select-query-builder';
export class GraphqlQueryParser { export class GraphqlQueryParser {
private fieldMetadataMapByName: FieldMetadataMap; private fieldMetadataMapByName: FieldMetadataMap;
@ -51,11 +47,11 @@ export class GraphqlQueryParser {
public applyFilterToBuilder( public applyFilterToBuilder(
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
queryBuilder: SelectQueryBuilder<any>, queryBuilder: WorkspaceSelectQueryBuilder<any>,
objectNameSingular: string, objectNameSingular: string,
recordFilter: Partial<ObjectRecordFilter>, recordFilter: Partial<ObjectRecordFilter>,
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
): SelectQueryBuilder<any> { ): WorkspaceSelectQueryBuilder<any> {
return this.filterConditionParser.parse( return this.filterConditionParser.parse(
queryBuilder, queryBuilder,
objectNameSingular, objectNameSingular,
@ -65,10 +61,10 @@ export class GraphqlQueryParser {
public applyDeletedAtToBuilder( public applyDeletedAtToBuilder(
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
queryBuilder: SelectQueryBuilder<any>, queryBuilder: WorkspaceSelectQueryBuilder<any>,
recordFilter: Partial<ObjectRecordFilter>, recordFilter: Partial<ObjectRecordFilter>,
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
): SelectQueryBuilder<any> { ): WorkspaceSelectQueryBuilder<any> {
if (this.checkForDeletedAtFilter(recordFilter)) { if (this.checkForDeletedAtFilter(recordFilter)) {
queryBuilder.withDeleted(); queryBuilder.withDeleted();
} }
@ -104,12 +100,12 @@ export class GraphqlQueryParser {
public applyOrderToBuilder( public applyOrderToBuilder(
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
queryBuilder: SelectQueryBuilder<any>, queryBuilder: WorkspaceSelectQueryBuilder<any>,
orderBy: ObjectRecordOrderBy, orderBy: ObjectRecordOrderBy,
objectNameSingular: string, objectNameSingular: string,
isForwardPagination = true, isForwardPagination = true,
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
): SelectQueryBuilder<any> { ): WorkspaceSelectQueryBuilder<any> {
const parsedOrderBys = this.orderFieldParser.parse( const parsedOrderBys = this.orderFieldParser.parse(
orderBy, orderBy,
objectNameSingular, objectNameSingular,

View File

@ -5,6 +5,7 @@ import { SelectQueryBuilder } from 'typeorm';
import { AggregateOperations } from 'src/engine/api/graphql/graphql-query-runner/constants/aggregate-operations.constant'; import { AggregateOperations } from 'src/engine/api/graphql/graphql-query-runner/constants/aggregate-operations.constant';
import { AggregationField } from 'src/engine/api/graphql/workspace-schema-builder/utils/get-available-aggregations-from-object-fields.util'; import { AggregationField } from 'src/engine/api/graphql/workspace-schema-builder/utils/get-available-aggregations-from-object-fields.util';
import { WorkspaceSelectQueryBuilder } from 'src/engine/twenty-orm/repository/workspace-select-query-builder';
import { formatColumnNamesFromCompositeFieldAndSubfields } from 'src/engine/twenty-orm/utils/format-column-names-from-composite-field-and-subfield.util'; import { formatColumnNamesFromCompositeFieldAndSubfields } from 'src/engine/twenty-orm/utils/format-column-names-from-composite-field-and-subfield.util';
@Injectable() @Injectable()
@ -15,7 +16,7 @@ export class ProcessAggregateHelper {
}: { }: {
selectedAggregatedFields: Record<string, AggregationField>; selectedAggregatedFields: Record<string, AggregationField>;
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
queryBuilder: SelectQueryBuilder<any>; queryBuilder: WorkspaceSelectQueryBuilder<any>;
}) => { }) => {
queryBuilder.select([]); queryBuilder.select([]);

View File

@ -1,11 +1,7 @@
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { FieldMetadataType } from 'twenty-shared/types'; import { FieldMetadataType } from 'twenty-shared/types';
import { import { FindOptionsRelations, ObjectLiteral } from 'typeorm';
FindOptionsRelations,
ObjectLiteral,
SelectQueryBuilder,
} from 'typeorm';
import { ObjectRecord } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface'; import { ObjectRecord } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface';
import { RelationType } from 'src/engine/metadata-modules/field-metadata/interfaces/relation-type.interface'; import { RelationType } from 'src/engine/metadata-modules/field-metadata/interfaces/relation-type.interface';
@ -22,6 +18,7 @@ import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/typ
import { ObjectMetadataMaps } from 'src/engine/metadata-modules/types/object-metadata-maps'; import { ObjectMetadataMaps } from 'src/engine/metadata-modules/types/object-metadata-maps';
import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util'; import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util';
import { WorkspaceDataSource } from 'src/engine/twenty-orm/datasource/workspace.datasource'; import { WorkspaceDataSource } from 'src/engine/twenty-orm/datasource/workspace.datasource';
import { WorkspaceSelectQueryBuilder } from 'src/engine/twenty-orm/repository/workspace-select-query-builder';
import { formatResult } from 'src/engine/twenty-orm/utils/format-result.util'; import { formatResult } from 'src/engine/twenty-orm/utils/format-result.util';
import { isFieldMetadataInterfaceOfType } from 'src/engine/utils/is-field-metadata-of-type.util'; import { isFieldMetadataInterfaceOfType } from 'src/engine/utils/is-field-metadata-of-type.util';
@ -269,7 +266,7 @@ export class ProcessNestedRelationsV2Helper {
sourceFieldName, sourceFieldName,
}: { }: {
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
referenceQueryBuilder: SelectQueryBuilder<any>; referenceQueryBuilder: WorkspaceSelectQueryBuilder<any>;
column: string; column: string;
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
ids: any[]; ids: any[];

View File

@ -13,9 +13,7 @@ import { FileModule } from 'src/engine/core-modules/file/file.module';
import { RecordPositionModule } from 'src/engine/core-modules/record-position/record-position.module'; import { RecordPositionModule } from 'src/engine/core-modules/record-position/record-position.module';
import { RecordTransformerModule } from 'src/engine/core-modules/record-transformer/record-transformer.module'; import { RecordTransformerModule } from 'src/engine/core-modules/record-transformer/record-transformer.module';
import { TelemetryModule } from 'src/engine/core-modules/telemetry/telemetry.module'; import { TelemetryModule } from 'src/engine/core-modules/telemetry/telemetry.module';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
import { EntityEventsToDbListener } from './listeners/entity-events-to-db.listener'; import { EntityEventsToDbListener } from './listeners/entity-events-to-db.listener';
@ -25,7 +23,6 @@ import { EntityEventsToDbListener } from './listeners/entity-events-to-db.listen
WorkspaceQueryBuilderModule, WorkspaceQueryBuilderModule,
WorkspaceDataSourceModule, WorkspaceDataSourceModule,
WorkspaceQueryHookModule, WorkspaceQueryHookModule,
ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]),
TypeOrmModule.forFeature([FeatureFlag], 'core'), TypeOrmModule.forFeature([FeatureFlag], 'core'),
AuditModule, AuditModule,
TelemetryModule, TelemetryModule,

View File

@ -3,7 +3,7 @@ import { BadRequestException, Inject } from '@nestjs/common';
import { Request } from 'express'; import { Request } from 'express';
import { FieldMetadataType } from 'twenty-shared/types'; import { FieldMetadataType } from 'twenty-shared/types';
import { capitalize, isDefined } from 'twenty-shared/utils'; import { capitalize, isDefined } from 'twenty-shared/utils';
import { In, ObjectLiteral, SelectQueryBuilder } from 'typeorm'; import { In, ObjectLiteral } from 'typeorm';
import { import {
ObjectRecord, ObjectRecord,
@ -12,6 +12,7 @@ import {
import { GraphqlQueryParser } from 'src/engine/api/graphql/graphql-query-runner/graphql-query-parsers/graphql-query.parser'; import { GraphqlQueryParser } from 'src/engine/api/graphql/graphql-query-runner/graphql-query-parsers/graphql-query.parser';
import { ApiEventEmitterService } from 'src/engine/api/graphql/graphql-query-runner/services/api-event-emitter.service'; import { ApiEventEmitterService } from 'src/engine/api/graphql/graphql-query-runner/services/api-event-emitter.service';
import { encodeCursor } from 'src/engine/api/graphql/graphql-query-runner/utils/cursors.util';
import { CoreQueryBuilderFactory } from 'src/engine/api/rest/core/query-builder/core-query-builder.factory'; import { CoreQueryBuilderFactory } from 'src/engine/api/rest/core/query-builder/core-query-builder.factory';
import { GetVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/get-variables.factory'; import { GetVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/get-variables.factory';
import { parseCorePath } from 'src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-path.utils'; import { parseCorePath } from 'src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-path.utils';
@ -21,18 +22,18 @@ import {
DepthInputFactory, DepthInputFactory,
MAX_DEPTH, MAX_DEPTH,
} from 'src/engine/api/rest/input-factories/depth-input.factory'; } from 'src/engine/api/rest/input-factories/depth-input.factory';
import { computeCursorArgFilter } from 'src/engine/api/utils/compute-cursor-arg-filter.utils';
import { CreatedByFromAuthContextService } from 'src/engine/core-modules/actor/services/created-by-from-auth-context.service';
import { AuthContext } from 'src/engine/core-modules/auth/types/auth-context.type'; import { AuthContext } from 'src/engine/core-modules/auth/types/auth-context.type';
import { RecordInputTransformerService } from 'src/engine/core-modules/record-transformer/services/record-input-transformer.service'; import { RecordInputTransformerService } from 'src/engine/core-modules/record-transformer/services/record-input-transformer.service';
import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps'; import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps';
import { ObjectMetadataMaps } from 'src/engine/metadata-modules/types/object-metadata-maps'; import { ObjectMetadataMaps } from 'src/engine/metadata-modules/types/object-metadata-maps';
import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util'; import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util';
import { WorkspacePermissionsCacheService } from 'src/engine/metadata-modules/workspace-permissions-cache/workspace-permissions-cache.service'; import { WorkspacePermissionsCacheService } from 'src/engine/metadata-modules/workspace-permissions-cache/workspace-permissions-cache.service';
import { WorkspaceSelectQueryBuilder } from 'src/engine/twenty-orm/repository/workspace-select-query-builder';
import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { formatResult as formatGetManyData } from 'src/engine/twenty-orm/utils/format-result.util'; import { formatResult as formatGetManyData } from 'src/engine/twenty-orm/utils/format-result.util';
import { encodeCursor } from 'src/engine/api/graphql/graphql-query-runner/utils/cursors.util';
import { computeCursorArgFilter } from 'src/engine/api/utils/compute-cursor-arg-filter.utils';
import { CreatedByFromAuthContextService } from 'src/engine/core-modules/actor/services/created-by-from-auth-context.service';
export interface PageInfo { export interface PageInfo {
hasNextPage?: boolean; hasNextPage?: boolean;
@ -392,7 +393,7 @@ export abstract class RestApiBaseHandler {
} }
async getTotalCount( async getTotalCount(
query: SelectQueryBuilder<ObjectLiteral>, query: WorkspaceSelectQueryBuilder<ObjectLiteral>,
): Promise<number> { ): Promise<number> {
const countQuery = query.clone(); const countQuery = query.clone();

View File

@ -2,16 +2,10 @@ import { Module } from '@nestjs/common';
import { AuditModule } from 'src/engine/core-modules/audit/audit.module'; import { AuditModule } from 'src/engine/core-modules/audit/audit.module';
import { CreateAuditLogFromInternalEvent } from 'src/engine/core-modules/audit/jobs/create-audit-log-from-internal-event'; import { CreateAuditLogFromInternalEvent } from 'src/engine/core-modules/audit/jobs/create-audit-log-from-internal-event';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { TimelineActivityModule } from 'src/modules/timeline/timeline-activity.module'; import { TimelineActivityModule } from 'src/modules/timeline/timeline-activity.module';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [ imports: [TimelineActivityModule, AuditModule],
ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]),
TimelineActivityModule,
AuditModule,
],
providers: [CreateAuditLogFromInternalEvent], providers: [CreateAuditLogFromInternalEvent],
}) })
export class AuditJobModule {} export class AuditJobModule {}

View File

@ -1,11 +1,11 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module';
import { RecordPositionService } from './services/record-position.service'; import { RecordPositionService } from './services/record-position.service';
@Module({ @Module({
imports: [WorkspaceDataSourceModule], imports: [TwentyORMModule],
providers: [RecordPositionService], providers: [RecordPositionService],
exports: [RecordPositionService], exports: [RecordPositionService],
}) })

View File

@ -1,24 +1,31 @@
import { Test, TestingModule } from '@nestjs/testing'; import { Test, TestingModule } from '@nestjs/testing';
import { RecordPositionService } from 'src/engine/core-modules/record-position/services/record-position.service'; import { RecordPositionService } from 'src/engine/core-modules/record-position/services/record-position.service';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
describe('RecordPositionService', () => { describe('RecordPositionService', () => {
let workspaceDataSourceService; let twentyORMGlobalManager: jest.Mocked<TwentyORMGlobalManager>;
let mockRepository: any;
let service: RecordPositionService; let service: RecordPositionService;
beforeEach(async () => { beforeEach(async () => {
workspaceDataSourceService = { mockRepository = {
getSchemaName: jest.fn().mockReturnValue('schemaName'), findOneBy: jest.fn(),
executeRawQuery: jest.fn().mockResolvedValue([{ position: 1 }]), update: jest.fn(),
minimum: jest.fn().mockResolvedValue(1),
maximum: jest.fn().mockResolvedValue(1),
}; };
twentyORMGlobalManager = {
getRepositoryForWorkspace: jest.fn().mockResolvedValue(mockRepository),
} as unknown as jest.Mocked<TwentyORMGlobalManager>;
const module: TestingModule = await Test.createTestingModule({ const module: TestingModule = await Test.createTestingModule({
providers: [ providers: [
RecordPositionService, RecordPositionService,
{ {
provide: WorkspaceDataSourceService, provide: TwentyORMGlobalManager,
useValue: workspaceDataSourceService, useValue: twentyORMGlobalManager,
}, },
], ],
}).compile(); }).compile();
@ -30,7 +37,7 @@ describe('RecordPositionService', () => {
expect(service).toBeDefined(); expect(service).toBeDefined();
}); });
describe('create', () => { describe('buildRecordPosition', () => {
const objectMetadata = { isCustom: false, nameSingular: 'company' }; const objectMetadata = { isCustom: false, nameSingular: 'company' };
const workspaceId = 'workspaceId'; const workspaceId = 'workspaceId';

View File

@ -1,13 +1,6 @@
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { isDefined } from 'class-validator'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import {
RecordPositionQueryArgs,
RecordPositionQueryType,
} from 'src/engine/core-modules/record-position/types/record-position-query.type';
import { buildRecordPositionQuery } from 'src/engine/core-modules/record-position/utils/build-record-position-query.util';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
export type RecordPositionServiceCreateArgs = { export type RecordPositionServiceCreateArgs = {
value: number | 'first' | 'last'; value: number | 'first' | 'last';
@ -19,7 +12,7 @@ export type RecordPositionServiceCreateArgs = {
@Injectable() @Injectable()
export class RecordPositionService { export class RecordPositionService {
constructor( constructor(
private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
) {} ) {}
async buildRecordPosition({ async buildRecordPosition({
@ -28,62 +21,101 @@ export class RecordPositionService {
workspaceId, workspaceId,
index = 0, index = 0,
}: RecordPositionServiceCreateArgs): Promise<number> { }: RecordPositionServiceCreateArgs): Promise<number> {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
if (typeof value === 'number') { if (typeof value === 'number') {
return value; return value;
} }
if (value === 'first') { if (value === 'first') {
const recordWithMinPosition = const recordWithMinPosition = await this.findMinPosition(
await this.createAndExecuteRecordPositionQuery(
{
recordPositionQueryType: RecordPositionQueryType.FIND_MIN_POSITION,
},
objectMetadata,
dataSourceSchema,
workspaceId,
);
return isDefined(recordWithMinPosition?.position)
? recordWithMinPosition.position - index - 1
: 1;
}
const recordWithMaxPosition =
await this.createAndExecuteRecordPositionQuery(
{
recordPositionQueryType: RecordPositionQueryType.FIND_MAX_POSITION,
},
objectMetadata, objectMetadata,
dataSourceSchema,
workspaceId, workspaceId,
); );
return isDefined(recordWithMaxPosition?.position) return recordWithMinPosition !== null
? recordWithMaxPosition.position + index + 1 ? recordWithMinPosition - index - 1
: 1; : 1;
} }
private async createAndExecuteRecordPositionQuery( const recordWithMaxPosition = await this.findMaxPosition(
recordPositionQueryArgs: RecordPositionQueryArgs,
objectMetadata: { isCustom: boolean; nameSingular: string },
dataSourceSchema: string,
workspaceId: string,
) {
const [query, params] = buildRecordPositionQuery(
recordPositionQueryArgs,
objectMetadata, objectMetadata,
dataSourceSchema,
);
const records = await this.workspaceDataSourceService.executeRawQuery(
query,
params,
workspaceId, workspaceId,
); );
return records?.[0]; return recordWithMaxPosition !== null
? recordWithMaxPosition + index + 1
: 1;
}
async findByPosition(
positionValue: number | null,
objectMetadata: { isCustom: boolean; nameSingular: string },
workspaceId: string,
): Promise<{ id: string; position: number } | null> {
const repository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
objectMetadata.nameSingular,
{
shouldBypassPermissionChecks: true,
},
);
const record = await repository.findOneBy({
position: positionValue,
});
return record ? { id: record.id, position: record.position } : null;
}
async updatePosition(
recordId: string,
positionValue: number,
objectMetadata: { isCustom: boolean; nameSingular: string },
workspaceId: string,
): Promise<void> {
const repository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
objectMetadata.nameSingular,
{
shouldBypassPermissionChecks: true,
},
);
await repository.update(recordId, {
position: positionValue,
});
}
private async findMinPosition(
objectMetadata: { isCustom: boolean; nameSingular: string },
workspaceId: string,
): Promise<number | null> {
const repository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
objectMetadata.nameSingular,
{
shouldBypassPermissionChecks: true,
},
);
return repository.minimum('position');
}
private async findMaxPosition(
objectMetadata: { isCustom: boolean; nameSingular: string },
workspaceId: string,
): Promise<number | null> {
const repository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
objectMetadata.nameSingular,
{
shouldBypassPermissionChecks: true,
},
);
return repository.maximum('position');
} }
} }

View File

@ -1,72 +0,0 @@
import { RecordPositionQueryType } from 'src/engine/core-modules/record-position/types/record-position-query.type';
import { buildRecordPositionQuery } from 'src/engine/core-modules/record-position/utils/build-record-position-query.util';
describe('buildRecordPositionQuery', () => {
const objectMetadataItem = {
isCustom: false,
nameSingular: 'company',
};
const dataSourceSchema = 'workspace_test';
it('should return query and params for FIND_BY_POSITION', async () => {
const positionValue = 1;
const queryType = RecordPositionQueryType.FIND_BY_POSITION;
const [query, params] = buildRecordPositionQuery(
{ positionValue, recordPositionQueryType: queryType },
objectMetadataItem,
dataSourceSchema,
);
expect(query).toEqual(
`SELECT id, position FROM ${dataSourceSchema}."${objectMetadataItem.nameSingular}"
WHERE "position" = $1`,
);
expect(params).toEqual([positionValue]);
});
it('should return query and params for FIND_MIN_POSITION', async () => {
const queryType = RecordPositionQueryType.FIND_MIN_POSITION;
const [query, params] = buildRecordPositionQuery(
{ recordPositionQueryType: queryType },
objectMetadataItem,
dataSourceSchema,
);
expect(query).toEqual(
`SELECT MIN(position) as position FROM ${dataSourceSchema}."${objectMetadataItem.nameSingular}"`,
);
expect(params).toEqual([]);
});
it('should return query and params for FIND_MAX_POSITION', async () => {
const queryType = RecordPositionQueryType.FIND_MAX_POSITION;
const [query, params] = buildRecordPositionQuery(
{ recordPositionQueryType: queryType },
objectMetadataItem,
dataSourceSchema,
);
expect(query).toEqual(
`SELECT MAX(position) as position FROM ${dataSourceSchema}."${objectMetadataItem.nameSingular}"`,
);
expect(params).toEqual([]);
});
it('should return query and params for UPDATE_POSITION', async () => {
const positionValue = 1;
const recordId = '1';
const queryType = RecordPositionQueryType.UPDATE_POSITION;
const [query, params] = buildRecordPositionQuery(
{ positionValue, recordId, recordPositionQueryType: queryType },
objectMetadataItem,
dataSourceSchema,
);
expect(query).toEqual(
`UPDATE ${dataSourceSchema}."${objectMetadataItem.nameSingular}"
SET "position" = $1
WHERE "id" = $2`,
);
expect(params).toEqual([positionValue, recordId]);
});
});

View File

@ -1,91 +0,0 @@
import {
FindByPositionQueryArgs,
RecordPositionQueryArgs,
RecordPositionQueryType,
UpdatePositionQueryArgs,
} from 'src/engine/core-modules/record-position/types/record-position-query.type';
import { computeTableName } from 'src/engine/utils/compute-table-name.util';
type RecordPositionQuery = string;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
type RecordPositionQueryParams = any[];
export const buildRecordPositionQuery = (
recordPositionQueryArgs: RecordPositionQueryArgs,
objectMetadata: { isCustom: boolean; nameSingular: string },
dataSourceSchema: string,
): [RecordPositionQuery, RecordPositionQueryParams] => {
const tableName = computeTableName(
objectMetadata.nameSingular,
objectMetadata.isCustom,
);
switch (recordPositionQueryArgs.recordPositionQueryType) {
case RecordPositionQueryType.FIND_BY_POSITION:
return buildFindByPositionQuery(
recordPositionQueryArgs satisfies FindByPositionQueryArgs,
tableName,
dataSourceSchema,
);
case RecordPositionQueryType.FIND_MIN_POSITION:
return buildFindMinPositionQuery(tableName, dataSourceSchema);
case RecordPositionQueryType.FIND_MAX_POSITION:
return buildFindMaxPositionQuery(tableName, dataSourceSchema);
case RecordPositionQueryType.UPDATE_POSITION:
return buildUpdatePositionQuery(
recordPositionQueryArgs satisfies UpdatePositionQueryArgs,
tableName,
dataSourceSchema,
);
default:
throw new Error('Invalid RecordPositionQueryType');
}
};
const buildFindByPositionQuery = (
{ positionValue }: FindByPositionQueryArgs,
name: string,
dataSourceSchema: string,
): [RecordPositionQuery, RecordPositionQueryParams] => {
const positionStringParam = positionValue ? '= $1' : 'IS NULL';
return [
`SELECT id, position FROM ${dataSourceSchema}."${name}"
WHERE "position" ${positionStringParam}`,
positionValue ? [positionValue] : [],
];
};
const buildFindMaxPositionQuery = (
name: string,
dataSourceSchema: string,
): [RecordPositionQuery, RecordPositionQueryParams] => {
return [
`SELECT MAX(position) as position FROM ${dataSourceSchema}."${name}"`,
[],
];
};
const buildFindMinPositionQuery = (
name: string,
dataSourceSchema: string,
): [RecordPositionQuery, RecordPositionQueryParams] => {
return [
`SELECT MIN(position) as position FROM ${dataSourceSchema}."${name}"`,
[],
];
};
const buildUpdatePositionQuery = (
{ recordId, positionValue }: UpdatePositionQueryArgs,
name: string,
dataSourceSchema: string,
): [RecordPositionQuery, RecordPositionQueryParams] => {
return [
`UPDATE ${dataSourceSchema}."${name}"
SET "position" = $1
WHERE "id" = $2`,
[positionValue, recordId],
];
};

View File

@ -31,6 +31,8 @@ export enum PermissionsExceptionCode {
ROLE_NOT_EDITABLE = 'ROLE_NOT_EDITABLE', ROLE_NOT_EDITABLE = 'ROLE_NOT_EDITABLE',
DEFAULT_ROLE_CANNOT_BE_DELETED = 'DEFAULT_ROLE_CANNOT_BE_DELETED', DEFAULT_ROLE_CANNOT_BE_DELETED = 'DEFAULT_ROLE_CANNOT_BE_DELETED',
NO_PERMISSIONS_FOUND_IN_DATASOURCE = 'NO_PERMISSIONS_FOUND_IN_DATASOURCE', NO_PERMISSIONS_FOUND_IN_DATASOURCE = 'NO_PERMISSIONS_FOUND_IN_DATASOURCE',
METHOD_NOT_ALLOWED = 'METHOD_NOT_ALLOWED',
RAW_SQL_NOT_ALLOWED = 'RAW_SQL_NOT_ALLOWED',
} }
export enum PermissionsExceptionMessage { export enum PermissionsExceptionMessage {

View File

@ -38,6 +38,8 @@ export const permissionGraphqlApiExceptionHandler = (
case PermissionsExceptionCode.UNKNOWN_REQUIRED_PERMISSION: case PermissionsExceptionCode.UNKNOWN_REQUIRED_PERMISSION:
case PermissionsExceptionCode.NO_ROLE_FOUND_FOR_USER_WORKSPACE: case PermissionsExceptionCode.NO_ROLE_FOUND_FOR_USER_WORKSPACE:
case PermissionsExceptionCode.NO_PERMISSIONS_FOUND_IN_DATASOURCE: case PermissionsExceptionCode.NO_PERMISSIONS_FOUND_IN_DATASOURCE:
case PermissionsExceptionCode.METHOD_NOT_ALLOWED:
case PermissionsExceptionCode.RAW_SQL_NOT_ALLOWED:
throw error; throw error;
default: { default: {
const _exhaustiveCheck: never = error.code; const _exhaustiveCheck: never = error.code;

View File

@ -271,6 +271,7 @@ export class RemoteServerService<T extends RemoteServerType> {
const [parameters, rawQuery] = const [parameters, rawQuery] =
buildUpdateRemoteServerRawQuery(remoteServerToUpdate); buildUpdateRemoteServerRawQuery(remoteServerToUpdate);
// TO DO: executeRawQuery is deprecated and will throw
const updateResult = await this.workspaceDataSourceService.executeRawQuery( const updateResult = await this.workspaceDataSourceService.executeRawQuery(
rawQuery, rawQuery,
parameters, parameters,

View File

@ -8,6 +8,7 @@ export const fetchTableColumns = async (
): Promise<PostgresTableSchemaColumn[]> => { ): Promise<PostgresTableSchemaColumn[]> => {
const schemaName = workspaceDataSourceService.getSchemaName(workspaceId); const schemaName = workspaceDataSourceService.getSchemaName(workspaceId);
// TODO: executeRawQuery is deprecated and will throw
const res = await workspaceDataSourceService.executeRawQuery( const res = await workspaceDataSourceService.executeRawQuery(
`SELECT column_name, data_type, udt_name `SELECT column_name, data_type, udt_name
FROM information_schema.columns FROM information_schema.columns

View File

@ -1,11 +1,11 @@
import { singular } from 'pluralize'; import { singular } from 'pluralize';
import { DataSource } from 'typeorm'; import { DataSource } from 'typeorm';
import { camelCase } from 'src/utils/camel-case';
import { import {
RemoteTableException, RemoteTableException,
RemoteTableExceptionCode, RemoteTableExceptionCode,
} from 'src/engine/metadata-modules/remote-server/remote-table/remote-table.exception'; } from 'src/engine/metadata-modules/remote-server/remote-table/remote-table.exception';
import { camelCase } from 'src/utils/camel-case';
const MAX_SUFFIX = 10; const MAX_SUFFIX = 10;
@ -19,6 +19,7 @@ const isNameAvailable = async (
workspaceSchemaName: string, workspaceSchemaName: string,
workspaceDataSource: DataSource, workspaceDataSource: DataSource,
) => { ) => {
// TO DO workspaceDataSource.query method is not allowed, this will throw
const numberOfTablesWithSameName = +( const numberOfTablesWithSameName = +(
await workspaceDataSource.query( await workspaceDataSource.query(
`SELECT count(table_name) FROM information_schema.tables WHERE table_name LIKE '${tableName}' AND table_schema IN ('core', 'metadata', '${workspaceSchemaName}')`, `SELECT count(table_name) FROM information_schema.tables WHERE table_name LIKE '${tableName}' AND table_schema IN ('core', 'metadata', '${workspaceSchemaName}')`,

View File

@ -1,9 +1,7 @@
import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository';
import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository'; import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository';
import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository';
export const metadataToRepositoryMapping = { export const metadataToRepositoryMapping = {
BlocklistWorkspaceEntity: BlocklistRepository, BlocklistWorkspaceEntity: BlocklistRepository,
TimelineActivityWorkspaceEntity: TimelineActivityRepository, TimelineActivityWorkspaceEntity: TimelineActivityRepository,
WorkspaceMemberWorkspaceEntity: WorkspaceMemberRepository,
}; };

View File

@ -3,8 +3,9 @@ import { DynamicModule, Global, Module, Provider } from '@nestjs/common';
import { capitalize } from 'twenty-shared/utils'; import { capitalize } from 'twenty-shared/utils';
import { metadataToRepositoryMapping } from 'src/engine/object-metadata-repository/metadata-to-repository.mapping'; import { metadataToRepositoryMapping } from 'src/engine/object-metadata-repository/metadata-to-repository.mapping';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
import { convertClassNameToObjectMetadataName } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/convert-class-to-object-metadata-name.util'; import { convertClassNameToObjectMetadataName } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/convert-class-to-object-metadata-name.util';
@Global() @Global()
@ -25,18 +26,16 @@ export class ObjectMetadataRepositoryModule {
provide: `${capitalize( provide: `${capitalize(
convertClassNameToObjectMetadataName(objectMetadata.name), convertClassNameToObjectMetadataName(objectMetadata.name),
)}Repository`, )}Repository`,
useFactory: ( useFactory: (twentyORMGlobalManager: TwentyORMGlobalManager) => {
workspaceDataSourceService: WorkspaceDataSourceService, return new repositoryClass(twentyORMGlobalManager);
) => {
return new repositoryClass(workspaceDataSourceService);
}, },
inject: [WorkspaceDataSourceService], inject: [TwentyORMGlobalManager],
}; };
}); });
return { return {
module: ObjectMetadataRepositoryModule, module: ObjectMetadataRepositoryModule,
imports: [WorkspaceDataSourceModule], imports: [WorkspaceDataSourceModule, TwentyORMModule],
providers: [...providers], providers: [...providers],
exports: providers, exports: providers,
}; };

View File

@ -11,6 +11,10 @@ import {
import { FeatureFlagMap } from 'src/engine/core-modules/feature-flag/interfaces/feature-flag-map.interface'; import { FeatureFlagMap } from 'src/engine/core-modules/feature-flag/interfaces/feature-flag-map.interface';
import { WorkspaceInternalContext } from 'src/engine/twenty-orm/interfaces/workspace-internal-context.interface'; import { WorkspaceInternalContext } from 'src/engine/twenty-orm/interfaces/workspace-internal-context.interface';
import {
PermissionsException,
PermissionsExceptionCode,
} from 'src/engine/metadata-modules/permissions/permissions.exception';
import { WorkspaceEntityManager } from 'src/engine/twenty-orm/entity-manager/workspace-entity-manager'; import { WorkspaceEntityManager } from 'src/engine/twenty-orm/entity-manager/workspace-entity-manager';
import { WorkspaceQueryRunner } from 'src/engine/twenty-orm/query-runner/workspace-query-runner'; import { WorkspaceQueryRunner } from 'src/engine/twenty-orm/query-runner/workspace-query-runner';
import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository';
@ -79,6 +83,26 @@ export class WorkspaceDataSource extends DataSource {
return queryRunner as any as WorkspaceQueryRunner; return queryRunner as any as WorkspaceQueryRunner;
} }
// eslint-disable-next-line @typescript-eslint/no-explicit-any
override query<T = any>(
query: string,
// eslint-disable-next-line @typescript-eslint/no-explicit-any
parameters?: any[],
queryRunner?: QueryRunner,
options?: {
shouldBypassPermissionChecks?: boolean;
},
): Promise<T> {
if (!options?.shouldBypassPermissionChecks) {
throw new PermissionsException(
'Method not allowed because permissions are not implemented at datasource level.',
PermissionsExceptionCode.METHOD_NOT_ALLOWED,
);
}
return super.query(query, parameters, queryRunner);
}
setRolesPermissionsVersion(rolesPermissionsVersion: string) { setRolesPermissionsVersion(rolesPermissionsVersion: string) {
this.rolesPermissionsVersion = rolesPermissionsVersion; this.rolesPermissionsVersion = rolesPermissionsVersion;
} }

View File

@ -406,15 +406,6 @@ export class WorkspaceEntityManager extends EntityManager {
return this.connection.getMetadata(entity.constructor).name; return this.connection.getMetadata(entity.constructor).name;
} }
// Forbidden methods
// eslint-disable-next-line @typescript-eslint/no-explicit-any
override query<T = any>(_query: string, _parameters?: any[]): Promise<T> {
throw new Error('Method not allowed.');
}
// Not in use methods - duplicated from TypeORM's EntityManager to use our createQueryBuilder
override find<Entity extends ObjectLiteral>( override find<Entity extends ObjectLiteral>(
entityClass: EntityTarget<Entity>, entityClass: EntityTarget<Entity>,
options?: FindManyOptions<Entity>, options?: FindManyOptions<Entity>,
@ -1098,4 +1089,14 @@ export class WorkspaceEntityManager extends EntityManager {
return super.decrement(target, criteria, propertyPath, value); return super.decrement(target, criteria, propertyPath, value);
} }
// Forbidden methods
// eslint-disable-next-line @typescript-eslint/no-explicit-any
override query<T = any>(_query: string, _parameters?: any[]): Promise<T> {
throw new PermissionsException(
'Method not allowed.',
PermissionsExceptionCode.RAW_SQL_NOT_ALLOWED,
);
}
} }

View File

@ -24,6 +24,10 @@ import { FeatureFlagMap } from 'src/engine/core-modules/feature-flag/interfaces/
import { WorkspaceInternalContext } from 'src/engine/twenty-orm/interfaces/workspace-internal-context.interface'; import { WorkspaceInternalContext } from 'src/engine/twenty-orm/interfaces/workspace-internal-context.interface';
import { FeatureFlagKey } from 'src/engine/core-modules/feature-flag/enums/feature-flag-key.enum'; import { FeatureFlagKey } from 'src/engine/core-modules/feature-flag/enums/feature-flag-key.enum';
import {
PermissionsException,
PermissionsExceptionCode,
} from 'src/engine/metadata-modules/permissions/permissions.exception';
import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps'; import { ObjectMetadataItemWithFieldMaps } from 'src/engine/metadata-modules/types/object-metadata-item-with-field-maps';
import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util'; import { getObjectMetadataMapItemByNameSingular } from 'src/engine/metadata-modules/utils/get-object-metadata-map-item-by-name-singular.util';
import { WorkspaceEntityManager } from 'src/engine/twenty-orm/entity-manager/workspace-entity-manager'; import { WorkspaceEntityManager } from 'src/engine/twenty-orm/entity-manager/workspace-entity-manager';
@ -890,7 +894,10 @@ export class WorkspaceRepository<
* DEPRECATED AND RESTRICTED METHODS * DEPRECATED AND RESTRICTED METHODS
*/ */
override async query(): Promise<unknown> { override async query(): Promise<unknown> {
throw new Error('Method not allowed.'); throw new PermissionsException(
'Method not allowed.',
PermissionsExceptionCode.RAW_SQL_NOT_ALLOWED,
);
} }
override async findByIds(): Promise<T[]> { override async findByIds(): Promise<T[]> {

View File

@ -4,6 +4,10 @@ import { DataSource, EntityManager } from 'typeorm';
import { TypeORMService } from 'src/database/typeorm/typeorm.service'; import { TypeORMService } from 'src/database/typeorm/typeorm.service';
import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service';
import {
PermissionsException,
PermissionsExceptionCode,
} from 'src/engine/metadata-modules/permissions/permissions.exception';
@Injectable() @Injectable()
export class WorkspaceDataSourceService { export class WorkspaceDataSourceService {
@ -99,24 +103,16 @@ export class WorkspaceDataSourceService {
} }
public async executeRawQuery( public async executeRawQuery(
query: string, _query: string,
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
parameters: any[] = [], _parameters: any[] = [],
workspaceId: string, _workspaceId: string,
transactionManager?: EntityManager, _transactionManager?: EntityManager,
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
): Promise<any> { ): Promise<any> {
try { throw new PermissionsException(
if (transactionManager) { 'Method not allowed as permissions are not handled at datasource level.',
return await transactionManager.query(query, parameters); PermissionsExceptionCode.METHOD_NOT_ALLOWED,
} );
const dataSource = await this.connectToMainDataSource();
return await dataSource.query(query, parameters);
} catch (error) {
throw new Error(
`Error executing raw query for workspace ${workspaceId}: ${error.message}`,
);
}
} }
} }

View File

@ -1,130 +0,0 @@
import { Logger } from '@nestjs/common';
import { InjectDataSource } from '@nestjs/typeorm';
import { Command, CommandRunner, Option } from 'nest-commander';
import { DataSource } from 'typeorm';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
interface RunCommandOptions {
workspaceId?: string;
}
@Command({
name: 'workspace:convert-record-positions-to-integers',
description: 'Convert record positions to integers',
})
export class ConvertRecordPositionsToIntegers extends CommandRunner {
private readonly logger = new Logger(ConvertRecordPositionsToIntegers.name);
constructor(
@InjectDataSource('metadata')
private readonly metadataDataSource: DataSource,
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
) {
super();
}
async run(_passedParam: string[], options: RunCommandOptions): Promise<void> {
const queryRunner = this.metadataDataSource.createQueryRunner();
const workspaceId = options.workspaceId;
if (!workspaceId || typeof workspaceId !== 'string') {
this.logger.error('Workspace id is required');
return;
}
const customObjectMetadataCollection = await this.metadataDataSource
.getRepository(ObjectMetadataEntity)
.findBy({
workspaceId,
isCustom: true,
});
const customObjectTableNames = customObjectMetadataCollection.map(
(metadata) => metadata.nameSingular,
);
await queryRunner.connect();
await queryRunner.startTransaction();
const transactionManager = queryRunner.manager;
this.logger.log('Converting record positions to integers');
try {
await this.convertRecordPositionsToIntegers(
customObjectTableNames,
workspaceId,
transactionManager,
);
await queryRunner.commitTransaction();
} catch (error) {
await queryRunner.rollbackTransaction();
this.logger.error('Error converting record positions to integers', error);
} finally {
await queryRunner.release();
this.logger.log('Record positions converted to integers');
}
}
private async convertRecordPositionsToIntegers(
customObjectTableNames: string[],
workspaceId: string,
// eslint-disable-next-line @typescript-eslint/no-explicit-any
transactionManager: any,
): Promise<void> {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
for (const tableName of ['company', 'person', 'opportunity']) {
await this.convertRecordPositionsToIntegersForTable(
tableName,
dataSourceSchema,
workspaceId,
transactionManager,
);
}
for (const tableName of customObjectTableNames) {
await this.convertRecordPositionsToIntegersForTable(
`_${tableName}`,
dataSourceSchema,
workspaceId,
transactionManager,
);
}
}
private async convertRecordPositionsToIntegersForTable(
tableName: string,
dataSourceSchema: string,
workspaceId: string,
// eslint-disable-next-line @typescript-eslint/no-explicit-any
transactionManager: any,
): Promise<void> {
await this.workspaceDataSourceService.executeRawQuery(
`UPDATE ${dataSourceSchema}."${tableName}" SET position = subquery.position
FROM (
SELECT id, ROW_NUMBER() OVER (ORDER BY position) as position
FROM ${dataSourceSchema}."${tableName}"
) as subquery
WHERE ${dataSourceSchema}."${tableName}".id = subquery.id`,
[],
workspaceId,
transactionManager,
);
}
@Option({
flags: '-w, --workspace-id [workspace_id]',
description: 'workspace id',
required: true,
})
parseWorkspaceId(value: string): string {
return value;
}
}

View File

@ -7,9 +7,8 @@ import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.mod
import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
import { WorkspaceHealthModule } from 'src/engine/workspace-manager/workspace-health/workspace-health.module'; import { WorkspaceHealthModule } from 'src/engine/workspace-manager/workspace-health/workspace-health.module';
import { ConvertRecordPositionsToIntegers } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/convert-record-positions-to-integers.command';
import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module';
import { SyncWorkspaceLoggerModule } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/services/sync-workspace-logger.module'; import { SyncWorkspaceLoggerModule } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/services/sync-workspace-logger.module';
import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module';
import { SyncWorkspaceMetadataCommand } from './sync-workspace-metadata.command'; import { SyncWorkspaceMetadataCommand } from './sync-workspace-metadata.command';
@ -24,7 +23,7 @@ import { SyncWorkspaceMetadataCommand } from './sync-workspace-metadata.command'
TypeOrmModule.forFeature([Workspace], 'core'), TypeOrmModule.forFeature([Workspace], 'core'),
SyncWorkspaceLoggerModule, SyncWorkspaceLoggerModule,
], ],
providers: [SyncWorkspaceMetadataCommand, ConvertRecordPositionsToIntegers], providers: [SyncWorkspaceMetadataCommand],
exports: [SyncWorkspaceMetadataCommand], exports: [SyncWorkspaceMetadataCommand],
}) })
export class WorkspaceSyncMetadataCommandsModule {} export class WorkspaceSyncMetadataCommandsModule {}

View File

@ -1,16 +1,14 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module';
import { BlocklistValidationService } from 'src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service'; import { BlocklistValidationService } from 'src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service';
import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [ imports: [
ObjectMetadataRepositoryModule.forFeature([ ObjectMetadataRepositoryModule.forFeature([BlocklistWorkspaceEntity]),
BlocklistWorkspaceEntity, TwentyORMModule,
WorkspaceMemberWorkspaceEntity,
]),
], ],
providers: [BlocklistValidationService], providers: [BlocklistValidationService],
exports: [BlocklistValidationService], exports: [BlocklistValidationService],

View File

@ -8,10 +8,10 @@ import {
} from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface';
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { isDomain } from 'src/engine/utils/is-domain'; import { isDomain } from 'src/engine/utils/is-domain';
import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository';
import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity';
import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository';
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';
export type BlocklistItem = Omit< export type BlocklistItem = Omit<
@ -28,8 +28,7 @@ export class BlocklistValidationService {
constructor( constructor(
@InjectObjectMetadataRepository(BlocklistWorkspaceEntity) @InjectObjectMetadataRepository(BlocklistWorkspaceEntity)
private readonly blocklistRepository: BlocklistRepository, private readonly blocklistRepository: BlocklistRepository,
@InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
private readonly workspaceMemberRepository: WorkspaceMemberRepository,
) {} ) {}
public async validateBlocklistForCreateMany( public async validateBlocklistForCreateMany(
@ -84,8 +83,15 @@ export class BlocklistValidationService {
userId: string, userId: string,
workspaceId: string, workspaceId: string,
) { ) {
const workspaceMemberRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
WorkspaceMemberWorkspaceEntity,
);
const currentWorkspaceMember = const currentWorkspaceMember =
await this.workspaceMemberRepository.getByIdOrFail(userId, workspaceId); await workspaceMemberRepository.findOneByOrFail({
userId,
});
const currentBlocklist = const currentBlocklist =
await this.blocklistRepository.getByWorkspaceMemberId( await this.blocklistRepository.getByWorkspaceMemberId(
@ -126,8 +132,16 @@ export class BlocklistValidationService {
return; return;
} }
const workspaceMemberRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
WorkspaceMemberWorkspaceEntity,
);
const currentWorkspaceMember = const currentWorkspaceMember =
await this.workspaceMemberRepository.getByIdOrFail(userId, workspaceId); await workspaceMemberRepository.findOneByOrFail({
userId,
});
const currentBlocklist = const currentBlocklist =
await this.blocklistRepository.getByWorkspaceMemberId( await this.blocklistRepository.getByWorkspaceMemberId(

View File

@ -1,46 +1,49 @@
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity';
@Injectable() @Injectable()
export class BlocklistRepository { export class BlocklistRepository {
constructor( constructor(
private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
) {} ) {}
public async getById( public async getById(
id: string, id: string,
workspaceId: string, workspaceId: string,
): Promise<BlocklistWorkspaceEntity | null> { ): Promise<BlocklistWorkspaceEntity | null> {
const dataSourceSchema = const blockListRepository =
this.workspaceDataSourceService.getSchemaName(workspaceId); await this.twentyORMGlobalManager.getRepositoryForWorkspace(
const blocklistItems =
await this.workspaceDataSourceService.executeRawQuery(
`SELECT * FROM ${dataSourceSchema}."blocklist" WHERE "id" = $1`,
[id],
workspaceId, workspaceId,
BlocklistWorkspaceEntity,
{
shouldBypassPermissionChecks: true,
},
); );
if (!blocklistItems || blocklistItems.length === 0) { return blockListRepository.findOneBy({
return null; id,
} });
return blocklistItems[0];
} }
public async getByWorkspaceMemberId( public async getByWorkspaceMemberId(
workspaceMemberId: string, workspaceMemberId: string,
workspaceId: string, workspaceId: string,
): Promise<BlocklistWorkspaceEntity[]> { ): Promise<BlocklistWorkspaceEntity[]> {
const dataSourceSchema = const blockListRepository =
this.workspaceDataSourceService.getSchemaName(workspaceId); await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
BlocklistWorkspaceEntity,
{
shouldBypassPermissionChecks: true,
},
);
return await this.workspaceDataSourceService.executeRawQuery( return blockListRepository.find({
`SELECT * FROM ${dataSourceSchema}."blocklist" WHERE "workspaceMemberId" = $1`, where: {
[workspaceMemberId], workspaceMemberId,
workspaceId, },
); });
} }
} }

View File

@ -31,14 +31,10 @@ import { CalendarCommonModule } from 'src/modules/calendar/common/calendar-commo
import { CalendarChannelSyncStatusService } from 'src/modules/calendar/common/services/calendar-channel-sync-status.service'; import { CalendarChannelSyncStatusService } from 'src/modules/calendar/common/services/calendar-channel-sync-status.service';
import { ConnectedAccountModule } from 'src/modules/connected-account/connected-account.module'; import { ConnectedAccountModule } from 'src/modules/connected-account/connected-account.module';
import { RefreshTokensManagerModule } from 'src/modules/connected-account/refresh-tokens-manager/connected-account-refresh-tokens-manager.module'; import { RefreshTokensManagerModule } from 'src/modules/connected-account/refresh-tokens-manager/connected-account-refresh-tokens-manager.module';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [ imports: [
ObjectMetadataRepositoryModule.forFeature([ ObjectMetadataRepositoryModule.forFeature([BlocklistWorkspaceEntity]),
BlocklistWorkspaceEntity,
WorkspaceMemberWorkspaceEntity,
]),
CalendarEventParticipantManagerModule, CalendarEventParticipantManagerModule,
TypeOrmModule.forFeature([FeatureFlag, Workspace], 'core'), TypeOrmModule.forFeature([FeatureFlag, Workspace], 'core'),
TypeOrmModule.forFeature([DataSourceEntity], 'metadata'), TypeOrmModule.forFeature([DataSourceEntity], 'metadata'),

View File

@ -1,15 +1,10 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { CalendarEventFindManyPostQueryHook } from 'src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-many.post-query.hook'; import { CalendarEventFindManyPostQueryHook } from 'src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-many.post-query.hook';
import { CalendarEventFindOnePostQueryHook } from 'src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-one.post-query.hook'; import { CalendarEventFindOnePostQueryHook } from 'src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-one.post-query.hook';
import { ApplyCalendarEventsVisibilityRestrictionsService } from 'src/modules/calendar/common/query-hooks/calendar-event/services/apply-calendar-events-visibility-restrictions.service'; import { ApplyCalendarEventsVisibilityRestrictionsService } from 'src/modules/calendar/common/query-hooks/calendar-event/services/apply-calendar-events-visibility-restrictions.service';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [
ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]),
],
providers: [ providers: [
ApplyCalendarEventsVisibilityRestrictionsService, ApplyCalendarEventsVisibilityRestrictionsService,
CalendarEventFindOnePostQueryHook, CalendarEventFindOnePostQueryHook,

View File

@ -4,18 +4,15 @@ import { TypeOrmModule } from '@nestjs/typeorm';
import { FeatureFlag } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; import { FeatureFlag } from 'src/engine/core-modules/feature-flag/feature-flag.entity';
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module';
import { AutoCompaniesAndContactsCreationCalendarChannelListener } from 'src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-calendar-channel.listener'; import { AutoCompaniesAndContactsCreationCalendarChannelListener } from 'src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-calendar-channel.listener';
import { AutoCompaniesAndContactsCreationMessageChannelListener } from 'src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-message-channel.listener'; import { AutoCompaniesAndContactsCreationMessageChannelListener } from 'src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-message-channel.listener';
import { CreateCompanyAndContactService } from 'src/modules/contact-creation-manager/services/create-company-and-contact.service'; import { CreateCompanyAndContactService } from 'src/modules/contact-creation-manager/services/create-company-and-contact.service';
import { CreateCompanyService } from 'src/modules/contact-creation-manager/services/create-company.service'; import { CreateCompanyService } from 'src/modules/contact-creation-manager/services/create-company.service';
import { CreateContactService } from 'src/modules/contact-creation-manager/services/create-contact.service'; import { CreateContactService } from 'src/modules/contact-creation-manager/services/create-contact.service';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [ imports: [
ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]),
WorkspaceDataSourceModule, WorkspaceDataSourceModule,
TypeOrmModule.forFeature([FeatureFlag], 'core'), TypeOrmModule.forFeature([FeatureFlag], 'core'),
TypeOrmModule.forFeature( TypeOrmModule.forFeature(

View File

@ -3,13 +3,12 @@ import { InjectRepository } from '@nestjs/typeorm';
import chunk from 'lodash.chunk'; import chunk from 'lodash.chunk';
import compact from 'lodash.compact'; import compact from 'lodash.compact';
import { Any, Repository } from 'typeorm'; import { Any, DeepPartial, Repository } from 'typeorm';
import { DatabaseEventAction } from 'src/engine/api/graphql/graphql-query-runner/enums/database-event-action'; import { DatabaseEventAction } from 'src/engine/api/graphql/graphql-query-runner/enums/database-event-action';
import { ExceptionHandlerService } from 'src/engine/core-modules/exception-handler/exception-handler.service'; import { ExceptionHandlerService } from 'src/engine/core-modules/exception-handler/exception-handler.service';
import { FieldActorSource } from 'src/engine/metadata-modules/field-metadata/composite-types/actor.composite-type'; import { FieldActorSource } from 'src/engine/metadata-modules/field-metadata/composite-types/actor.composite-type';
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { WorkspaceEventEmitter } from 'src/engine/workspace-event-emitter/workspace-event-emitter'; import { WorkspaceEventEmitter } from 'src/engine/workspace-event-emitter/workspace-event-emitter';
import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids';
@ -22,7 +21,6 @@ import { filterOutSelfAndContactsFromCompanyOrWorkspace } from 'src/modules/cont
import { getDomainNameFromHandle } from 'src/modules/contact-creation-manager/utils/get-domain-name-from-handle.util'; import { getDomainNameFromHandle } from 'src/modules/contact-creation-manager/utils/get-domain-name-from-handle.util';
import { getUniqueContactsAndHandles } from 'src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util'; import { getUniqueContactsAndHandles } from 'src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util';
import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity';
import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository';
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 { isWorkDomain, isWorkEmail } from 'src/utils/is-work-email'; import { isWorkDomain, isWorkEmail } from 'src/utils/is-work-email';
@ -31,8 +29,6 @@ export class CreateCompanyAndContactService {
constructor( constructor(
private readonly createContactService: CreateContactService, private readonly createContactService: CreateContactService,
private readonly createCompaniesService: CreateCompanyService, private readonly createCompaniesService: CreateCompanyService,
@InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity)
private readonly workspaceMemberRepository: WorkspaceMemberRepository,
private readonly workspaceEventEmitter: WorkspaceEventEmitter, private readonly workspaceEventEmitter: WorkspaceEventEmitter,
@InjectRepository(ObjectMetadataEntity, 'metadata') @InjectRepository(ObjectMetadataEntity, 'metadata')
private readonly objectMetadataRepository: Repository<ObjectMetadataEntity>, private readonly objectMetadataRepository: Repository<ObjectMetadataEntity>,
@ -59,8 +55,16 @@ export class CreateCompanyAndContactService {
}, },
); );
const workspaceMembers = const workspaceMemberRepository =
await this.workspaceMemberRepository.getAllByWorkspaceId(workspaceId); await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
WorkspaceMemberWorkspaceEntity,
{
shouldBypassPermissionChecks: true,
},
);
const workspaceMembers = await workspaceMemberRepository.find();
const contactsToCreateFromOtherCompanies = const contactsToCreateFromOtherCompanies =
filterOutSelfAndContactsFromCompanyOrWorkspace( filterOutSelfAndContactsFromCompanyOrWorkspace(

View File

@ -1,15 +1,11 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { ApplyMessagesVisibilityRestrictionsService } from 'src/modules/messaging/common/query-hooks/message/apply-messages-visibility-restrictions.service'; import { ApplyMessagesVisibilityRestrictionsService } from 'src/modules/messaging/common/query-hooks/message/apply-messages-visibility-restrictions.service';
import { MessageFindManyPostQueryHook } from 'src/modules/messaging/common/query-hooks/message/message-find-many.post-query.hook'; import { MessageFindManyPostQueryHook } from 'src/modules/messaging/common/query-hooks/message/message-find-many.post-query.hook';
import { MessageFindOnePostQueryHook } from 'src/modules/messaging/common/query-hooks/message/message-find-one.post-query.hook'; import { MessageFindOnePostQueryHook } from 'src/modules/messaging/common/query-hooks/message/message-find-one.post-query.hook';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [ imports: [],
ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]),
],
providers: [ providers: [
ApplyMessagesVisibilityRestrictionsService, ApplyMessagesVisibilityRestrictionsService,
MessageFindOnePostQueryHook, MessageFindOnePostQueryHook,

View File

@ -1,17 +1,12 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { AuditModule } from 'src/engine/core-modules/audit/audit.module'; import { AuditModule } from 'src/engine/core-modules/audit/audit.module';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module';
import { UpsertTimelineActivityFromInternalEvent } from 'src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job'; import { UpsertTimelineActivityFromInternalEvent } from 'src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job';
import { TimelineActivityModule } from 'src/modules/timeline/timeline-activity.module'; import { TimelineActivityModule } from 'src/modules/timeline/timeline-activity.module';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Module({ @Module({
imports: [ imports: [TimelineActivityModule, AuditModule, TwentyORMModule],
ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]),
TimelineActivityModule,
AuditModule,
],
providers: [UpsertTimelineActivityFromInternalEvent], providers: [UpsertTimelineActivityFromInternalEvent],
}) })
export class TimelineJobModule {} export class TimelineJobModule {}

View File

@ -2,18 +2,16 @@ import { ObjectRecordNonDestructiveEvent } from 'src/engine/core-modules/event-e
import { Process } from 'src/engine/core-modules/message-queue/decorators/process.decorator'; import { Process } from 'src/engine/core-modules/message-queue/decorators/process.decorator';
import { Processor } from 'src/engine/core-modules/message-queue/decorators/processor.decorator'; import { Processor } from 'src/engine/core-modules/message-queue/decorators/processor.decorator';
import { MessageQueue } from 'src/engine/core-modules/message-queue/message-queue.constants'; import { MessageQueue } from 'src/engine/core-modules/message-queue/message-queue.constants';
import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { WorkspaceEventBatch } from 'src/engine/workspace-event-emitter/types/workspace-event.type'; import { WorkspaceEventBatch } from 'src/engine/workspace-event-emitter/types/workspace-event.type';
import { TimelineActivityService } from 'src/modules/timeline/services/timeline-activity.service'; import { TimelineActivityService } from 'src/modules/timeline/services/timeline-activity.service';
import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository';
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';
@Processor(MessageQueue.entityEventsToDbQueue) @Processor(MessageQueue.entityEventsToDbQueue)
export class UpsertTimelineActivityFromInternalEvent { export class UpsertTimelineActivityFromInternalEvent {
constructor( constructor(
@InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity)
private readonly workspaceMemberService: WorkspaceMemberRepository,
private readonly timelineActivityService: TimelineActivityService, private readonly timelineActivityService: TimelineActivityService,
private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
) {} ) {}
@Process(UpsertTimelineActivityFromInternalEvent.name) @Process(UpsertTimelineActivityFromInternalEvent.name)
@ -22,9 +20,18 @@ export class UpsertTimelineActivityFromInternalEvent {
): Promise<void> { ): Promise<void> {
for (const eventData of workspaceEventBatch.events) { for (const eventData of workspaceEventBatch.events) {
if (eventData.userId) { if (eventData.userId) {
const workspaceMember = await this.workspaceMemberService.getByIdOrFail( const workspaceMemberRepository =
eventData.userId, await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceEventBatch.workspaceId, workspaceEventBatch.workspaceId,
WorkspaceMemberWorkspaceEntity,
{
shouldBypassPermissionChecks: true,
},
);
const workspaceMember = await workspaceMemberRepository.findOneByOrFail(
{
userId: eventData.userId,
},
); );
eventData.workspaceMemberId = workspaceMember.id; eventData.workspaceMemberId = workspaceMember.id;

View File

@ -1,14 +1,16 @@
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { MoreThan } from 'typeorm';
import { ObjectRecord } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface'; import { ObjectRecord } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface';
import { objectRecordDiffMerge } from 'src/engine/core-modules/event-emitter/utils/object-record-diff-merge'; import { objectRecordDiffMerge } from 'src/engine/core-modules/event-emitter/utils/object-record-diff-merge';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
@Injectable() @Injectable()
export class TimelineActivityRepository { export class TimelineActivityRepository {
constructor( constructor(
private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
) {} ) {}
async upsertOne( async upsertOne(
@ -22,11 +24,7 @@ export class TimelineActivityRepository {
linkedRecordId?: string, linkedRecordId?: string,
linkedObjectMetadataId?: string, linkedObjectMetadataId?: string,
) { ) {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
const recentTimelineActivity = await this.findRecentTimelineActivity( const recentTimelineActivity = await this.findRecentTimelineActivity(
dataSourceSchema,
name, name,
objectName, objectName,
recordId, recordId,
@ -53,7 +51,6 @@ export class TimelineActivityRepository {
); );
return this.updateTimelineActivity( return this.updateTimelineActivity(
dataSourceSchema,
recentTimelineActivity[0].id, recentTimelineActivity[0].id,
newProps, newProps,
workspaceMemberId, workspaceMemberId,
@ -62,7 +59,6 @@ export class TimelineActivityRepository {
} }
return this.insertTimelineActivity( return this.insertTimelineActivity(
dataSourceSchema,
name, name,
properties, properties,
objectName, objectName,
@ -76,7 +72,6 @@ export class TimelineActivityRepository {
} }
private async findRecentTimelineActivity( private async findRecentTimelineActivity(
dataSourceSchema: string,
name: string, name: string,
objectName: string, objectName: string,
recordId: string, recordId: string,
@ -84,40 +79,59 @@ export class TimelineActivityRepository {
linkedRecordId: string | undefined, linkedRecordId: string | undefined,
workspaceId: string, workspaceId: string,
) { ) {
return this.workspaceDataSourceService.executeRawQuery( const timelineActivityTypeORMRepository =
`SELECT * FROM ${dataSourceSchema}."timelineActivity" await this.twentyORMGlobalManager.getRepositoryForWorkspace(
WHERE "${objectName}Id" = $1 workspaceId,
AND "name" = $2 'timelineActivity',
AND "workspaceMemberId" = $3 {
AND ${ shouldBypassPermissionChecks: true,
linkedRecordId ? `"linkedRecordId" = $4` : `"linkedRecordId" IS NULL` },
} );
AND "createdAt" >= NOW() - interval '10 minutes'`,
linkedRecordId const tenMinutesAgo = new Date(Date.now() - 10 * 60 * 1000);
? [recordId, name, workspaceMemberId, linkedRecordId]
: [recordId, name, workspaceMemberId], const whereConditions: Record<string, unknown> = {
workspaceId, [objectName + 'Id']: recordId,
); name: name,
workspaceMemberId: workspaceMemberId,
createdAt: MoreThan(tenMinutesAgo),
};
if (linkedRecordId) {
whereConditions.linkedRecordId = linkedRecordId;
} else {
whereConditions.linkedRecordId = null;
}
return timelineActivityTypeORMRepository.find({
where: whereConditions,
order: { createdAt: 'DESC' },
take: 1,
});
} }
private async updateTimelineActivity( private async updateTimelineActivity(
dataSourceSchema: string,
id: string, id: string,
properties: Partial<ObjectRecord>, properties: Partial<ObjectRecord>,
workspaceMemberId: string | undefined, workspaceMemberId: string | undefined,
workspaceId: string, workspaceId: string,
) { ) {
return this.workspaceDataSourceService.executeRawQuery( const timelineActivityTypeORMRepository =
`UPDATE ${dataSourceSchema}."timelineActivity" await this.twentyORMGlobalManager.getRepositoryForWorkspace(
SET "properties" = $2, "workspaceMemberId" = $3 workspaceId,
WHERE "id" = $1`, 'timelineActivity',
[id, properties, workspaceMemberId], {
workspaceId, shouldBypassPermissionChecks: true,
); },
);
return timelineActivityTypeORMRepository.update(id, {
properties: properties,
workspaceMemberId: workspaceMemberId,
});
} }
private async insertTimelineActivity( private async insertTimelineActivity(
dataSourceSchema: string,
name: string, name: string,
properties: Partial<ObjectRecord>, properties: Partial<ObjectRecord>,
objectName: string, objectName: string,
@ -128,21 +142,24 @@ export class TimelineActivityRepository {
linkedObjectMetadataId: string | undefined, linkedObjectMetadataId: string | undefined,
workspaceId: string, workspaceId: string,
) { ) {
return this.workspaceDataSourceService.executeRawQuery( const timelineActivityTypeORMRepository =
`INSERT INTO ${dataSourceSchema}."timelineActivity" await this.twentyORMGlobalManager.getRepositoryForWorkspace(
("name", "properties", "workspaceMemberId", "${objectName}Id", "linkedRecordCachedName", "linkedRecordId", "linkedObjectMetadataId") workspaceId,
VALUES ($1, $2, $3, $4, $5, $6, $7)`, 'timelineActivity',
[ {
name, shouldBypassPermissionChecks: true,
properties, },
workspaceMemberId, );
recordId,
linkedRecordCachedName ?? '', return timelineActivityTypeORMRepository.insert({
linkedRecordId, name: name,
linkedObjectMetadataId, properties: properties,
], workspaceMemberId: workspaceMemberId,
workspaceId, [objectName + 'Id']: recordId,
); linkedRecordCachedName: linkedRecordCachedName ?? '',
linkedRecordId: linkedRecordId,
linkedObjectMetadataId: linkedObjectMetadataId,
});
} }
public async insertTimelineActivitiesForObject( public async insertTimelineActivitiesForObject(
@ -161,33 +178,25 @@ export class TimelineActivityRepository {
if (activities.length === 0) { if (activities.length === 0) {
return; return;
} }
const timelineActivityTypeORMRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
'timelineActivity',
{
shouldBypassPermissionChecks: true,
},
);
const dataSourceSchema = return timelineActivityTypeORMRepository.insert(
this.workspaceDataSourceService.getSchemaName(workspaceId); activities.map((activity) => ({
name: activity.name,
return this.workspaceDataSourceService.executeRawQuery( properties: activity.properties,
`INSERT INTO ${dataSourceSchema}."timelineActivity" workspaceMemberId: activity.workspaceMemberId,
("name", "properties", "workspaceMemberId", "${objectName}Id", "linkedRecordCachedName", "linkedRecordId", "linkedObjectMetadataId") [objectName + 'Id']: activity.recordId,
VALUES ${activities linkedRecordCachedName: activity.linkedRecordCachedName ?? '',
.map( linkedRecordId: activity.linkedRecordId,
(_, index) => linkedObjectMetadataId: activity.linkedObjectMetadataId,
`($${index * 7 + 1}, $${index * 7 + 2}, $${index * 7 + 3}, $${ })),
index * 7 + 4
}, $${index * 7 + 5}, $${index * 7 + 6}, $${index * 7 + 7})`,
)
.join(',')}`,
activities
.map((activity) => [
activity.name,
activity.properties,
activity.workspaceMemberId,
activity.recordId,
activity.linkedRecordCachedName ?? '',
activity.linkedRecordId,
activity.linkedObjectMetadataId,
])
.flat(),
workspaceId,
); );
} }
} }

View File

@ -1,9 +1,11 @@
import { Injectable } from '@nestjs/common'; import { Injectable } from '@nestjs/common';
import { isDefined } from 'twenty-shared/utils';
import { ObjectRecordNonDestructiveEvent } from 'src/engine/core-modules/event-emitter/types/object-record-non-destructive-event'; import { ObjectRecordNonDestructiveEvent } from 'src/engine/core-modules/event-emitter/types/object-record-non-destructive-event';
import { ObjectRecordBaseEvent } from 'src/engine/core-modules/event-emitter/types/object-record.base.event'; import { ObjectRecordBaseEvent } from 'src/engine/core-modules/event-emitter/types/object-record.base.event';
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 { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository'; import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository';
import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity';
@ -22,7 +24,7 @@ export class TimelineActivityService {
constructor( constructor(
@InjectObjectMetadataRepository(TimelineActivityWorkspaceEntity) @InjectObjectMetadataRepository(TimelineActivityWorkspaceEntity)
private readonly timelineActivityRepository: TimelineActivityRepository, private readonly timelineActivityRepository: TimelineActivityRepository,
private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
) {} ) {}
private targetObjects: Record<string, string> = { private targetObjects: Record<string, string> = {
@ -110,14 +112,10 @@ export class TimelineActivityService {
workspaceId: string; workspaceId: string;
eventName: string; eventName: string;
}): Promise<TimelineActivity[] | undefined> { }): Promise<TimelineActivity[] | undefined> {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
switch (event.objectMetadata.nameSingular) { switch (event.objectMetadata.nameSingular) {
case 'noteTarget': case 'noteTarget':
return this.computeActivityTargets({ return this.computeActivityTargets({
event, event,
dataSourceSchema,
activityType: 'note', activityType: 'note',
eventName, eventName,
workspaceId, workspaceId,
@ -125,7 +123,6 @@ export class TimelineActivityService {
case 'taskTarget': case 'taskTarget':
return this.computeActivityTargets({ return this.computeActivityTargets({
event, event,
dataSourceSchema,
activityType: 'task', activityType: 'task',
eventName, eventName,
workspaceId, workspaceId,
@ -134,7 +131,6 @@ export class TimelineActivityService {
case 'task': case 'task':
return this.computeActivities({ return this.computeActivities({
event, event,
dataSourceSchema,
activityType: event.objectMetadata.nameSingular, activityType: event.objectMetadata.nameSingular,
eventName, eventName,
workspaceId, workspaceId,
@ -146,100 +142,119 @@ export class TimelineActivityService {
private async computeActivities({ private async computeActivities({
event, event,
dataSourceSchema,
activityType, activityType,
eventName, eventName,
workspaceId, workspaceId,
}: { }: {
event: ObjectRecordBaseEvent; event: ObjectRecordBaseEvent;
dataSourceSchema: string;
activityType: string; activityType: string;
eventName: string; eventName: string;
workspaceId: string; workspaceId: string;
}) { }) {
const activityTargets = const activityTargetRepository =
await this.workspaceDataSourceService.executeRawQuery( await this.twentyORMGlobalManager.getRepositoryForWorkspace(
`SELECT * FROM ${dataSourceSchema}."${this.targetObjects[activityType]}"
WHERE "${activityType}Id" = $1`,
[event.recordId],
workspaceId, workspaceId,
this.targetObjects[activityType],
{
shouldBypassPermissionChecks: true,
},
); );
const activity = await this.workspaceDataSourceService.executeRawQuery( const activityTargets = await activityTargetRepository.find({
`SELECT * FROM ${dataSourceSchema}."${activityType}" where: {
WHERE "id" = $1`, [activityType + 'Id']: event.recordId,
[event.recordId], },
workspaceId, });
);
const activityRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
activityType,
{
shouldBypassPermissionChecks: true,
},
);
const activity = await activityRepository.findOneBy({
id: event.recordId,
});
if (activityTargets.length === 0) return; if (activityTargets.length === 0) return;
if (activity.length === 0) return; if (!isDefined(activity)) return;
return ( return activityTargets
activityTargets .map((activityTarget) => {
const targetColumn: string[] = Object.entries(activityTarget)
.map(([columnName, columnValue]: [string, string]) => {
if (
columnName === activityType + 'Id' ||
!columnName.endsWith('Id')
)
return;
if (columnValue === null) return;
return columnName;
})
.filter((column): column is string => column !== undefined);
if (targetColumn.length === 0) return;
return {
...event,
name: 'linked-' + eventName,
objectName: targetColumn[0].replace(/Id$/, ''),
recordId: activityTarget[targetColumn[0]],
linkedRecordCachedName: activity.title,
linkedRecordId: activity.id,
linkedObjectMetadataId: event.objectMetadata.id,
} satisfies TimelineActivity;
})
.filter(
// @ts-expect-error legacy noImplicitAny // @ts-expect-error legacy noImplicitAny
.map((activityTarget) => { (event): event is TimelineActivity => event !== undefined,
const targetColumn: string[] = Object.entries(activityTarget) ) as TimelineActivity[];
.map(([columnName, columnValue]: [string, string]) => {
if (
columnName === activityType + 'Id' ||
!columnName.endsWith('Id')
)
return;
if (columnValue === null) return;
return columnName;
})
.filter((column): column is string => column !== undefined);
if (targetColumn.length === 0) return;
return {
...event,
name: 'linked-' + eventName,
objectName: targetColumn[0].replace(/Id$/, ''),
recordId: activityTarget[targetColumn[0]],
linkedRecordCachedName: activity[0].title,
linkedRecordId: activity[0].id,
linkedObjectMetadataId: event.objectMetadata.id,
} satisfies TimelineActivity;
})
// @ts-expect-error legacy noImplicitAny
.filter((event): event is TimelineActivity => event !== undefined)
);
} }
private async computeActivityTargets({ private async computeActivityTargets({
event, event,
dataSourceSchema,
activityType, activityType,
eventName, eventName,
workspaceId, workspaceId,
}: { }: {
event: ObjectRecordBaseEvent; event: ObjectRecordBaseEvent;
dataSourceSchema: string; activityType: 'task' | 'note';
activityType: string;
eventName: string; eventName: string;
workspaceId: string; workspaceId: string;
}): Promise<TimelineActivity[] | undefined> { }): Promise<TimelineActivity[] | undefined> {
const activityTarget = const activityTargetRepository =
await this.workspaceDataSourceService.executeRawQuery( await this.twentyORMGlobalManager.getRepositoryForWorkspace(
`SELECT * FROM ${dataSourceSchema}."${this.targetObjects[activityType]}"
WHERE "id" = $1`,
[event.recordId],
workspaceId, workspaceId,
this.targetObjects[activityType],
{
shouldBypassPermissionChecks: true,
},
); );
if (activityTarget.length === 0) return; const activityTarget = await activityTargetRepository.findOneBy({
id: event.recordId,
});
const activity = await this.workspaceDataSourceService.executeRawQuery( if (!isDefined(activityTarget)) return;
`SELECT * FROM ${dataSourceSchema}."${activityType}"
WHERE "id" = $1`,
[activityTarget[0].activityId],
workspaceId,
);
if (activity.length === 0) return; const activityRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace(
workspaceId,
activityType,
{
shouldBypassPermissionChecks: true,
},
);
const activity = await activityRepository.findOneBy({
id: activityTarget.activityId,
});
if (!isDefined(activity)) return;
const activityObjectMetadataId = event.objectMetadata.fields.find( const activityObjectMetadataId = event.objectMetadata.fields.find(
(field) => field.name === activityType, (field) => field.name === activityType,

View File

@ -1,16 +1,16 @@
import { Module } from '@nestjs/common'; import { Module } from '@nestjs/common';
import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module';
import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module';
import { TimelineActivityService } from 'src/modules/timeline/services/timeline-activity.service'; import { TimelineActivityService } from 'src/modules/timeline/services/timeline-activity.service';
import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity';
@Module({ @Module({
imports: [ imports: [
WorkspaceDataSourceModule,
ObjectMetadataRepositoryModule.forFeature([ ObjectMetadataRepositoryModule.forFeature([
TimelineActivityWorkspaceEntity, TimelineActivityWorkspaceEntity,
]), ]),
TwentyORMModule,
], ],
providers: [TimelineActivityService], providers: [TimelineActivityService],
exports: [TimelineActivityService], exports: [TimelineActivityService],

View File

@ -1,64 +0,0 @@
import { Injectable, NotFoundException } from '@nestjs/common';
import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service';
import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity';
@Injectable()
export class WorkspaceMemberRepository {
constructor(
private readonly workspaceDataSourceService: WorkspaceDataSourceService,
) {}
public async find(workspaceMemberId: string, workspaceId: string) {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
const workspaceMembers =
await this.workspaceDataSourceService.executeRawQuery(
`SELECT * FROM ${dataSourceSchema}."workspaceMember" WHERE "id" = $1`,
[workspaceMemberId],
workspaceId,
);
return workspaceMembers?.[0];
}
public async getByIdOrFail(
userId: string,
workspaceId: string,
): Promise<WorkspaceMemberWorkspaceEntity> {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
const workspaceMembers =
await this.workspaceDataSourceService.executeRawQuery(
`SELECT * FROM ${dataSourceSchema}."workspaceMember" WHERE "userId" = $1`,
[userId],
workspaceId,
);
if (!workspaceMembers || workspaceMembers.length === 0) {
throw new NotFoundException(
`No workspace member found for user ${userId} in workspace ${workspaceId}`,
);
}
return workspaceMembers[0];
}
public async getAllByWorkspaceId(
workspaceId: string,
): Promise<WorkspaceMemberWorkspaceEntity[]> {
const dataSourceSchema =
this.workspaceDataSourceService.getSchemaName(workspaceId);
const workspaceMembers =
await this.workspaceDataSourceService.executeRawQuery(
`SELECT * FROM ${dataSourceSchema}."workspaceMember"`,
[],
workspaceId,
);
return workspaceMembers;
}
}