Skip to content

Commit 7b7d958

Browse files
committed
support batching queries with limits > 100
1 parent c22c2da commit 7b7d958

File tree

2 files changed

+70
-4
lines changed

2 files changed

+70
-4
lines changed

lib/model.js

+15-4
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ const kConditionGT = Symbol('>');
3434
const kConditionGTE = Symbol('>=');
3535
const kConditionBetween = Symbol('between');
3636
const kConditionBegins = Symbol('begins');
37+
const kBatchGetItemLimit = 100;
3738

3839
const supportedQueryConditions = new Map([
3940
// dynamodm query condition => [internal identifier, number of arguments required]
@@ -315,7 +316,7 @@ class BaseModel {
315316
let {rawQueryOptions, rawFetchOptions, ...otherOptions} = options ?? {};
316317

317318
// returns an array of models (possibly empty)
318-
const rawQuery = BaseModel.#convertQuery(this, query, Object.assign({startAfter: otherOptions.startAfter, limit: otherOptions.limmit}, rawQueryOptions));
319+
const rawQuery = BaseModel.#convertQuery(this, query, Object.assign({startAfter: otherOptions.startAfter, limit: otherOptions.limit}, rawQueryOptions));
319320
const pending = [];
320321
let errorOccurred = false;
321322
const setErrorOccurred = () => { errorOccurred = true; };
@@ -348,7 +349,7 @@ class BaseModel {
348349
// options are as queryMany, except Ids are returned, so there are no rawFetchOptions
349350
let {rawQueryOptions, ...otherOptions} = options ?? {};
350351
otherOptions = Object.assign({limit: 50}, otherOptions);
351-
const rawQuery = BaseModel.#convertQuery(this, query, Object.assign({startAfter: otherOptions.startAfter, limit: otherOptions.limmit}, rawQueryOptions));
352+
const rawQuery = BaseModel.#convertQuery(this, query, Object.assign({startAfter: otherOptions.startAfter, limit: otherOptions.limit}, rawQueryOptions));
352353
const results = [];
353354
for await (const batch of BaseModel.#rawQueryIdsBatchIterator(this, rawQuery, otherOptions)) {
354355
results.push(batch);
@@ -592,8 +593,6 @@ class BaseModel {
592593
}
593594

594595
// get an array of instances of this schema by id
595-
// At most 100 items can be fetched at one time (the limit to the dynamodb BatchGetItem request size)
596-
// TODO: the 100 item limit should be lifted here by splitting into 100-item batches internally.
597596
static async #getByIds(DerivedModel, ids, rawOptions) {
598597
// only the ConsistentRead option is supported
599598
const { ConsistentRead, abortSignal } = rawOptions ?? {};
@@ -617,6 +616,12 @@ class BaseModel {
617616
let Keys = ids.map(id => ({ [schema.idFieldName]: id }));
618617
const results = new Map();
619618
let retryCount = 0;
619+
let keysExceedingLimit;
620+
// At most kBatchGetItemLimit (100) items can be fetched at one time
621+
// (the limit to the dynamodb BatchGetItem request size), so if
622+
// rawOptions.limit is greater than this, batch the request.
623+
keysExceedingLimit = Keys.slice(kBatchGetItemLimit);
624+
Keys = Keys.slice(0, kBatchGetItemLimit);
620625
while(Keys.length) {
621626
const command = new BatchGetCommand({
622627
RequestItems: {
@@ -639,6 +644,12 @@ class BaseModel {
639644
retryCount += 1;
640645
await delayMs(table[kTableGetBackoffDelayMs](retryCount));
641646
}
647+
// if there's any room after the unprocessed keys from the
648+
// response, request some of the keys that haven't been requested
649+
// yet as well:
650+
const spaceAvailable = kBatchGetItemLimit - Keys.length;
651+
Keys = Keys.concat(keysExceedingLimit.slice(0, spaceAvailable));
652+
keysExceedingLimit = keysExceedingLimit.slice(spaceAvailable);
642653
}
643654
// return the results by mapping the original ids, so that the results are in the same order
644655
return ids.map(

test/queries.js

+55
Original file line numberDiff line numberDiff line change
@@ -681,3 +681,58 @@ t.test('queries:', async t => {
681681

682682
t.end();
683683
});
684+
685+
t.test('largeQueries', async t => {
686+
const table = DynamoDM.Table({ name: 'test-table-largequeries'});
687+
const XSchema = DynamoDM.Schema('x', {
688+
properties: {
689+
id: DynamoDM.DocIdField,
690+
x: {type: 'string'},
691+
n: {type: 'number'},
692+
b: DynamoDM.Binary,
693+
}
694+
}, {index: {
695+
'sortedByN': {
696+
hashKey: 'type',
697+
sortKey: 'n'
698+
},
699+
'sortedByB': {
700+
hashKey: 'x',
701+
sortKey: 'b'
702+
},
703+
} });
704+
705+
const X = table.model(XSchema);
706+
const all_xs = [];
707+
708+
const N = 507;
709+
for (let i = 0; i < N; i++) {
710+
const x = await new X({x:'constant', n: 1000 - i, b: Buffer.from(`b=${i}`), }).save();
711+
all_xs.push(x);
712+
}
713+
714+
t.after(async () => {
715+
await table.deleteTable();
716+
table.destroyConnection();
717+
});
718+
719+
t.test('queryMany', async t => {
720+
t.test('on type index', async t => {
721+
const xs = await X.queryMany({ type: 'x' }, {limit: 1000});
722+
t.equal(xs.length, all_xs.length, 'should return all N of this type');
723+
t.equal(xs[0].constructor, (new X()).constructor, 'should have the correct constructor');
724+
});
725+
726+
t.test('with limit', async t => {
727+
const xs = await X.queryMany({ type: 'x' }, {limit: N-13});
728+
t.equal(xs.length, N-13, 'should return the requested number of items');
729+
});
730+
731+
t.test('sorted result', async t => {
732+
const xs = await X.queryMany({ type: 'x', n: {$gte:0} }, {limit: 1000});
733+
t.equal(xs.length, all_xs.length, 'should return all N of this type');
734+
t.equal(xs.every((x, idx) => (idx === 0 || x.n > xs[idx-1].n)), true, 'should return correctly sorted result');
735+
});
736+
});
737+
738+
});

0 commit comments

Comments
 (0)