Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Aggregated queries #1 #8345

Merged
merged 33 commits into from
Nov 14, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
4b51b58
wip
ijreilly Nov 5, 2024
3f929c9
Refactor code
ijreilly Nov 6, 2024
49bc25a
Add test for isUuid util
ijreilly Nov 6, 2024
10fdfd5
fix selectedFields and add number fields aggregations
ijreilly Nov 6, 2024
6b3e533
minor fixes
ijreilly Nov 6, 2024
3fb0fe0
Change way to filter out duplicates of fieldMetadataMap
ijreilly Nov 8, 2024
7adb8ce
Revert changes to tsconfig files
ijreilly Nov 8, 2024
ad441bd
Merge branch 'main' into aggregate-queries-1
charlesBochet Nov 8, 2024
fe53fb8
Merge branch 'main' into aggregate-queries-1
charlesBochet Nov 11, 2024
cd4465f
Fix
charlesBochet Nov 11, 2024
d26a1bd
Fix
charlesBochet Nov 11, 2024
870fc61
Fix ci
charlesBochet Nov 11, 2024
dc72de9
Try ci
charlesBochet Nov 11, 2024
d3dbf46
Fix tests
charlesBochet Nov 11, 2024
52e8a88
Fix
charlesBochet Nov 11, 2024
55ce64d
Fix
charlesBochet Nov 11, 2024
f8d4c2a
Fix
charlesBochet Nov 11, 2024
5af68f8
Fix
charlesBochet Nov 11, 2024
db02867
Fix
charlesBochet Nov 11, 2024
2c6549a
Add feature flag
charlesBochet Nov 11, 2024
b347f39
Refacto maps
charlesBochet Nov 11, 2024
6ca4628
Fix
charlesBochet Nov 11, 2024
7c7ee44
Fix
charlesBochet Nov 11, 2024
22d0fb9
Fix
charlesBochet Nov 11, 2024
b182252
Fix
charlesBochet Nov 11, 2024
2d7ff9c
Fix
charlesBochet Nov 11, 2024
fce5c81
Fixes
charlesBochet Nov 11, 2024
c1df0e5
Remove unused date time scalar
charlesBochet Nov 11, 2024
765d9a6
Fix tests
charlesBochet Nov 11, 2024
e5a9e3d
Merge branch 'main' into aggregate-queries-1
Weiko Nov 12, 2024
b44ceb9
Add aggregation to nested queries + totalCount + fix aggregate for pa…
Weiko Nov 14, 2024
1aee63c
fix
Weiko Nov 14, 2024
853126a
fix tests
Weiko Nov 14, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -33,12 +33,18 @@ export class GraphqlQuerySelectedFieldsParser {
relations: {},
};

const hasEdges = Object.keys(graphqlSelectedFields).includes('edges');
Weiko marked this conversation as resolved.
Show resolved Hide resolved

for (const [fieldKey, fieldValue] of Object.entries(
graphqlSelectedFields,
)) {
if (hasEdges && fieldKey !== 'edges') {
continue;
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

logic: skipping non-edges fields when edges exist could prevent aggregation fields from being processed at the root level

if (this.shouldNotParseField(fieldKey)) {
continue;
}

if (this.isConnectionField(fieldKey, fieldValue)) {
const subResult = this.parse(fieldValue, fieldMetadataMap);

Expand Down Expand Up @@ -83,9 +89,7 @@ export class GraphqlQuerySelectedFieldsParser {
}

private shouldNotParseField(fieldKey: string): boolean {
return ['__typename', 'totalCount', 'pageInfo', 'cursor'].includes(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we replace totalCount everywhere in the app by countUniqueId? We'll need to give a heads up for deprecation but we can start replacing it in our frontend at least. We plan to email people about API Key deprecation so we can let them know about this at the same time.
If I'm not mistaken we're doing a separate query for totalCount now while this could fit in the same query with all other aggregated fields?

fieldKey,
);
return ['__typename', 'cursor'].includes(fieldKey);
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

logic: removing 'totalCount' and 'pageInfo' from shouldNotParseField may cause these fields to be incorrectly included in the select statement


private parseCompositeField(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,12 @@ export class GraphqlQueryParser {
return queryBuilder;
}

public addGetMinCreatedAtToBuilder(
queryBuilder: SelectQueryBuilder<any>,
): SelectQueryBuilder<any> {
return queryBuilder.addSelect('MIN("createdAt") OVER()', 'minCreatedAt');
}

private checkForDeletedAtFilter = (
filter: FindOptionsWhere<ObjectLiteral> | FindOptionsWhere<ObjectLiteral>[],
): boolean => {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import { Injectable } from '@nestjs/common';

import { isDefined } from 'class-validator';
import graphqlFields from 'graphql-fields';

import { ResolverService } from 'src/engine/api/graphql/graphql-query-runner/interfaces/resolver-service.interface';
Expand All @@ -27,8 +26,10 @@ import {
getCursor,
getPaginationInfo,
} from 'src/engine/api/graphql/graphql-query-runner/utils/cursors.util';
import { getAvailableAggregationsFromObjectFields } from 'src/engine/api/graphql/workspace-schema-builder/utils/get-available-aggregations-from-object-fields.util';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { formatResult } from 'src/engine/twenty-orm/utils/format-result.util';
import { isDefined } from 'src/utils/is-defined';

@Injectable()
export class GraphqlQueryFindManyResolverService
Expand Down Expand Up @@ -85,8 +86,6 @@ export class GraphqlQueryFindManyResolverService
);
const isForwardPagination = !isDefined(args.before);

const limit = args.first ?? args.last ?? QUERY_MAX_RECORDS;

const withDeletedCountQueryBuilder =
graphqlQueryParser.applyDeletedAtToBuilder(
withFilterCountQueryBuilder,
Expand All @@ -95,7 +94,7 @@ export class GraphqlQueryFindManyResolverService

const totalCount = isDefined(selectedFields.totalCount)
? await withDeletedCountQueryBuilder.getCount()
: 0;
: 0; // Blocked at 60 it seems

const cursor = getCursor(args);

Expand Down Expand Up @@ -139,12 +138,57 @@ export class GraphqlQueryFindManyResolverService
args.filter ?? ({} as Filter),
);

const allAggregatedFields = getAvailableAggregationsFromObjectFields(
Object.values(objectMetadataMapItem.fields),
);

const selectedAggregatedFields = allAggregatedFields.reduce(
(acc, aggregatedField) => {
const key = Object.keys(aggregatedField)[0];

if (!Object.keys(selectedFields).includes(key)) return acc;
if (acc.some((field) => Object.keys(field)[0] === key)) return acc;

return [...acc, aggregatedField];
},
[] as typeof allAggregatedFields,
);

if (selectedAggregatedFields.length > 0) {
selectedAggregatedFields.forEach((aggregatedField) => {
const [[aggregatedFieldName, aggregatedFieldDetails]] =
Object.entries(aggregatedField);
const operation = aggregatedFieldDetails.aggregationType;
const fieldName = aggregatedFieldDetails.fromField;

withDeletedQueryBuilder.addSelect(
`${operation}("${fieldName}") OVER()`,
`${aggregatedFieldName}`,
);
});
}

const limit = args.first ?? args.last ?? QUERY_MAX_RECORDS;

const nonFormattedObjectRecords = await withDeletedQueryBuilder
.take(limit + 1)
.getMany();
.getRawAndEntities();

const aggregatedFieldsResults = selectedAggregatedFields.reduce(
(acc, aggregatedField) => {
const aggregatedFieldName = Object.keys(aggregatedField)[0];

return {
...acc,
[aggregatedFieldName]:
nonFormattedObjectRecords.raw[0][aggregatedFieldName],
};
},
{},
);

const objectRecords = formatResult(
nonFormattedObjectRecords,
nonFormattedObjectRecords.entities,
objectMetadataMapItem,
objectMetadataMap,
);
Expand Down Expand Up @@ -186,7 +230,7 @@ export class GraphqlQueryFindManyResolverService
hasPreviousPage,
});

return result;
return { ...result, ...aggregatedFieldsResults };
}

async validate<Filter extends RecordFilter>(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,8 @@ export interface WorkspaceQueryRunnerOptions {
authContext: AuthContext;
info: GraphQLResolveInfo;
objectMetadataItem: ObjectMetadataInterface;
fieldMetadataCollection: FieldMetadataInterface[];
objectMetadataCollection: ObjectMetadataInterface[];
fieldMetadataCollection: FieldMetadataInterface[]; // Legacy
objectMetadataCollection: ObjectMetadataInterface[]; // Legacy
objectMetadataMap: ObjectMetadataMap;
objectMetadataMapItem: ObjectMetadataMapItem;
}
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,14 @@ import { GraphQLFieldConfigMap, GraphQLInt, GraphQLObjectType } from 'graphql';
import { WorkspaceBuildSchemaOptions } from 'src/engine/api/graphql/workspace-schema-builder/interfaces/workspace-build-schema-optionts.interface';
import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface';

import { getAvailableAggregationsFromObjectFields } from 'src/engine/api/graphql/workspace-schema-builder/utils/get-available-aggregations-from-object-fields.util';
import { pascalCase } from 'src/utils/pascal-case';

import { ConnectionTypeFactory } from './connection-type.factory';
import {
ObjectTypeDefinition,
ObjectTypeDefinitionKind,
} from './object-type-definition.factory';
import { ConnectionTypeFactory } from './connection-type.factory';

export enum ConnectionTypeDefinitionKind {
Edge = 'Edge',
Expand Down Expand Up @@ -43,7 +44,25 @@ export class ConnectionTypeDefinitionFactory {
objectMetadata: ObjectMetadataInterface,
options: WorkspaceBuildSchemaOptions,
): GraphQLFieldConfigMap<any, any> {
const fields: GraphQLFieldConfigMap<any, any> = {};
const fields: GraphQLFieldConfigMap<any, any> = Object.assign(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

style: Consider using Array.reduce instead of Object.assign + map for better readability and performance

{},
...getAvailableAggregationsFromObjectFields(objectMetadata.fields).map(
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should follow the existing factory pattern here:
this.connectionTypeFactory.create(...) see lines below

(agg) => {
const [
[
key,
{
aggregationType: _aggregationType,
fromField: _fromField,
...rest
},
],
] = Object.entries(agg);

return { [key]: rest };
},
),
);

fields.edges = {
type: this.connectionTypeFactory.create(
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
import { GraphQLISODateTime } from '@nestjs/graphql';

import { GraphQLString } from 'graphql';

import { FieldMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/field-metadata.interface';

import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
import { capitalize } from 'src/utils/capitalize';

enum AGGREGATIONS_TYPES {
min = 'MIN',
max = 'MAX',
avg = 'Avg',
}

type AggregationValue = {
type: typeof GraphQLString;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

not a big fan of typeOf! why not directly putting the right one?

description: string;
fromField: string;
aggregationType: AGGREGATIONS_TYPES;
};

type AggregationField = {
[key: string]: AggregationValue;
};

export const getAvailableAggregationsFromObjectFields = (
fields: FieldMetadataInterface[],
): AggregationField[] => {
return fields.reduce<Array<Record<string, any>>>((acc, field) => {
Weiko marked this conversation as resolved.
Show resolved Hide resolved
if (field.type === FieldMetadataType.DATE_TIME) {
return [
...acc,
{
[`min${capitalize(field.name)}`]: {
type: GraphQLISODateTime,
Weiko marked this conversation as resolved.
Show resolved Hide resolved
description: `Oldest date contained in the field ${field.name}`,
fromField: field.name,
aggregationType: AGGREGATIONS_TYPES.min,
},
},
{
[`max${capitalize(field.name)}`]: {
type: GraphQLISODateTime,
description: `Most recent date contained in the field ${field.name}`,
fromField: field.name,
aggregationType: AGGREGATIONS_TYPES.max,
},
},
];
}

return acc;
}, []);
};
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,8 @@ export class WorkspaceSchemaFactory {
authContext.workspace.id,
currentCacheVersion,
);

typeDefs = undefined;
let usedScalarNames =
await this.workspaceCacheStorageService.getGraphQLUsedScalarNames(
authContext.workspace.id,
Expand Down
Loading