From cea989064e5562a5cadaafd24967822fc3cc2f62 Mon Sep 17 00:00:00 2001 From: Oskar Dudycz Date: Sun, 27 Oct 2024 18:39:54 +0100 Subject: [PATCH 1/2] Added deep equals to object comparison in handle method to support correctly mutable entities --- .../src/core/collection/pongoCollection.ts | 5 +- src/packages/pongo/src/core/index.ts | 1 + .../pongo/src/core/utils/deepEquals.ts | 56 +++++++++++++++++++ src/packages/pongo/src/core/utils/index.ts | 1 + .../pongo/src/e2e/postgres.e2e.spec.ts | 37 ++++++++++++ 5 files changed, 98 insertions(+), 2 deletions(-) create mode 100644 src/packages/pongo/src/core/utils/deepEquals.ts create mode 100644 src/packages/pongo/src/core/utils/index.ts diff --git a/src/packages/pongo/src/core/collection/pongoCollection.ts b/src/packages/pongo/src/core/collection/pongoCollection.ts index 4a48bbf..82dee54 100644 --- a/src/packages/pongo/src/core/collection/pongoCollection.ts +++ b/src/packages/pongo/src/core/collection/pongoCollection.ts @@ -14,6 +14,7 @@ import { } from '@event-driven-io/dumbo'; import { v7 as uuid } from 'uuid'; import { + deepEquals, expectedVersionValue, operationResult, type CollectionOperationOptions, @@ -368,9 +369,9 @@ export const pongoCollection = < ); } - const result = await handle(existing as T); + const result = await handle({ ...existing } as T); - if (existing === result) + if (deepEquals(existing as T, result)) return operationResult>( { successful: true, diff --git a/src/packages/pongo/src/core/index.ts b/src/packages/pongo/src/core/index.ts index a5475b6..7009e43 100644 --- a/src/packages/pongo/src/core/index.ts +++ b/src/packages/pongo/src/core/index.ts @@ -6,3 +6,4 @@ export * from './pongoSession'; export * from './pongoTransaction'; export * from './schema'; export * from './typing'; +export * from './utils'; diff --git a/src/packages/pongo/src/core/utils/deepEquals.ts b/src/packages/pongo/src/core/utils/deepEquals.ts new file mode 100644 index 0000000..f0c513e --- /dev/null +++ b/src/packages/pongo/src/core/utils/deepEquals.ts @@ -0,0 +1,56 @@ +export const deepEquals = (left: T, right: T): boolean => { + if (isEquatable(left)) { + return left.equals(right); + } + + if (Array.isArray(left)) { + return ( + Array.isArray(right) && + left.length === right.length && + left.every((val, index) => deepEquals(val, right[index])) + ); + } + + if ( + typeof left !== 'object' || + typeof right !== 'object' || + left === null || + right === null + ) { + return left === right; + } + + if (Array.isArray(right)) return false; + + const keys1 = Object.keys(left); + const keys2 = Object.keys(right); + + if ( + keys1.length !== keys2.length || + !keys1.every((key) => keys2.includes(key)) + ) + return false; + + for (const key in left) { + if (left[key] instanceof Function && right[key] instanceof Function) + continue; + + const isEqual = deepEquals(left[key], right[key]); + if (!isEqual) { + return false; + } + } + + return true; +}; + +export type Equatable = { equals: (right: T) => boolean } & T; + +export const isEquatable = (left: T): left is Equatable => { + return ( + left && + typeof left === 'object' && + 'equals' in left && + typeof left['equals'] === 'function' + ); +}; diff --git a/src/packages/pongo/src/core/utils/index.ts b/src/packages/pongo/src/core/utils/index.ts new file mode 100644 index 0000000..f8de580 --- /dev/null +++ b/src/packages/pongo/src/core/utils/index.ts @@ -0,0 +1 @@ +export * from './deepEquals'; diff --git a/src/packages/pongo/src/e2e/postgres.e2e.spec.ts b/src/packages/pongo/src/e2e/postgres.e2e.spec.ts index 821dbaa..bdff085 100644 --- a/src/packages/pongo/src/e2e/postgres.e2e.spec.ts +++ b/src/packages/pongo/src/e2e/postgres.e2e.spec.ts @@ -1177,6 +1177,43 @@ void describe('MongoDB Compatibility Tests', () => { }); }); + void it('should make the change if the handler returns the existing document changed', async () => { + const pongoCollection = pongoDb.collection('handleCollection'); + + const existingDoc: User = { name: 'John', age: 25 }; + + const pongoInsertResult = await pongoCollection.insertOne(existingDoc); + + const handle = (existing: User | null) => { + if (existing) existing.name = 'New'; + return existing; + }; + + const resultPongo = await pongoCollection.handle( + pongoInsertResult.insertedId!, + handle, + ); + + assert(resultPongo.successful); + assert.deepStrictEqual(resultPongo.document, { + ...existingDoc, + _id: pongoInsertResult.insertedId, + name: 'New', + _version: 2n, + }); + + const pongoDoc = await pongoCollection.findOne({ + _id: pongoInsertResult.insertedId!, + }); + + assert.deepStrictEqual(pongoDoc, { + ...existingDoc, + _id: pongoInsertResult.insertedId, + name: 'New', + _version: 2n, + }); + }); + void describe('No filter', () => { void it('should filter and count without filter specified', async () => { const pongoCollection = pongoDb.collection('nofilter'); From c9190ca91f71acf28f45eeaa9bf16b3f2134310e Mon Sep 17 00:00:00 2001 From: Oskar Dudycz Date: Sun, 27 Oct 2024 18:55:19 +0100 Subject: [PATCH 2/2] Bumped to 0.16.2 --- src/package-lock.json | 6 +++--- src/package.json | 2 +- src/packages/pongo/package.json | 2 +- src/packages/pongo/src/commandLine/shell.ts | 2 +- 4 files changed, 6 insertions(+), 6 deletions(-) diff --git a/src/package-lock.json b/src/package-lock.json index ea29600..10d6a73 100644 --- a/src/package-lock.json +++ b/src/package-lock.json @@ -1,12 +1,12 @@ { "name": "@event-driven-io/pongo-core", - "version": "0.16.1", + "version": "0.16.2", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "@event-driven-io/pongo-core", - "version": "0.16.1", + "version": "0.16.2", "workspaces": [ "packages/dumbo", "packages/pongo" @@ -8674,7 +8674,7 @@ }, "packages/pongo": { "name": "@event-driven-io/pongo", - "version": "0.16.1", + "version": "0.16.2", "bin": { "pongo": "dist/cli.js" }, diff --git a/src/package.json b/src/package.json index 7f4e36e..d1679c0 100644 --- a/src/package.json +++ b/src/package.json @@ -1,6 +1,6 @@ { "name": "@event-driven-io/pongo-core", - "version": "0.16.1", + "version": "0.16.2", "description": "Pongo - Mongo with strong consistency on top of Postgres", "type": "module", "engines": { diff --git a/src/packages/pongo/package.json b/src/packages/pongo/package.json index b33e05f..cb5299b 100644 --- a/src/packages/pongo/package.json +++ b/src/packages/pongo/package.json @@ -1,6 +1,6 @@ { "name": "@event-driven-io/pongo", - "version": "0.16.1", + "version": "0.16.2", "description": "Pongo - Mongo with strong consistency on top of Postgres", "type": "module", "scripts": { diff --git a/src/packages/pongo/src/commandLine/shell.ts b/src/packages/pongo/src/commandLine/shell.ts index 1a0e88d..d08c479 100644 --- a/src/packages/pongo/src/commandLine/shell.ts +++ b/src/packages/pongo/src/commandLine/shell.ts @@ -124,7 +124,7 @@ const startRepl = async (options: { setLogLevel(process.env.DUMBO_LOG_LEVEL ?? options.logging.logLevel); setLogStyle(process.env.DUMBO_LOG_STYLE ?? options.logging.logStyle); - console.log(chalk.green('Starting Pongo Shell (version: 0.16.1)')); + console.log(chalk.green('Starting Pongo Shell (version: 0.16.2)')); if (options.logging.printOptions) { console.log(chalk.green('With Options:'));