From 2daa93b74980674404f4c122a05647d46dff3ffd Mon Sep 17 00:00:00 2001 From: Leo Lamprecht Date: Sun, 12 Jan 2025 22:02:11 +0100 Subject: [PATCH] Ensure correct batch context for queries (#23) --- src/queries.ts | 7 ++++++- tests/queries.test.ts | 45 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 51 insertions(+), 1 deletion(-) diff --git a/src/queries.ts b/src/queries.ts index 1e96aad..f6e01dd 100644 --- a/src/queries.ts +++ b/src/queries.ts @@ -83,6 +83,10 @@ export const getSyntaxProxy = (config?: { // all queries within it think they are running inside a batch transaction, // in order to retrieve their serialized values. if (typeof value === 'function') { + // Temporarily store the original value of `IN_BATCH`, so that we can resume it + // after the nested function has been called. + const ORIGINAL_IN_BATCH = IN_BATCH; + // Since `value()` is synchronous, `IN_BATCH` should not affect any // other queries somewhere else in the app, even if those are run inside // an asynchronous function, so we don't need to use `IN_BATCH_ASYNC`, @@ -117,7 +121,8 @@ export const getSyntaxProxy = (config?: { value = wrapExpressions(value); } - IN_BATCH = false; + // Restore the original value of `IN_BATCH`. + IN_BATCH = ORIGINAL_IN_BATCH; } // If the function call is happening after an existing function call in the diff --git a/tests/queries.test.ts b/tests/queries.test.ts index d085a16..e4bceb2 100644 --- a/tests/queries.test.ts +++ b/tests/queries.test.ts @@ -208,6 +208,51 @@ describe('syntax proxy', () => { ]); }); + test('using nested function as argument in batch', async () => { + // It's important to define the `callback` here, in order to guarantee that the + // queries are executed standalone if no batch context is detected. + const callback = () => undefined; + + const addProxy = getSyntaxProxy({ rootProperty: 'add', callback }); + const getProxy = getSyntaxProxy({ rootProperty: 'get', callback }); + const alterProxy = getSyntaxProxy({ rootProperty: 'alter', callback }); + + const queryList = getBatchProxy(() => [ + addProxy.newUsers.to(() => getProxy.oldUsers()), + // Assert whether function chaining is still possible after a nested function call, + // since the latter is able to manipulate the batch context. + alterProxy + .model('newUsers') + .to({ slug: 'accounts' }), + ]); + + expect(queryList).toEqual([ + { + structure: { + add: { + newUsers: { + to: { + __RONIN_QUERY: { + get: { oldUsers: {} }, + }, + }, + }, + }, + }, + }, + { + structure: { + alter: { + model: 'newUsers', + to: { + slug: 'accounts', + }, + }, + }, + }, + ]); + }); + test('using schema query types', async () => { const callback = () => undefined;