2021-01-22 19:41:11 +02:00
import ItemChange from '../models/ItemChange' ;
import Note from '../models/Note' ;
import Folder from '../models/Folder' ;
import Setting from '../models/Setting' ;
import Revision from '../models/Revision' ;
import BaseModel from '../BaseModel' ;
import ItemChangeUtils from './ItemChangeUtils' ;
import shim from '../shim' ;
import BaseService from './BaseService' ;
import { _ } from '../locale' ;
import { ItemChangeEntity , NoteEntity , RevisionEntity } from './database/types' ;
2019-05-14 23:23:34 +02:00
const { sprintf } = require ( 'sprintf-js' ) ;
2020-11-05 18:58:23 +02:00
const { wrapError } = require ( '../errorUtils' ) ;
2019-05-06 22:35:29 +02:00
2021-01-22 19:41:11 +02:00
export default class RevisionService extends BaseService {
2019-05-06 22:35:29 +02:00
2021-01-22 19:41:11 +02:00
public static instance_ : RevisionService ;
2020-03-16 04:30:54 +02:00
2021-01-22 19:41:11 +02:00
// 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.
private isOldNotesCache_ : any = { } ;
private maintenanceCalls_ : any [ ] = [ ] ;
private maintenanceTimer1_ : any = null ;
private maintenanceTimer2_ : any = null ;
private isCollecting_ = false ;
public isRunningInBackground_ = false ;
2019-05-06 22:35:29 +02:00
static instance() {
if ( this . instance_ ) return this . instance_ ;
this . instance_ = new RevisionService ( ) ;
return this . instance_ ;
}
2019-05-08 01:51:56 +02:00
oldNoteCutOffDate_() {
return Date . now ( ) - Setting . value ( 'revisionService.oldNoteInterval' ) ;
}
2021-01-22 19:41:11 +02:00
async isOldNote ( noteId : string ) {
2019-05-06 22:35:29 +02:00
if ( noteId in this . isOldNotesCache_ ) return this . isOldNotesCache_ [ noteId ] ;
2019-05-08 01:51:56 +02:00
const isOld = await Note . noteIsOlderThan ( noteId , this . oldNoteCutOffDate_ ( ) ) ;
this . isOldNotesCache_ [ noteId ] = isOld ;
return isOld ;
2019-05-06 22:35:29 +02:00
}
2021-01-22 19:41:11 +02:00
noteMetadata_ ( note : NoteEntity ) {
2019-05-06 22:35:29 +02:00
const excludedFields = [ 'type_' , 'title' , 'body' , 'created_time' , 'updated_time' , 'encryption_applied' , 'encryption_cipher_text' , 'is_conflict' ] ;
2021-01-22 19:41:11 +02:00
const md : any = { } ;
2020-03-14 01:46:14 +02:00
for ( const k in note ) {
2019-05-06 22:35:29 +02:00
if ( excludedFields . indexOf ( k ) >= 0 ) continue ;
2021-01-22 19:41:11 +02:00
md [ k ] = ( note as any ) [ k ] ;
2019-05-06 22:35:29 +02:00
}
2019-05-07 21:46:58 +02: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 22:35:29 +02:00
return md ;
}
2021-01-22 19:41:11 +02:00
isEmptyRevision_ ( rev : RevisionEntity ) {
2019-07-29 15:43:53 +02:00
if ( rev . title_diff ) return false ;
if ( rev . body_diff ) return false ;
2019-05-07 21:46:58 +02:00
const md = JSON . parse ( rev . metadata_diff ) ;
2019-05-07 23:15:47 +02:00
if ( md . new && Object . keys ( md . new ) . length ) return false ;
if ( md . deleted && Object . keys ( md . deleted ) . length ) return false ;
2019-05-07 21:46:58 +02:00
return true ;
}
2021-01-22 19:41:11 +02:00
async createNoteRevision_ ( note : NoteEntity , parentRevId : string = null ) {
2020-10-09 19:35:46 +02:00
try {
const parentRev = parentRevId ? await Revision . load ( parentRevId ) : await Revision . latestRevision ( BaseModel . TYPE_NOTE , note . id ) ;
2021-01-22 19:41:11 +02:00
const output : RevisionEntity = {
2020-10-09 19:35:46 +02:00
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 22:35:29 +02:00
2020-10-09 19:35:46 +02: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 22:35:29 +02:00
2020-10-09 19:35:46 +02: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 22:35:29 +02: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.
2021-01-22 19:41:11 +02:00
const changes : ItemChangeEntity [ ] = await ItemChange . modelSelectAll (
2019-07-29 15:43:53 +02:00
`
2019-05-06 22:35:29 +02:00
SELECT id , item_id , type , before_change_item
FROM item_changes
WHERE item_type = ?
AND source != ?
2019-05-28 19:10:21 +02:00
AND source != ?
2019-05-06 22:35:29 +02: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 22:35:29 +02:00
if ( ! changes . length ) break ;
2021-01-22 19:41:11 +02:00
const noteIds = changes . map ( ( a ) = > a . item_id ) ;
2019-09-19 23:51:18 +02: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 22:35:29 +02: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 01:51:56 +02:00
if ( oldNote && oldNote . updated_time < this . oldNoteCutOffDate_ ( ) ) {
2019-05-06 22:35:29 +02:00
// This is where we save the original version of this old note
2019-05-14 23:23:34 +02: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 22:35:29 +02:00
}
2019-05-14 23:23:34 +02: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 22:35:29 +02: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 23:23:34 +02: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 22:35:29 +02: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 22:35:29 +02:00
this . isCollecting_ = false ;
2019-09-19 23:51:18 +02:00
this . logger ( ) . info ( ` RevisionService::collectRevisions: Created revisions for ${ doneNoteIds . length } notes ` ) ;
2019-05-06 22:35:29 +02:00
}
2021-01-22 19:41:11 +02:00
async deleteOldRevisions ( ttl : number ) {
2019-05-06 22:35:29 +02:00
return Revision . deleteOldRevisions ( ttl ) ;
}
2021-01-22 19:41:11 +02:00
async revisionNote ( revisions : RevisionEntity [ ] , index : number ) {
2019-09-19 23:51:18 +02:00
if ( index < 0 || index >= revisions . length ) throw new Error ( ` Invalid revision index: ${ index } ` ) ;
2019-05-06 22:35:29 +02:00
const rev = revisions [ index ] ;
const merged = await Revision . mergeDiffs ( rev , revisions ) ;
2021-01-22 19:41:11 +02:00
const output : NoteEntity = Object . assign (
2019-07-29 15:43:53 +02:00
{
title : merged.title ,
body : merged.body ,
} ,
merged . metadata
) ;
2019-05-06 22:35:29 +02:00
output . updated_time = output . user_updated_time ;
output . created_time = output . user_created_time ;
2021-01-22 19:41:11 +02:00
( output as any ) . type_ = BaseModel . TYPE_NOTE ;
2019-05-06 22:35:29 +02:00
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 ;
}
2021-01-22 19:41:11 +02:00
async importRevisionNote ( note : NoteEntity ) {
2019-05-06 22:35:29 +02:00
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 04:30:54 +02: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 22:35:29 +02:00
2020-03-16 04:30:54 +02:00
this . logger ( ) . info ( ` RevisionService::maintenance: Done in ${ Date . now ( ) - startTime } ms ` ) ;
}
} finally {
this . maintenanceCalls_ . pop ( ) ;
}
2019-05-06 22:35:29 +02:00
}
2021-01-22 19:41:11 +02:00
runInBackground ( collectRevisionInterval : number = null ) {
2019-05-06 22:35:29 +02:00
if ( this . isRunningInBackground_ ) return ;
this . isRunningInBackground_ = true ;
if ( collectRevisionInterval === null ) collectRevisionInterval = 1000 * 60 * 10 ;
2019-09-19 23:51:18 +02:00
this . logger ( ) . info ( ` RevisionService::runInBackground: Starting background service with revision collection interval ${ collectRevisionInterval } ` ) ;
2019-05-06 22:35:29 +02:00
2020-10-09 19:35:46 +02:00
this . maintenanceTimer1_ = shim . setTimeout ( ( ) = > {
2021-01-22 19:41:11 +02:00
void this . maintenance ( ) ;
2019-05-06 22:35:29 +02:00
} , 1000 * 4 ) ;
2019-07-29 15:43:53 +02:00
2021-01-22 19:41:11 +02:00
this . maintenanceTimer2_ = shim . setInterval ( ( ) = > {
void this . maintenance ( ) ;
2019-05-06 22:35:29 +02:00
} , collectRevisionInterval ) ;
}
2020-03-16 04:30:54 +02:00
async cancelTimers() {
if ( this . maintenanceTimer1_ ) {
2021-01-22 19:41:11 +02:00
shim . clearTimeout ( this . maintenanceTimer1_ ) ;
2020-03-16 04:30:54 +02:00
this . maintenanceTimer1_ = null ;
}
if ( this . maintenanceTimer2_ ) {
2021-01-22 19:41:11 +02:00
shim . clearInterval ( this . maintenanceTimer2_ ) ;
2020-03-16 04:30:54 +02:00
this . maintenanceTimer2_ = null ;
}
return new Promise ( ( resolve ) = > {
2020-10-09 19:35:46 +02:00
const iid = shim . setInterval ( ( ) = > {
2020-03-16 04:30:54 +02:00
if ( ! this . maintenanceCalls_ . length ) {
2020-10-09 19:35:46 +02:00
shim . clearInterval ( iid ) ;
2021-01-22 19:41:11 +02:00
resolve ( null ) ;
2020-03-16 04:30:54 +02:00
}
} , 100 ) ;
} ) ;
}
2019-05-06 22:35:29 +02:00
}