2019-05-06 21:35:29 +01:00
const ItemChange = require ( 'lib/models/ItemChange' ) ;
const Note = require ( 'lib/models/Note' ) ;
const Folder = require ( 'lib/models/Folder' ) ;
2020-10-09 18:35:46 +01:00
const Setting = require ( 'lib/models/Setting' ) . default ;
2019-05-06 21:35:29 +01:00
const Revision = require ( 'lib/models/Revision' ) ;
const BaseModel = require ( 'lib/BaseModel' ) ;
const ItemChangeUtils = require ( 'lib/services/ItemChangeUtils' ) ;
2020-10-09 18:35:46 +01:00
const shim = require ( 'lib/shim' ) . default ;
const BaseService = require ( 'lib/services/BaseService' ) . default ;
const { _ } = require ( 'lib/locale' ) ;
2019-05-14 22:23:34 +01:00
const { sprintf } = require ( 'sprintf-js' ) ;
2020-10-09 18:35:46 +01:00
const { wrapError } = require ( 'lib/errorUtils' ) ;
2019-05-06 21:35:29 +01:00
class RevisionService extends BaseService {
constructor ( ) {
super ( ) ;
// An "old note" is one that has been created before the revision service existed. These
// notes never benefited from revisions so the first time they are modified, a copy of
// the original note is saved. The goal is to have at least one revision in case the note
// is deleted or modified as a result of a bug or user mistake.
this . isOldNotesCache _ = { } ;
2020-03-16 13:30:54 +11:00
this . maintenanceCalls _ = [ ] ;
this . maintenanceTimer1 _ = null ;
this . maintenanceTimer2 _ = null ;
2019-05-06 21:35:29 +01:00
}
static instance ( ) {
if ( this . instance _ ) return this . instance _ ;
this . instance _ = new RevisionService ( ) ;
return this . instance _ ;
}
2019-05-08 00:51:56 +01:00
oldNoteCutOffDate _ ( ) {
return Date . now ( ) - Setting . value ( 'revisionService.oldNoteInterval' ) ;
}
2019-05-06 21:35:29 +01:00
async isOldNote ( noteId ) {
if ( noteId in this . isOldNotesCache _ ) return this . isOldNotesCache _ [ noteId ] ;
2019-05-08 00:51:56 +01:00
const isOld = await Note . noteIsOlderThan ( noteId , this . oldNoteCutOffDate _ ( ) ) ;
this . isOldNotesCache _ [ noteId ] = isOld ;
return isOld ;
2019-05-06 21:35:29 +01:00
}
noteMetadata _ ( note ) {
const excludedFields = [ 'type_' , 'title' , 'body' , 'created_time' , 'updated_time' , 'encryption_applied' , 'encryption_cipher_text' , 'is_conflict' ] ;
const md = { } ;
2020-03-13 23:46:14 +00:00
for ( const k in note ) {
2019-05-06 21:35:29 +01:00
if ( excludedFields . indexOf ( k ) >= 0 ) continue ;
md [ k ] = note [ k ] ;
}
2019-05-07 20:46:58 +01:00
if ( note . user _updated _time === note . updated _time ) delete md . user _updated _time ;
if ( note . user _created _time === note . created _time ) delete md . user _created _time ;
2019-05-06 21:35:29 +01:00
return md ;
}
2019-05-07 20:46:58 +01:00
isEmptyRevision _ ( rev ) {
2019-07-29 15:43:53 +02:00
if ( rev . title _diff ) return false ;
if ( rev . body _diff ) return false ;
2019-05-07 20:46:58 +01:00
const md = JSON . parse ( rev . metadata _diff ) ;
2019-05-07 22:15:47 +01:00
if ( md . new && Object . keys ( md . new ) . length ) return false ;
if ( md . deleted && Object . keys ( md . deleted ) . length ) return false ;
2019-05-07 20:46:58 +01:00
return true ;
}
async createNoteRevision _ ( note , parentRevId = null ) {
2020-10-09 18:35:46 +01:00
try {
const parentRev = parentRevId ? await Revision . load ( parentRevId ) : await Revision . latestRevision ( BaseModel . TYPE _NOTE , note . id ) ;
const output = {
parent _id : '' ,
item _type : BaseModel . TYPE _NOTE ,
item _id : note . id ,
item _updated _time : note . updated _time ,
} ;
const noteMd = this . noteMetadata _ ( note ) ;
const noteTitle = note . title ? note . title : '' ;
const noteBody = note . body ? note . body : '' ;
if ( ! parentRev ) {
output . title _diff = Revision . createTextPatch ( '' , noteTitle ) ;
output . body _diff = Revision . createTextPatch ( '' , noteBody ) ;
output . metadata _diff = Revision . createObjectPatch ( { } , noteMd ) ;
} else {
if ( Date . now ( ) - parentRev . updated _time < Setting . value ( 'revisionService.intervalBetweenRevisions' ) ) return null ;
2019-05-06 21:35:29 +01:00
2020-10-09 18:35:46 +01:00
const merged = await Revision . mergeDiffs ( parentRev ) ;
output . parent _id = parentRev . id ;
output . title _diff = Revision . createTextPatch ( merged . title , noteTitle ) ;
output . body _diff = Revision . createTextPatch ( merged . body , noteBody ) ;
output . metadata _diff = Revision . createObjectPatch ( merged . metadata , noteMd ) ;
}
2019-05-06 21:35:29 +01:00
2020-10-09 18:35:46 +01:00
if ( this . isEmptyRevision _ ( output ) ) return null ;
return Revision . save ( output ) ;
} catch ( error ) {
const newError = wrapError ( ` Could not create revision for note: ${ note . id } ` , error ) ;
throw newError ;
}
2019-05-06 21:35:29 +01:00
}
async collectRevisions ( ) {
if ( this . isCollecting _ ) return ;
this . isCollecting _ = true ;
await ItemChange . waitForAllSaved ( ) ;
const doneNoteIds = [ ] ;
try {
while ( true ) {
// See synchronizer test units to see why changes coming
// from sync are skipped.
2019-07-29 15:43:53 +02:00
const changes = await ItemChange . modelSelectAll (
`
2019-05-06 21:35:29 +01:00
SELECT id , item _id , type , before _change _item
FROM item _changes
WHERE item _type = ?
AND source != ?
2019-05-28 18:10:21 +01:00
AND source != ?
2019-05-06 21:35:29 +01:00
AND id > ?
ORDER BY id ASC
LIMIT 10
2019-07-29 15:43:53 +02:00
` ,
[ BaseModel . TYPE _NOTE , ItemChange . SOURCE _SYNC , ItemChange . SOURCE _DECRYPTION , Setting . value ( 'revisionService.lastProcessedChangeId' ) ]
) ;
2019-05-06 21:35:29 +01:00
if ( ! changes . length ) break ;
2020-05-21 09:14:33 +01:00
const noteIds = changes . map ( a => a . item _id ) ;
2019-09-19 22:51:18 +01:00
const notes = await Note . modelSelectAll ( ` SELECT * FROM notes WHERE is_conflict = 0 AND encryption_applied = 0 AND id IN (" ${ noteIds . join ( '","' ) } ") ` ) ;
2019-05-06 21:35:29 +01:00
for ( let i = 0 ; i < changes . length ; i ++ ) {
const change = changes [ i ] ;
const noteId = change . item _id ;
if ( change . type === ItemChange . TYPE _UPDATE && doneNoteIds . indexOf ( noteId ) < 0 ) {
const note = BaseModel . byId ( notes , noteId ) ;
const oldNote = change . before _change _item ? JSON . parse ( change . before _change _item ) : null ;
if ( note ) {
2019-05-08 00:51:56 +01:00
if ( oldNote && oldNote . updated _time < this . oldNoteCutOffDate _ ( ) ) {
2019-05-06 21:35:29 +01:00
// This is where we save the original version of this old note
2019-05-14 22:23:34 +01:00
const rev = await this . createNoteRevision _ ( oldNote ) ;
if ( rev ) this . logger ( ) . debug ( sprintf ( 'RevisionService::collectRevisions: Saved revision %s (old note)' , rev . id ) ) ;
2019-05-06 21:35:29 +01:00
}
2019-05-14 22:23:34 +01:00
const rev = await this . createNoteRevision _ ( note ) ;
if ( rev ) this . logger ( ) . debug ( sprintf ( 'RevisionService::collectRevisions: Saved revision %s (Last rev was more than %d ms ago)' , rev . id , Setting . value ( 'revisionService.intervalBetweenRevisions' ) ) ) ;
2019-05-06 21:35:29 +01:00
doneNoteIds . push ( noteId ) ;
this . isOldNotesCache _ [ noteId ] = false ;
}
}
if ( change . type === ItemChange . TYPE _DELETE && ! ! change . before _change _item ) {
const note = JSON . parse ( change . before _change _item ) ;
const revExists = await Revision . revisionExists ( BaseModel . TYPE _NOTE , note . id , note . updated _time ) ;
2019-05-14 22:23:34 +01:00
if ( ! revExists ) {
const rev = await this . createNoteRevision _ ( note ) ;
if ( rev ) this . logger ( ) . debug ( sprintf ( 'RevisionService::collectRevisions: Saved revision %s (for deleted note)' , rev . id ) ) ;
}
2019-05-06 21:35:29 +01:00
doneNoteIds . push ( noteId ) ;
}
Setting . setValue ( 'revisionService.lastProcessedChangeId' , change . id ) ;
}
}
} catch ( error ) {
if ( error . code === 'revision_encrypted' ) {
// One or more revisions are encrypted - stop processing for now
// and these revisions will be processed next time the revision
// collector runs.
this . logger ( ) . info ( 'RevisionService::collectRevisions: One or more revision was encrypted. Processing was stopped but will resume later when the revision is decrypted.' , error ) ;
} else {
this . logger ( ) . error ( 'RevisionService::collectRevisions:' , error ) ;
}
}
await Setting . saveAll ( ) ;
2019-07-29 15:43:53 +02:00
await ItemChangeUtils . deleteProcessedChanges ( ) ;
2019-05-06 21:35:29 +01:00
this . isCollecting _ = false ;
2019-09-19 22:51:18 +01:00
this . logger ( ) . info ( ` RevisionService::collectRevisions: Created revisions for ${ doneNoteIds . length } notes ` ) ;
2019-05-06 21:35:29 +01:00
}
async deleteOldRevisions ( ttl ) {
return Revision . deleteOldRevisions ( ttl ) ;
}
async revisionNote ( revisions , index ) {
2019-09-19 22:51:18 +01:00
if ( index < 0 || index >= revisions . length ) throw new Error ( ` Invalid revision index: ${ index } ` ) ;
2019-05-06 21:35:29 +01:00
const rev = revisions [ index ] ;
const merged = await Revision . mergeDiffs ( rev , revisions ) ;
2019-07-29 15:43:53 +02:00
const output = Object . assign (
{
title : merged . title ,
body : merged . body ,
} ,
merged . metadata
) ;
2019-05-06 21:35:29 +01:00
output . updated _time = output . user _updated _time ;
output . created _time = output . user _created _time ;
output . type _ = BaseModel . TYPE _NOTE ;
return output ;
}
restoreFolderTitle ( ) {
return _ ( 'Restored Notes' ) ;
}
async restoreFolder ( ) {
let folder = await Folder . loadByTitle ( this . restoreFolderTitle ( ) ) ;
if ( ! folder ) {
folder = await Folder . save ( { title : this . restoreFolderTitle ( ) } ) ;
}
return folder ;
}
async importRevisionNote ( note ) {
const toImport = Object . assign ( { } , note ) ;
delete toImport . id ;
delete toImport . updated _time ;
delete toImport . created _time ;
delete toImport . encryption _applied ;
delete toImport . encryption _cipher _text ;
const folder = await this . restoreFolder ( ) ;
toImport . parent _id = folder . id ;
await Note . save ( toImport ) ;
}
async maintenance ( ) {
2020-03-16 13:30:54 +11:00
this . maintenanceCalls _ . push ( true ) ;
try {
const startTime = Date . now ( ) ;
this . logger ( ) . info ( 'RevisionService::maintenance: Starting...' ) ;
if ( ! Setting . value ( 'revisionService.enabled' ) ) {
this . logger ( ) . info ( 'RevisionService::maintenance: Service is disabled' ) ;
// We do as if we had processed all the latest changes so that they can be cleaned up
// later on by ItemChangeUtils.deleteProcessedChanges().
Setting . setValue ( 'revisionService.lastProcessedChangeId' , await ItemChange . lastChangeId ( ) ) ;
await this . deleteOldRevisions ( Setting . value ( 'revisionService.ttlDays' ) * 24 * 60 * 60 * 1000 ) ;
} else {
this . logger ( ) . info ( 'RevisionService::maintenance: Service is enabled' ) ;
await this . collectRevisions ( ) ;
await this . deleteOldRevisions ( Setting . value ( 'revisionService.ttlDays' ) * 24 * 60 * 60 * 1000 ) ;
2019-05-06 21:35:29 +01:00
2020-03-16 13:30:54 +11:00
this . logger ( ) . info ( ` RevisionService::maintenance: Done in ${ Date . now ( ) - startTime } ms ` ) ;
}
} finally {
this . maintenanceCalls _ . pop ( ) ;
}
2019-05-06 21:35:29 +01:00
}
runInBackground ( collectRevisionInterval = null ) {
if ( this . isRunningInBackground _ ) return ;
this . isRunningInBackground _ = true ;
if ( collectRevisionInterval === null ) collectRevisionInterval = 1000 * 60 * 10 ;
2019-09-19 22:51:18 +01:00
this . logger ( ) . info ( ` RevisionService::runInBackground: Starting background service with revision collection interval ${ collectRevisionInterval } ` ) ;
2019-05-06 21:35:29 +01:00
2020-10-09 18:35:46 +01:00
this . maintenanceTimer1 _ = shim . setTimeout ( ( ) => {
2019-05-06 21:35:29 +01:00
this . maintenance ( ) ;
} , 1000 * 4 ) ;
2019-07-29 15:43:53 +02:00
2020-03-16 13:30:54 +11:00
this . maintenanceTImer2 _ = shim . setInterval ( ( ) => {
2019-05-06 21:35:29 +01:00
this . maintenance ( ) ;
} , collectRevisionInterval ) ;
}
2020-03-16 13:30:54 +11:00
async cancelTimers ( ) {
if ( this . maintenanceTimer1 _ ) {
2020-10-09 18:35:46 +01:00
shim . clearTimeout ( this . maintenanceTimer1 ) ;
2020-03-16 13:30:54 +11:00
this . maintenanceTimer1 _ = null ;
}
if ( this . maintenanceTimer2 _ ) {
shim . clearInterval ( this . maintenanceTimer2 ) ;
this . maintenanceTimer2 _ = null ;
}
return new Promise ( ( resolve ) => {
2020-10-09 18:35:46 +01:00
const iid = shim . setInterval ( ( ) => {
2020-03-16 13:30:54 +11:00
if ( ! this . maintenanceCalls _ . length ) {
2020-10-09 18:35:46 +01:00
shim . clearInterval ( iid ) ;
2020-03-16 13:30:54 +11:00
resolve ( ) ;
}
} , 100 ) ;
} ) ;
}
2019-05-06 21:35:29 +01:00
}
2019-07-29 15:43:53 +02:00
module . exports = RevisionService ;