2020-11-05 16:58:23 +00:00
const BaseItem = require ( '../models/BaseItem' ) ;
const BaseModel = require ( '../BaseModel' ) . default ;
const MasterKey = require ( '../models/MasterKey' ) ;
const Resource = require ( '../models/Resource' ) ;
2020-11-08 16:46:48 +00:00
const ResourceService = require ( './ResourceService' ) . default ;
2020-11-05 16:58:23 +00:00
const Logger = require ( '../Logger' ) . default ;
2019-05-22 15:56:07 +01:00
const EventEmitter = require ( 'events' ) ;
2020-11-05 16:58:23 +00:00
const shim = require ( '../shim' ) . default ;
2017-12-14 18:53:08 +00:00
2017-12-14 17:58:10 +00:00
class DecryptionWorker {
constructor ( ) {
2018-03-09 20:59:12 +00:00
this . state _ = 'idle' ;
2017-12-31 15:23:05 +01:00
this . logger _ = new Logger ( ) ;
2017-12-14 18:53:08 +00:00
2019-09-12 22:16:42 +00:00
this . dispatch = ( ) => { } ;
2017-12-14 19:39:13 +00:00
this . scheduleId _ = null ;
2019-05-22 15:56:07 +01:00
this . eventEmitter _ = new EventEmitter ( ) ;
2019-06-07 23:11:08 +01:00
this . kvStore _ = null ;
this . maxDecryptionAttempts _ = 2 ;
2020-02-22 22:25:16 +11:00
this . startCalls _ = [ ] ;
2017-12-14 19:39:13 +00:00
}
setLogger ( l ) {
this . logger _ = l ;
}
logger ( ) {
return this . logger _ ;
2017-12-14 18:53:08 +00:00
}
2019-05-22 15:56:07 +01:00
on ( eventName , callback ) {
return this . eventEmitter _ . on ( eventName , callback ) ;
}
off ( eventName , callback ) {
return this . eventEmitter _ . removeListener ( eventName , callback ) ;
}
2017-12-14 18:53:08 +00:00
static instance ( ) {
2020-02-28 05:25:42 +11:00
if ( DecryptionWorker . instance _ ) return DecryptionWorker . instance _ ;
DecryptionWorker . instance _ = new DecryptionWorker ( ) ;
return DecryptionWorker . instance _ ;
2017-12-14 18:53:08 +00:00
}
2017-12-14 19:39:13 +00:00
setEncryptionService ( v ) {
this . encryptionService _ = v ;
}
2019-06-07 23:11:08 +01:00
setKvStore ( v ) {
this . kvStore _ = v ;
}
2017-12-14 19:39:13 +00:00
encryptionService ( ) {
2018-03-09 20:59:12 +00:00
if ( ! this . encryptionService _ ) throw new Error ( 'DecryptionWorker.encryptionService_ is not set!!' ) ;
2017-12-14 18:53:08 +00:00
return this . encryptionService _ ;
2017-12-14 17:58:10 +00:00
}
2019-06-07 23:11:08 +01:00
kvStore ( ) {
if ( ! this . kvStore _ ) throw new Error ( 'DecryptionWorker.kvStore_ is not set!!' ) ;
return this . kvStore _ ;
}
2017-12-14 19:39:13 +00:00
async scheduleStart ( ) {
if ( this . scheduleId _ ) return ;
2020-10-09 18:35:46 +01:00
this . scheduleId _ = shim . setTimeout ( ( ) => {
2017-12-14 19:39:13 +00:00
this . scheduleId _ = null ;
2017-12-26 11:38:53 +01:00
this . start ( {
2018-10-08 19:11:53 +01:00
masterKeyNotLoadedHandler : 'dispatch' ,
2017-12-26 11:38:53 +01:00
} ) ;
2017-12-14 19:39:13 +00:00
} , 1000 ) ;
}
2019-06-07 23:11:08 +01:00
async decryptionDisabledItems ( ) {
let items = await this . kvStore ( ) . searchByPrefix ( 'decrypt:' ) ;
2020-05-21 09:14:33 +01:00
items = items . filter ( item => item . value > this . maxDecryptionAttempts _ ) ;
items = items . map ( item => {
2019-06-07 23:11:08 +01:00
const s = item . key . split ( ':' ) ;
return {
type _ : Number ( s [ 1 ] ) ,
id : s [ 2 ] ,
} ;
} ) ;
return items ;
}
async clearDisabledItem ( typeId , itemId ) {
2019-09-19 22:51:18 +01:00
await this . kvStore ( ) . deleteValue ( ` decrypt: ${ typeId } : ${ itemId } ` ) ;
2019-06-07 23:11:08 +01:00
}
2020-04-08 18:02:31 +01:00
async clearDisabledItems ( ) {
await this . kvStore ( ) . deleteByPrefix ( 'decrypt:' ) ;
}
2018-06-10 17:43:24 +01:00
dispatchReport ( report ) {
const action = Object . assign ( { } , report ) ;
action . type = 'DECRYPTION_WORKER_SET' ;
this . dispatch ( action ) ;
}
2020-02-22 22:25:16 +11:00
async start _ ( options = null ) {
2017-12-26 11:38:53 +01:00
if ( options === null ) options = { } ;
2018-10-08 19:11:53 +01:00
if ( ! ( 'masterKeyNotLoadedHandler' in options ) ) options . masterKeyNotLoadedHandler = 'throw' ;
2019-06-07 23:11:08 +01:00
if ( ! ( 'errorHandler' in options ) ) options . errorHandler = 'log' ;
2017-12-26 11:38:53 +01:00
2018-03-09 20:59:12 +00:00
if ( this . state _ !== 'idle' ) {
2020-10-29 23:37:19 +00:00
const msg = ` DecryptionWorker: cannot start because state is " ${ this . state _ } " ` ;
this . logger ( ) . debug ( msg ) ;
return { error : new Error ( msg ) } ;
2017-12-26 11:38:53 +01:00
}
2019-06-10 08:55:36 +01:00
// Note: the logic below is an optimisation to avoid going through the loop if no master key exists
// or if none is loaded. It means this logic needs to be duplicate a bit what's in the loop, like the
// "throw" and "dispatch" logic.
2019-05-28 22:05:11 +01:00
const loadedMasterKeyCount = await this . encryptionService ( ) . loadedMasterKeysCount ( ) ;
if ( ! loadedMasterKeyCount ) {
2020-10-29 23:37:19 +00:00
const msg = 'DecryptionWorker: cannot start because no master key is currently loaded.' ;
this . logger ( ) . info ( msg ) ;
2019-05-28 22:05:11 +01:00
const ids = await MasterKey . allIds ( ) ;
if ( ids . length ) {
2019-06-10 08:55:36 +01:00
if ( options . masterKeyNotLoadedHandler === 'throw' ) {
// By trying to load the master key here, we throw the "masterKeyNotLoaded" error
// which the caller needs.
await this . encryptionService ( ) . loadedMasterKey ( ids [ 0 ] ) ;
} else {
this . dispatch ( {
type : 'MASTERKEY_SET_NOT_LOADED' ,
ids : ids ,
} ) ;
}
2019-05-28 22:05:11 +01:00
}
2020-10-29 23:37:19 +00:00
return { error : new Error ( msg ) } ;
2019-05-28 22:05:11 +01:00
}
2018-03-09 20:59:12 +00:00
this . logger ( ) . info ( 'DecryptionWorker: starting decryption...' ) ;
2017-12-14 17:58:10 +00:00
2018-03-09 20:59:12 +00:00
this . state _ = 'started' ;
2017-12-14 18:53:08 +00:00
2020-03-13 23:46:14 +00:00
const excludedIds = [ ] ;
2020-04-19 10:11:46 +01:00
const decryptedItemCounts = { } ;
2020-05-13 16:28:54 +01:00
let skippedItemCount = 0 ;
2017-12-14 18:53:08 +00:00
2020-03-13 17:42:50 +00:00
this . dispatch ( { type : 'ENCRYPTION_HAS_DISABLED_ITEMS' , value : false } ) ;
2018-06-10 17:43:24 +01:00
this . dispatchReport ( { state : 'started' } ) ;
2017-12-14 19:39:13 +00:00
try {
2018-01-28 17:37:29 +00:00
const notLoadedMasterKeyDisptaches = [ ] ;
2017-12-14 19:39:13 +00:00
while ( true ) {
const result = await BaseItem . itemsThatNeedDecryption ( excludedIds ) ;
const items = result . items ;
for ( let i = 0 ; i < items . length ; i ++ ) {
const item = items [ i ] ;
2018-01-31 19:51:29 +00:00
2017-12-14 19:39:13 +00:00
const ItemClass = BaseItem . itemClass ( item ) ;
2018-06-10 17:43:24 +01:00
this . dispatchReport ( {
itemIndex : i ,
itemCount : items . length ,
} ) ;
2019-06-07 23:11:08 +01:00
2019-09-19 22:51:18 +01:00
const counterKey = ` decrypt: ${ item . type _ } : ${ item . id } ` ;
2019-06-07 23:11:08 +01:00
const clearDecryptionCounter = async ( ) => {
await this . kvStore ( ) . deleteValue ( counterKey ) ;
2019-07-29 15:43:53 +02:00
} ;
2018-03-05 18:21:42 +00:00
// Don't log in production as it results in many messages when importing many items
2019-05-28 22:05:11 +01:00
// this.logger().debug('DecryptionWorker: decrypting: ' + item.id + ' (' + ItemClass.tableName() + ')');
2017-12-14 19:39:13 +00:00
try {
2019-06-07 23:11:08 +01:00
const decryptCounter = await this . kvStore ( ) . incValue ( counterKey ) ;
if ( decryptCounter > this . maxDecryptionAttempts _ ) {
2020-04-08 01:00:01 +01:00
this . logger ( ) . debug ( ` DecryptionWorker: ${ BaseModel . modelTypeToName ( item . type _ ) } ${ item . id } : Decryption has failed more than 2 times - skipping it ` ) ;
2020-03-13 17:42:50 +00:00
this . dispatch ( { type : 'ENCRYPTION_HAS_DISABLED_ITEMS' , value : true } ) ;
2020-05-13 16:28:54 +01:00
skippedItemCount ++ ;
2019-06-07 23:11:08 +01:00
excludedIds . push ( item . id ) ;
continue ;
}
2019-05-22 15:56:07 +01:00
const decryptedItem = await ItemClass . decrypt ( item ) ;
2019-06-07 23:11:08 +01:00
await clearDecryptionCounter ( ) ;
2020-04-19 10:11:46 +01:00
if ( ! decryptedItemCounts [ decryptedItem . type _ ] ) decryptedItemCounts [ decryptedItem . type _ ] = 0 ;
decryptedItemCounts [ decryptedItem . type _ ] ++ ;
2019-05-22 15:56:07 +01:00
if ( decryptedItem . type _ === Resource . modelType ( ) && ! ! decryptedItem . encryption _blob _encrypted ) {
// itemsThatNeedDecryption() will return the resource again if the blob has not been decrypted,
// but that will result in an infinite loop if the blob simply has not been downloaded yet.
2019-07-29 15:43:53 +02:00
// So skip the ID for now, and the service will try to decrypt the blob again the next time.
2019-05-22 15:56:07 +01:00
excludedIds . push ( decryptedItem . id ) ;
}
if ( decryptedItem . type _ === Resource . modelType ( ) && ! decryptedItem . encryption _blob _encrypted ) {
this . eventEmitter _ . emit ( 'resourceDecrypted' , { id : decryptedItem . id } ) ;
}
2019-05-28 22:05:11 +01:00
if ( decryptedItem . type _ === Resource . modelType ( ) && ! decryptedItem . encryption _applied && ! ! decryptedItem . encryption _blob _encrypted ) {
this . eventEmitter _ . emit ( 'resourceMetadataButNotBlobDecrypted' , { id : decryptedItem . id } ) ;
}
2017-12-14 19:39:13 +00:00
} catch ( error ) {
2018-02-14 15:28:56 +00:00
excludedIds . push ( item . id ) ;
2019-05-22 15:56:07 +01:00
2018-10-08 19:11:53 +01:00
if ( error . code === 'masterKeyNotLoaded' && options . masterKeyNotLoadedHandler === 'dispatch' ) {
2018-01-28 17:37:29 +00:00
if ( notLoadedMasterKeyDisptaches . indexOf ( error . masterKeyId ) < 0 ) {
this . dispatch ( {
2018-03-09 20:59:12 +00:00
type : 'MASTERKEY_ADD_NOT_LOADED' ,
2018-01-28 17:37:29 +00:00
id : error . masterKeyId ,
} ) ;
notLoadedMasterKeyDisptaches . push ( error . masterKeyId ) ;
}
2019-06-07 23:11:08 +01:00
await clearDecryptionCounter ( ) ;
2017-12-14 19:39:13 +00:00
continue ;
}
2018-02-26 19:25:54 +00:00
2018-10-08 19:11:53 +01:00
if ( error . code === 'masterKeyNotLoaded' && options . masterKeyNotLoadedHandler === 'throw' ) {
2019-06-07 23:11:08 +01:00
await clearDecryptionCounter ( ) ;
2018-02-26 19:25:54 +00:00
throw error ;
}
2019-06-07 23:11:08 +01:00
if ( options . errorHandler === 'log' ) {
2019-09-19 22:51:18 +01:00
this . logger ( ) . warn ( ` DecryptionWorker: error for: ${ item . id } ( ${ ItemClass . tableName ( ) } ) ` , error , item ) ;
2019-06-07 23:11:08 +01:00
} else {
throw error ;
}
2017-12-14 18:53:08 +00:00
}
}
2017-12-14 19:39:13 +00:00
if ( ! result . hasMore ) break ;
}
} catch ( error ) {
2018-03-09 20:59:12 +00:00
this . logger ( ) . error ( 'DecryptionWorker:' , error ) ;
this . state _ = 'idle' ;
2018-06-10 17:43:24 +01:00
this . dispatchReport ( { state : 'idle' } ) ;
2017-12-26 11:38:53 +01:00
throw error ;
2017-12-14 18:53:08 +00:00
}
2017-12-14 19:39:13 +00:00
2019-05-12 15:53:42 +01:00
// 2019-05-12: Temporary to set the file size of the resources
// that weren't set in migration/20.js due to being on the sync target
await ResourceService . autoSetFileSizes ( ) ;
2018-03-09 20:59:12 +00:00
this . logger ( ) . info ( 'DecryptionWorker: completed decryption.' ) ;
2017-12-26 11:38:53 +01:00
2020-04-08 01:00:01 +01:00
const downloadedButEncryptedBlobCount = await Resource . downloadedButEncryptedBlobCount ( excludedIds ) ;
2019-05-22 15:56:07 +01:00
2018-03-09 20:59:12 +00:00
this . state _ = 'idle' ;
2019-05-22 15:56:07 +01:00
2020-05-13 16:28:54 +01:00
let decryptedItemCount = 0 ;
for ( const itemType in decryptedItemCounts ) decryptedItemCount += decryptedItemCounts [ itemType ] ;
const finalReport = {
skippedItemCount : skippedItemCount ,
2020-04-19 10:11:46 +01:00
decryptedItemCounts : decryptedItemCounts ,
2020-05-13 16:28:54 +01:00
decryptedItemCount : decryptedItemCount ,
} ;
this . dispatchReport ( Object . assign ( { } , finalReport , { state : 'idle' } ) ) ;
2019-05-28 22:05:11 +01:00
2019-05-22 15:56:07 +01:00
if ( downloadedButEncryptedBlobCount ) {
2019-09-19 22:51:18 +01:00
this . logger ( ) . info ( ` DecryptionWorker: Some resources have been downloaded but are not decrypted yet. Scheduling another decryption. Resource count: ${ downloadedButEncryptedBlobCount } ` ) ;
2019-05-22 15:56:07 +01:00
this . scheduleStart ( ) ;
}
2020-05-13 16:28:54 +01:00
return finalReport ;
2017-12-14 17:58:10 +00:00
}
2020-02-22 22:25:16 +11:00
async start ( options ) {
this . startCalls _ . push ( true ) ;
2020-05-13 16:28:54 +01:00
let output = null ;
2020-02-22 22:25:16 +11:00
try {
2020-05-13 16:28:54 +01:00
output = await this . start _ ( options ) ;
2020-02-22 22:25:16 +11:00
} finally {
this . startCalls _ . pop ( ) ;
}
2020-05-13 16:28:54 +01:00
return output ;
2020-02-22 22:25:16 +11:00
}
2020-02-28 05:25:42 +11:00
async destroy ( ) {
this . eventEmitter _ . removeAllListeners ( ) ;
if ( this . scheduleId _ ) {
2020-10-09 18:35:46 +01:00
shim . clearTimeout ( this . scheduleId _ ) ;
2020-02-28 05:25:42 +11:00
this . scheduleId _ = null ;
}
this . eventEmitter _ = null ;
DecryptionWorker . instance _ = null ;
2020-02-22 22:25:16 +11:00
return new Promise ( ( resolve ) => {
2020-10-09 18:35:46 +01:00
const iid = shim . setInterval ( ( ) => {
2020-02-22 22:25:16 +11:00
if ( ! this . startCalls _ . length ) {
2020-10-09 18:35:46 +01:00
shim . clearInterval ( iid ) ;
2020-02-22 22:25:16 +11:00
resolve ( ) ;
}
} , 100 ) ;
} ) ;
}
2017-12-14 17:58:10 +00:00
}
2020-02-28 05:25:42 +11:00
DecryptionWorker . instance _ = null ;
2019-07-29 15:43:53 +02:00
module . exports = DecryptionWorker ;