mirror of
https://github.com/lingble/twenty.git
synced 2025-11-01 21:27:58 +00:00
Fixes https://github.com/twentyhq/twenty/issues/6859 This PR adds all the remaining resolvers for - updateOne/updateMany - createOne/createMany - deleteOne/deleteMany - destroyOne - restoreMany Also - refactored the graphql-query-runner to be able to add other resolvers without too much boilerplate. - add missing events that were not sent anymore as well as webhooks - make resolver injectable so they can inject other services as well - use objectMetadataMap from cache instead of computing it multiple time - various fixes (mutation not correctly parsing JSON, relationHelper fetching data with empty ids set, ...) Next steps: - Wrapping query builder to handle DB events properly - Move webhook emitters to db event listener - Add pagination where it's missing (findDuplicates, nested relations, etc...)
139 lines
5.0 KiB
TypeScript
139 lines
5.0 KiB
TypeScript
import { Injectable } from '@nestjs/common';
|
|
|
|
import { makeExecutableSchema } from '@graphql-tools/schema';
|
|
import { GraphQLSchema, printSchema } from 'graphql';
|
|
import { gql } from 'graphql-tag';
|
|
|
|
import {
|
|
GraphqlQueryRunnerException,
|
|
GraphqlQueryRunnerExceptionCode,
|
|
} from 'src/engine/api/graphql/graphql-query-runner/errors/graphql-query-runner.exception';
|
|
import { ScalarsExplorerService } from 'src/engine/api/graphql/services/scalars-explorer.service';
|
|
import { workspaceResolverBuilderMethodNames } from 'src/engine/api/graphql/workspace-resolver-builder/factories/factories';
|
|
import { WorkspaceResolverFactory } from 'src/engine/api/graphql/workspace-resolver-builder/workspace-resolver.factory';
|
|
import { WorkspaceGraphQLSchemaFactory } from 'src/engine/api/graphql/workspace-schema-builder/workspace-graphql-schema.factory';
|
|
import { AuthContext } from 'src/engine/core-modules/auth/types/auth-context.type';
|
|
import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service';
|
|
import { WorkspaceMetadataCacheService } from 'src/engine/metadata-modules/workspace-metadata-cache/services/workspace-metadata-cache.service';
|
|
import { WorkspaceCacheStorageService } from 'src/engine/workspace-cache-storage/workspace-cache-storage.service';
|
|
@Injectable()
|
|
export class WorkspaceSchemaFactory {
|
|
constructor(
|
|
private readonly dataSourceService: DataSourceService,
|
|
private readonly scalarsExplorerService: ScalarsExplorerService,
|
|
private readonly workspaceGraphQLSchemaFactory: WorkspaceGraphQLSchemaFactory,
|
|
private readonly workspaceResolverFactory: WorkspaceResolverFactory,
|
|
private readonly workspaceCacheStorageService: WorkspaceCacheStorageService,
|
|
private readonly workspaceMetadataCacheService: WorkspaceMetadataCacheService,
|
|
) {}
|
|
|
|
async createGraphQLSchema(authContext: AuthContext): Promise<GraphQLSchema> {
|
|
if (!authContext.workspace?.id) {
|
|
return new GraphQLSchema({});
|
|
}
|
|
|
|
const dataSourcesMetadata =
|
|
await this.dataSourceService.getDataSourcesMetadataFromWorkspaceId(
|
|
authContext.workspace.id,
|
|
);
|
|
|
|
if (!dataSourcesMetadata || dataSourcesMetadata.length === 0) {
|
|
return new GraphQLSchema({});
|
|
}
|
|
|
|
const currentCacheVersion =
|
|
await this.workspaceCacheStorageService.getMetadataVersion(
|
|
authContext.workspace.id,
|
|
);
|
|
|
|
if (currentCacheVersion === undefined) {
|
|
await this.workspaceMetadataCacheService.recomputeMetadataCache({
|
|
workspaceId: authContext.workspace.id,
|
|
});
|
|
throw new GraphqlQueryRunnerException(
|
|
'Metadata cache version not found',
|
|
GraphqlQueryRunnerExceptionCode.METADATA_CACHE_VERSION_NOT_FOUND,
|
|
);
|
|
}
|
|
|
|
const objectMetadataMap =
|
|
await this.workspaceCacheStorageService.getObjectMetadataMap(
|
|
authContext.workspace.id,
|
|
currentCacheVersion,
|
|
);
|
|
|
|
if (!objectMetadataMap) {
|
|
await this.workspaceMetadataCacheService.recomputeMetadataCache({
|
|
workspaceId: authContext.workspace.id,
|
|
});
|
|
throw new GraphqlQueryRunnerException(
|
|
'Object metadata collection not found',
|
|
GraphqlQueryRunnerExceptionCode.METADATA_CACHE_VERSION_NOT_FOUND,
|
|
);
|
|
}
|
|
|
|
const objectMetadataCollection = Object.values(objectMetadataMap).map(
|
|
(objectMetadataItem) => ({
|
|
...objectMetadataItem,
|
|
fields: Object.values(objectMetadataItem.fields),
|
|
}),
|
|
);
|
|
|
|
// Get typeDefs from cache
|
|
let typeDefs = await this.workspaceCacheStorageService.getGraphQLTypeDefs(
|
|
authContext.workspace.id,
|
|
currentCacheVersion,
|
|
);
|
|
let usedScalarNames =
|
|
await this.workspaceCacheStorageService.getGraphQLUsedScalarNames(
|
|
authContext.workspace.id,
|
|
currentCacheVersion,
|
|
);
|
|
|
|
// If typeDefs are not cached, generate them
|
|
if (!typeDefs || !usedScalarNames) {
|
|
const autoGeneratedSchema =
|
|
await this.workspaceGraphQLSchemaFactory.create(
|
|
objectMetadataCollection,
|
|
workspaceResolverBuilderMethodNames,
|
|
);
|
|
|
|
usedScalarNames =
|
|
this.scalarsExplorerService.getUsedScalarNames(autoGeneratedSchema);
|
|
typeDefs = printSchema(autoGeneratedSchema);
|
|
|
|
await this.workspaceCacheStorageService.setGraphQLTypeDefs(
|
|
authContext.workspace.id,
|
|
currentCacheVersion,
|
|
typeDefs,
|
|
);
|
|
await this.workspaceCacheStorageService.setGraphQLUsedScalarNames(
|
|
authContext.workspace.id,
|
|
currentCacheVersion,
|
|
usedScalarNames,
|
|
);
|
|
}
|
|
|
|
const autoGeneratedResolvers = await this.workspaceResolverFactory.create(
|
|
authContext,
|
|
objectMetadataCollection,
|
|
objectMetadataMap,
|
|
workspaceResolverBuilderMethodNames,
|
|
);
|
|
const scalarsResolvers =
|
|
this.scalarsExplorerService.getScalarResolvers(usedScalarNames);
|
|
|
|
const executableSchema = makeExecutableSchema({
|
|
typeDefs: gql`
|
|
${typeDefs}
|
|
`,
|
|
resolvers: {
|
|
...scalarsResolvers,
|
|
...autoGeneratedResolvers,
|
|
},
|
|
});
|
|
|
|
return executableSchema;
|
|
}
|
|
}
|