[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:
@ -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(
|
||||||
|
|||||||
@ -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,
|
||||||
|
},
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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}`, [
|
||||||
|
|||||||
@ -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}`)
|
||||||
|
|||||||
@ -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}`)
|
||||||
|
|||||||
@ -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}`, [
|
||||||
|
|||||||
@ -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}`)
|
||||||
|
|||||||
@ -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}`)
|
||||||
|
|||||||
@ -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}`)
|
||||||
|
|||||||
@ -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}`, [
|
||||||
|
|||||||
@ -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;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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([]);
|
||||||
|
|
||||||
|
|||||||
@ -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[];
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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();
|
||||||
|
|
||||||
|
|||||||
@ -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 {}
|
||||||
|
|||||||
@ -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],
|
||||||
})
|
})
|
||||||
|
|||||||
@ -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';
|
||||||
|
|
||||||
|
|||||||
@ -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');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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]);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@ -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],
|
|
||||||
];
|
|
||||||
};
|
|
||||||
@ -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 {
|
||||||
|
|||||||
@ -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;
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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
|
||||||
|
|||||||
@ -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}')`,
|
||||||
|
|||||||
@ -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,
|
|
||||||
};
|
};
|
||||||
|
|||||||
@ -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,
|
||||||
};
|
};
|
||||||
|
|||||||
@ -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;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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[]> {
|
||||||
|
|||||||
@ -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}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@ -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 {}
|
||||||
|
|||||||
@ -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],
|
||||||
|
|||||||
@ -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(
|
||||||
|
|||||||
@ -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,
|
},
|
||||||
);
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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'),
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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(
|
||||||
|
|||||||
@ -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(
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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 {}
|
||||||
|
|||||||
@ -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;
|
||||||
|
|||||||
@ -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,
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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,
|
||||||
|
|||||||
@ -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],
|
||||||
|
|||||||
@ -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;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user