From 30fe169762b2ed26fcfb1d058dea79994151fb84 Mon Sep 17 00:00:00 2001 From: Weiko Date: Thu, 19 Dec 2024 16:41:04 +0100 Subject: [PATCH] Fix sort with pagination and composite fields (#9150) Fixes https://github.com/twentyhq/twenty/issues/8863 ## Description This PR fixes an issue with cursor-based pagination when dealing with composite fields (like `fullName`). Previously, the pagination direction was incorrectly determined for composite fields because the code wasn't properly handling nested object structures in the `orderBy` parameter. Refactored the code accordingly. --- .../graphql-query-order.parser.ts | 5 +- .../compute-cursor-arg-filter.spec.ts | 141 ++++++++++++++++++ .../utils/compute-cursor-arg-filter.ts | 111 ++++++++++---- .../interfaces/object-record.interface.ts | 4 +- 4 files changed, 228 insertions(+), 33 deletions(-) create mode 100644 packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/__tests__/compute-cursor-arg-filter.spec.ts diff --git a/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/graphql-query-parsers/graphql-query-order/graphql-query-order.parser.ts b/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/graphql-query-parsers/graphql-query-order/graphql-query-order.parser.ts index a16a9c0c149c5..681613c282f62 100644 --- a/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/graphql-query-parsers/graphql-query-order/graphql-query-order.parser.ts +++ b/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/graphql-query-parsers/graphql-query-order/graphql-query-order.parser.ts @@ -48,7 +48,10 @@ export class GraphqlQueryOrderFieldParser { Object.assign(acc, compositeOrder); } else { acc[`"${objectNameSingular}"."${key}"`] = - this.convertOrderByToFindOptionsOrder(value, isForwardPagination); + this.convertOrderByToFindOptionsOrder( + value as OrderByDirection, + isForwardPagination, + ); } }); diff --git a/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/__tests__/compute-cursor-arg-filter.spec.ts b/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/__tests__/compute-cursor-arg-filter.spec.ts new file mode 100644 index 0000000000000..8aaaeb003b504 --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/__tests__/compute-cursor-arg-filter.spec.ts @@ -0,0 +1,141 @@ +import { OrderByDirection } from 'src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface'; + +import { GraphqlQueryRunnerException } from 'src/engine/api/graphql/graphql-query-runner/errors/graphql-query-runner.exception'; +import { computeCursorArgFilter } from 'src/engine/api/graphql/graphql-query-runner/utils/compute-cursor-arg-filter'; +import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; + +describe('computeCursorArgFilter', () => { + const mockFieldMetadataMap = { + name: { + type: FieldMetadataType.TEXT, + id: 'name-id', + name: 'name', + label: 'Name', + objectMetadataId: 'object-id', + }, + age: { + type: FieldMetadataType.NUMBER, + id: 'age-id', + name: 'age', + label: 'Age', + objectMetadataId: 'object-id', + }, + fullName: { + type: FieldMetadataType.FULL_NAME, + id: 'fullname-id', + name: 'fullName', + label: 'Full Name', + objectMetadataId: 'object-id', + }, + }; + + describe('basic cursor filtering', () => { + it('should return empty array when cursor is empty', () => { + const result = computeCursorArgFilter({}, [], mockFieldMetadataMap, true); + + expect(result).toEqual([]); + }); + + it('should compute forward pagination filter for single field', () => { + const cursor = { name: 'John' }; + const orderBy = [{ name: OrderByDirection.AscNullsLast }]; + + const result = computeCursorArgFilter( + cursor, + orderBy, + mockFieldMetadataMap, + true, + ); + + expect(result).toEqual([{ name: { gt: 'John' } }]); + }); + + it('should compute backward pagination filter for single field', () => { + const cursor = { name: 'John' }; + const orderBy = [{ name: OrderByDirection.AscNullsLast }]; + + const result = computeCursorArgFilter( + cursor, + orderBy, + mockFieldMetadataMap, + false, + ); + + expect(result).toEqual([{ name: { lt: 'John' } }]); + }); + }); + + describe('multiple fields cursor filtering', () => { + it('should handle multiple cursor fields with forward pagination', () => { + const cursor = { name: 'John', age: 30 }; + const orderBy = [ + { name: OrderByDirection.AscNullsLast }, + { age: OrderByDirection.DescNullsLast }, + ]; + + const result = computeCursorArgFilter( + cursor, + orderBy, + mockFieldMetadataMap, + true, + ); + + expect(result).toEqual([ + { name: { gt: 'John' } }, + { name: { eq: 'John' }, age: { lt: 30 } }, + ]); + }); + }); + + describe('composite field handling', () => { + it('should handle fullName composite field', () => { + const cursor = { + fullName: { firstName: 'John', lastName: 'Doe' }, + }; + const orderBy = [ + { + fullName: { + firstName: OrderByDirection.AscNullsLast, + lastName: OrderByDirection.AscNullsLast, + }, + }, + ]; + + const result = computeCursorArgFilter( + cursor, + orderBy, + mockFieldMetadataMap, + true, + ); + + expect(result).toEqual([ + { + fullName: { + firstName: { gt: 'John' }, + lastName: { gt: 'Doe' }, + }, + }, + ]); + }); + }); + + describe('error handling', () => { + it('should throw error for invalid field metadata', () => { + const cursor = { invalidField: 'value' }; + const orderBy = [{ invalidField: OrderByDirection.AscNullsLast }]; + + expect(() => + computeCursorArgFilter(cursor, orderBy, mockFieldMetadataMap, true), + ).toThrow(GraphqlQueryRunnerException); + }); + + it('should throw error for missing orderBy entry', () => { + const cursor = { name: 'John' }; + const orderBy = [{ age: OrderByDirection.AscNullsLast }]; + + expect(() => + computeCursorArgFilter(cursor, orderBy, mockFieldMetadataMap, true), + ).toThrow(GraphqlQueryRunnerException); + }); + }); +}); diff --git a/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/compute-cursor-arg-filter.ts b/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/compute-cursor-arg-filter.ts index 02cd804447db6..030c515251f44 100644 --- a/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/compute-cursor-arg-filter.ts +++ b/packages/twenty-server/src/engine/api/graphql/graphql-query-runner/utils/compute-cursor-arg-filter.ts @@ -13,6 +13,42 @@ import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/fi import { isCompositeFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-composite-field-metadata-type.util'; import { FieldMetadataMap } from 'src/engine/metadata-modules/types/field-metadata-map'; +const computeOperator = ( + isAscending: boolean, + isForwardPagination: boolean, + defaultOperator?: string, +): string => { + if (defaultOperator) return defaultOperator; + + return isAscending + ? isForwardPagination + ? 'gt' + : 'lt' + : isForwardPagination + ? 'lt' + : 'gt'; +}; + +const validateAndGetOrderBy = ( + key: string, + orderBy: ObjectRecordOrderBy, +): Record => { + const keyOrderBy = orderBy.find((order) => key in order); + + if (!keyOrderBy) { + throw new GraphqlQueryRunnerException( + 'Invalid cursor', + GraphqlQueryRunnerExceptionCode.INVALID_CURSOR, + ); + } + + return keyOrderBy; +}; + +const isAscendingOrder = (direction: OrderByDirection): boolean => + direction === OrderByDirection.AscNullsFirst || + direction === OrderByDirection.AscNullsLast; + export const computeCursorArgFilter = ( cursor: Record, orderBy: ObjectRecordOrderBy, @@ -40,35 +76,22 @@ export const computeCursorArgFilter = ( cursorKeys[subConditionIndex], cursorValues[subConditionIndex], fieldMetadataMapByName, + orderBy, + isForwardPagination, 'eq', ), }; } - const keyOrderBy = orderBy.find((order) => key in order); - - if (!keyOrderBy) { - throw new GraphqlQueryRunnerException( - 'Invalid cursor', - GraphqlQueryRunnerExceptionCode.INVALID_CURSOR, - ); - } - - const isAscending = - keyOrderBy[key] === OrderByDirection.AscNullsFirst || - keyOrderBy[key] === OrderByDirection.AscNullsLast; - - const operator = isAscending - ? isForwardPagination - ? 'gt' - : 'lt' - : isForwardPagination - ? 'lt' - : 'gt'; - return { ...whereCondition, - ...buildWhereCondition(key, value, fieldMetadataMapByName, operator), + ...buildWhereCondition( + key, + value, + fieldMetadataMapByName, + orderBy, + isForwardPagination, + ), } as ObjectRecordFilter; }); }; @@ -77,7 +100,9 @@ const buildWhereCondition = ( key: string, value: any, fieldMetadataMapByName: FieldMetadataMap, - operator: string, + orderBy: ObjectRecordOrderBy, + isForwardPagination: boolean, + operator?: string, ): Record => { const fieldMetadata = fieldMetadataMapByName[key]; @@ -93,18 +118,30 @@ const buildWhereCondition = ( key, value, fieldMetadata.type, + orderBy, + isForwardPagination, operator, ); } - return { [key]: { [operator]: value } }; + const keyOrderBy = validateAndGetOrderBy(key, orderBy); + const isAscending = isAscendingOrder(keyOrderBy[key]); + const computedOperator = computeOperator( + isAscending, + isForwardPagination, + operator, + ); + + return { [key]: { [computedOperator]: value } }; }; const buildCompositeWhereCondition = ( key: string, value: any, fieldType: FieldMetadataType, - operator: string, + orderBy: ObjectRecordOrderBy, + isForwardPagination: boolean, + operator?: string, ): Record => { const compositeType = compositeTypeDefinitions.get(fieldType); @@ -115,18 +152,30 @@ const buildCompositeWhereCondition = ( ); } + const keyOrderBy = validateAndGetOrderBy(key, orderBy); const result: Record = {}; compositeType.properties.forEach((property) => { if ( - property.type !== FieldMetadataType.RAW_JSON && - value[property.name] !== undefined + property.type === FieldMetadataType.RAW_JSON || + value[property.name] === undefined ) { - result[key] = { - ...result[key], - [property.name]: { [operator]: value[property.name] }, - }; + return; } + + const isAscending = isAscendingOrder(keyOrderBy[key][property.name]); + const computedOperator = computeOperator( + isAscending, + isForwardPagination, + operator, + ); + + result[key] = { + ...result[key], + [property.name]: { + [computedOperator]: value[property.name], + }, + }; }); return result; diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface.ts index a93e752e2267f..5c0395acac09a 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/object-record.interface.ts @@ -18,7 +18,9 @@ export enum OrderByDirection { } export type ObjectRecordOrderBy = Array<{ - [Property in keyof ObjectRecord]?: OrderByDirection; + [Property in keyof ObjectRecord]?: + | OrderByDirection + | Record; }>; export interface ObjectRecordDuplicateCriteria {