@ -222,6 +222,38 @@ class Datastore extends EventEmitter {
}
}
}
}
_getCandidates ( query ) {
const indexNames = Object . keys ( this . indexes )
// STEP 1: get candidates list by checking indexes from most to least frequent usecase
// For a basic match
let usableQuery
usableQuery = Object . entries ( query )
. filter ( ( [ k , v ] ) =>
! ! ( typeof v === 'string' || typeof v === 'number' || typeof v === 'boolean' || isDate ( v ) || v === null ) &&
indexNames . includes ( k )
)
. pop ( )
if ( usableQuery ) return this . indexes [ usableQuery [ 0 ] ] . getMatching ( usableQuery [ 1 ] )
// For a $in match
usableQuery = Object . entries ( query )
. filter ( ( [ k , v ] ) =>
! ! ( query [ k ] && Object . prototype . hasOwnProperty . call ( query [ k ] , '$in' ) ) &&
indexNames . includes ( k )
)
. pop ( )
if ( usableQuery ) return this . indexes [ usableQuery [ 0 ] ] . getMatching ( usableQuery [ 1 ] . $in )
// For a comparison match
usableQuery = Object . entries ( query )
. filter ( ( [ k , v ] ) =>
! ! ( query [ k ] && ( Object . prototype . hasOwnProperty . call ( query [ k ] , '$lt' ) || Object . prototype . hasOwnProperty . call ( query [ k ] , '$lte' ) || Object . prototype . hasOwnProperty . call ( query [ k ] , '$gt' ) || Object . prototype . hasOwnProperty . call ( query [ k ] , '$gte' ) ) ) &&
indexNames . includes ( k )
)
. pop ( )
if ( usableQuery ) return this . indexes [ usableQuery [ 0 ] ] . getBetweenBounds ( usableQuery [ 1 ] )
// By default, return all the DB data
return this . getAllData ( )
}
/ * *
/ * *
* Return the list of candidates for a given query
* Return the list of candidates for a given query
* Crude implementation for now , we return the candidates given by the first usable index if any
* Crude implementation for now , we return the candidates given by the first usable index if any
@ -235,87 +267,41 @@ class Datastore extends EventEmitter {
* @ param { Boolean } dontExpireStaleDocs Optional , defaults to false , if true don 't remove stale docs. Useful for the remove function which shouldn' t be impacted by expirations
* @ param { Boolean } dontExpireStaleDocs Optional , defaults to false , if true don 't remove stale docs. Useful for the remove function which shouldn' t be impacted by expirations
* @ param { Function } callback Signature err , candidates
* @ param { Function } callback Signature err , candidates
* /
* /
getCandidates ( query , dontExpireStaleDocs , callback ) {
async getCandidates ( query , dontExpireStaleDocs , callback ) {
const indexNames = Object . keys ( this . indexes )
const validDocs = [ ]
let usableQueryKeys
if ( typeof dontExpireStaleDocs === 'function' ) {
if ( typeof dontExpireStaleDocs === 'function' ) {
callback = dontExpireStaleDocs
callback = dontExpireStaleDocs
dontExpireStaleDocs = false
dontExpireStaleDocs = false
}
}
async . waterfall ( [
try {
// STEP 1: get candidates list by checking indexes from most to least frequent usecase
// STEP 1: get candidates list by checking indexes from most to least frequent usecase
cb => {
const docs = this . _getCandidates ( query )
// For a basic match
usableQueryKeys = [ ]
Object . keys ( query ) . forEach ( k => {
if ( typeof query [ k ] === 'string' || typeof query [ k ] === 'number' || typeof query [ k ] === 'boolean' || isDate ( query [ k ] ) || query [ k ] === null ) {
usableQueryKeys . push ( k )
}
} )
usableQueryKeys = usableQueryKeys . filter ( k => indexNames . includes ( k ) )
if ( usableQueryKeys . length > 0 ) {
return cb ( null , this . indexes [ usableQueryKeys [ 0 ] ] . getMatching ( query [ usableQueryKeys [ 0 ] ] ) )
}
// For a $in match
usableQueryKeys = [ ]
Object . keys ( query ) . forEach ( k => {
if ( query [ k ] && Object . prototype . hasOwnProperty . call ( query [ k ] , '$in' ) ) {
usableQueryKeys . push ( k )
}
} )
usableQueryKeys = usableQueryKeys . filter ( k => indexNames . includes ( k ) )
if ( usableQueryKeys . length > 0 ) {
return cb ( null , this . indexes [ usableQueryKeys [ 0 ] ] . getMatching ( query [ usableQueryKeys [ 0 ] ] . $in ) )
}
// For a comparison match
usableQueryKeys = [ ]
Object . keys ( query ) . forEach ( k => {
if ( query [ k ] && ( Object . prototype . hasOwnProperty . call ( query [ k ] , '$lt' ) || Object . prototype . hasOwnProperty . call ( query [ k ] , '$lte' ) || Object . prototype . hasOwnProperty . call ( query [ k ] , '$gt' ) || Object . prototype . hasOwnProperty . call ( query [ k ] , '$gte' ) ) ) {
usableQueryKeys . push ( k )
}
} )
usableQueryKeys = usableQueryKeys . filter ( k => indexNames . includes ( k ) )
if ( usableQueryKeys . length > 0 ) {
return cb ( null , this . indexes [ usableQueryKeys [ 0 ] ] . getBetweenBounds ( query [ usableQueryKeys [ 0 ] ] ) )
}
// By default, return all the DB data
return cb ( null , this . getAllData ( ) )
} ,
// STEP 2: remove all expired documents
// STEP 2: remove all expired documents
docs => {
if ( ! dontExpireStaleDocs ) {
if ( dontExpireStaleDocs ) return callback ( null , docs )
const expiredDocsIds = [ ]
const expiredDocsIds = [ ]
const validDocs = [ ]
const ttlIndexesFieldNames = Object . keys ( this . ttlIndexes )
const ttlIndexesFieldNames = Object . keys ( this . ttlIndexes )
docs . forEach ( doc => {
docs . forEach ( doc => {
let valid = true
if ( ttlIndexesFieldNames . every ( i => ! ( doc [ i ] !== undefined && isDate ( doc [ i ] ) && Date . now ( ) > doc [ i ] . getTime ( ) + this . ttlIndexes [ i ] * 1000 ) ) ) validDocs . push ( doc )
ttlIndexesFieldNames . forEach ( i => {
if ( doc [ i ] !== undefined && isDate ( doc [ i ] ) && Date . now ( ) > doc [ i ] . getTime ( ) + this . ttlIndexes [ i ] * 1000 ) {
valid = false
}
} )
if ( valid ) validDocs . push ( doc )
else expiredDocsIds . push ( doc . _id )
else expiredDocsIds . push ( doc . _id )
} )
} )
for ( const _id of expiredDocsIds ) {
async . eachSeries ( expiredDocsIds , ( _id , cb ) => {
await new Promise ( ( resolve , reject ) => {
this . _remove ( { _id : _id } , { } , err => {
this . _remove ( { _id : _id } , { } , err => {
if ( err ) return callback ( err )
if ( err ) return reject ( err )
return cb ( )
return resolve ( )
} )
} )
// eslint-disable-next-line node/handle-callback-err
} , err => {
// TODO: handle error
return callback ( null , validDocs )
} )
} )
} ] )
}
} else validDocs . push ( ... docs )
} catch ( error ) {
if ( typeof callback === 'function' ) callback ( error , null )
else throw error
}
if ( typeof callback === 'function' ) callback ( null , validDocs )
else return validDocs
}
}
/ * *
/ * *