@@ -34,6 +34,7 @@ const kConditionGT = Symbol('>');
34
34
const kConditionGTE = Symbol ( '>=' ) ;
35
35
const kConditionBetween = Symbol ( 'between' ) ;
36
36
const kConditionBegins = Symbol ( 'begins' ) ;
37
+ const kBatchGetItemLimit = 100 ;
37
38
38
39
const supportedQueryConditions = new Map ( [
39
40
// dynamodm query condition => [internal identifier, number of arguments required]
@@ -315,7 +316,7 @@ class BaseModel {
315
316
let { rawQueryOptions, rawFetchOptions, ...otherOptions } = options ?? { } ;
316
317
317
318
// 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 ) ) ;
319
320
const pending = [ ] ;
320
321
let errorOccurred = false ;
321
322
const setErrorOccurred = ( ) => { errorOccurred = true ; } ;
@@ -348,7 +349,7 @@ class BaseModel {
348
349
// options are as queryMany, except Ids are returned, so there are no rawFetchOptions
349
350
let { rawQueryOptions, ...otherOptions } = options ?? { } ;
350
351
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 ) ) ;
352
353
const results = [ ] ;
353
354
for await ( const batch of BaseModel . #rawQueryIdsBatchIterator( this , rawQuery , otherOptions ) ) {
354
355
results . push ( batch ) ;
@@ -592,8 +593,6 @@ class BaseModel {
592
593
}
593
594
594
595
// 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.
597
596
static async #getByIds( DerivedModel , ids , rawOptions ) {
598
597
// only the ConsistentRead option is supported
599
598
const { ConsistentRead, abortSignal } = rawOptions ?? { } ;
@@ -617,6 +616,12 @@ class BaseModel {
617
616
let Keys = ids . map ( id => ( { [ schema . idFieldName ] : id } ) ) ;
618
617
const results = new Map ( ) ;
619
618
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 ) ;
620
625
while ( Keys . length ) {
621
626
const command = new BatchGetCommand ( {
622
627
RequestItems : {
@@ -639,6 +644,12 @@ class BaseModel {
639
644
retryCount += 1 ;
640
645
await delayMs ( table [ kTableGetBackoffDelayMs ] ( retryCount ) ) ;
641
646
}
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 ) ;
642
653
}
643
654
// return the results by mapping the original ids, so that the results are in the same order
644
655
return ids . map (
0 commit comments