2018-03-09 20:59:12 +00:00
const { Logger } = require ( 'lib/logger.js' ) ;
const Setting = require ( 'lib/models/Setting.js' ) ;
const { shim } = require ( 'lib/shim.js' ) ;
const SyncTargetRegistry = require ( 'lib/SyncTargetRegistry.js' ) ;
const { _ } = require ( 'lib/locale.js' ) ;
2017-07-06 19:48:17 +00:00
const reg = { } ;
2017-11-23 23:10:55 +00:00
reg . syncTargets _ = { } ;
2017-07-31 18:32:51 +00:00
2017-07-06 19:48:17 +00:00
reg . logger = ( ) => {
2017-07-15 00:12:32 +01:00
if ( ! reg . logger _ ) {
2019-10-09 21:35:13 +02:00
// console.warn('Calling logger before it is initialized');
2017-07-15 00:12:32 +01:00
return new Logger ( ) ;
}
2017-07-06 19:48:17 +00:00
return reg . logger _ ;
2019-07-29 15:43:53 +02:00
} ;
2017-07-06 19:48:17 +00:00
2019-07-29 15:43:53 +02:00
reg . setLogger = l => {
2017-07-09 16:47:05 +01:00
reg . logger _ = l ;
2019-07-29 15:43:53 +02:00
} ;
2017-07-09 16:47:05 +01:00
2019-07-29 15:43:53 +02:00
reg . setShowErrorMessageBoxHandler = v => {
2017-12-01 17:47:18 +00:00
reg . showErrorMessageBoxHandler _ = v ;
2019-07-29 15:43:53 +02:00
} ;
2017-12-01 17:47:18 +00:00
2019-07-29 15:43:53 +02:00
reg . showErrorMessageBox = message => {
2017-12-01 17:47:18 +00:00
if ( ! reg . showErrorMessageBoxHandler _ ) return ;
reg . showErrorMessageBoxHandler _ ( message ) ;
2019-07-29 15:43:53 +02:00
} ;
2017-12-01 17:47:18 +00:00
2018-05-21 16:26:01 +01:00
reg . resetSyncTarget = ( syncTargetId = null ) => {
if ( syncTargetId === null ) syncTargetId = Setting . value ( 'sync.target' ) ;
delete reg . syncTargets _ [ syncTargetId ] ;
2019-07-29 15:43:53 +02:00
} ;
2018-05-21 16:26:01 +01:00
2019-12-13 01:16:34 +00:00
reg . syncTargetNextcloud = ( ) => {
return reg . syncTarget ( SyncTargetRegistry . nameToId ( 'nextcloud' ) ) ;
} ;
2017-11-23 23:10:55 +00:00
reg . syncTarget = ( syncTargetId = null ) => {
2018-03-09 20:59:12 +00:00
if ( syncTargetId === null ) syncTargetId = Setting . value ( 'sync.target' ) ;
2017-11-23 23:10:55 +00:00
if ( reg . syncTargets _ [ syncTargetId ] ) return reg . syncTargets _ [ syncTargetId ] ;
2017-07-06 19:48:17 +00:00
2017-11-24 18:37:40 +00:00
const SyncTargetClass = SyncTargetRegistry . classById ( syncTargetId ) ;
2018-03-09 20:59:12 +00:00
if ( ! reg . db ( ) ) throw new Error ( 'Cannot initialize sync without a db' ) ;
2017-07-06 19:48:17 +00:00
2017-11-23 23:10:55 +00:00
const target = new SyncTargetClass ( reg . db ( ) ) ;
target . setLogger ( reg . logger ( ) ) ;
reg . syncTargets _ [ syncTargetId ] = target ;
return target ;
2019-07-29 15:43:53 +02:00
} ;
2017-07-06 19:48:17 +00:00
2020-03-13 17:42:50 +00:00
// This can be used when some data has been modified and we want to make
// sure it gets synced. So we wait for the current sync operation to
// finish (if one is running), then we trigger a sync just after.
reg . waitForSyncFinishedThenSync = async ( ) => {
2020-03-16 13:30:54 +11:00
reg . waitForReSyncCalls _ . push ( true ) ;
try {
const synchronizer = await reg . syncTarget ( ) . synchronizer ( ) ;
await synchronizer . waitForSyncToFinish ( ) ;
await reg . scheduleSync ( 0 ) ;
} finally {
reg . waitForReSyncCalls _ . pop ( ) ;
}
2020-03-13 17:42:50 +00:00
} ;
2020-03-16 13:30:54 +11:00
reg . scheduleSync = async ( delay = null , syncOptions = null ) => {
reg . schedSyncCalls _ . push ( true ) ;
2017-07-16 22:17:22 +01:00
2020-03-16 13:30:54 +11:00
try {
if ( delay === null ) delay = 1000 * 10 ;
if ( syncOptions === null ) syncOptions = { } ;
2017-07-16 22:17:22 +01:00
2020-03-16 13:30:54 +11:00
let promiseResolve = null ;
const promise = new Promise ( ( resolve ) => {
promiseResolve = resolve ;
} ) ;
2017-11-03 18:51:13 +00:00
2020-03-16 13:30:54 +11:00
if ( reg . scheduleSyncId _ ) {
clearTimeout ( reg . scheduleSyncId _ ) ;
reg . scheduleSyncId _ = null ;
}
2017-07-16 22:17:22 +01:00
2020-03-16 13:30:54 +11:00
reg . logger ( ) . info ( 'Scheduling sync operation...' , delay ) ;
2017-07-26 17:49:01 +00:00
2020-03-16 13:30:54 +11:00
if ( Setting . value ( 'env' ) === 'dev' && delay !== 0 ) {
reg . logger ( ) . info ( 'Schedule sync DISABLED!!!' ) ;
2017-07-16 22:17:22 +01:00
return ;
}
2020-03-16 13:30:54 +11:00
const timeoutCallback = async ( ) => {
reg . timerCallbackCalls _ . push ( true ) ;
2018-01-07 19:20:10 +00:00
try {
2020-03-16 13:30:54 +11:00
reg . scheduleSyncId _ = null ;
reg . logger ( ) . info ( 'Preparing scheduled sync' ) ;
2018-01-07 19:20:10 +00:00
2020-03-16 13:30:54 +11:00
const syncTargetId = Setting . value ( 'sync.target' ) ;
if ( ! ( await reg . syncTarget ( syncTargetId ) . isAuthenticated ( ) ) ) {
reg . logger ( ) . info ( 'Synchroniser is missing credentials - manual sync required to authenticate.' ) ;
2017-11-28 00:22:38 +00:00
promiseResolve ( ) ;
2020-03-16 13:30:54 +11:00
return ;
2017-07-30 22:22:57 +02:00
}
2017-07-26 17:49:01 +00:00
2020-03-16 13:30:54 +11:00
try {
const sync = await reg . syncTarget ( syncTargetId ) . synchronizer ( ) ;
const contextKey = ` sync. ${ syncTargetId } .context ` ;
let context = Setting . value ( contextKey ) ;
try {
context = context ? JSON . parse ( context ) : { } ;
} catch ( error ) {
// Clearing the context is inefficient since it means all items are going to be re-downloaded
// however it won't result in duplicate items since the synchroniser is going to compare each
// item to the current state.
reg . logger ( ) . warn ( ` Could not parse JSON sync context ${ contextKey } : ` , context ) ;
reg . logger ( ) . info ( 'Clearing context and starting from scratch' ) ;
context = null ;
}
try {
reg . logger ( ) . info ( 'Starting scheduled sync' ) ;
const options = Object . assign ( { } , syncOptions , { context : context } ) ;
if ( ! options . saveContextHandler ) {
options . saveContextHandler = newContext => {
Setting . setValue ( contextKey , JSON . stringify ( newContext ) ) ;
} ;
}
const newContext = await sync . start ( options ) ;
Setting . setValue ( contextKey , JSON . stringify ( newContext ) ) ;
} catch ( error ) {
if ( error . code == 'alreadyStarted' ) {
reg . logger ( ) . info ( error . message ) ;
} else {
promiseResolve ( ) ;
throw error ;
}
}
} catch ( error ) {
reg . logger ( ) . info ( 'Could not run background sync:' ) ;
reg . logger ( ) . info ( error ) ;
// Special case to display OneDrive Business error. This is the full error that's received when trying to use a OneDrive Business account:
//
// {"error":"invalid_client","error_description":"AADSTS50011: The reply address 'http://localhost:1917' does not match the reply addresses configured for
// the application: 'cbabb902-d276-4ea4-aa88-062a5889d6dc'. More details: not specified\r\nTrace ID: 6e63dac6-8b37-47e2-bd1b-4768f8713400\r\nCorrelation
// ID: acfd6503-8d97-4349-ae2e-e7a19dd7b6bc\r\nTimestamp: 2017-12-01 13:35:55Z","error_codes":[50011],"timestamp":"2017-12-01 13:35:55Z","trace_id":
// "6e63dac6-8b37-47e2-bd1b-4768f8713400","correlation_id":"acfd6503-8d97-4349-ae2e-e7a19dd7b6bc"}: TOKEN: null Error: {"error":"invalid_client",
// "error_description":"AADSTS50011: The reply address 'http://localhost:1917' does not match the reply addresses configured for the application:
// 'cbabb902-d276-4ea4-aa88-062a5889d6dc'. More details: not specified\r\nTrace ID: 6e63dac6-8b37-47e2-bd1b-4768f8713400\r\nCorrelation ID
// acfd6503-8d97-4349-ae2e-e7a19dd7b6bc\r\nTimestamp: 2017-12-01 13:35:55Z","error_codes":[50011],"timestamp":"2017-12-01 13:35:55Z","trace_id":
// "6e63dac6-8b37-47e2-bd1b-4768f8713400","correlation_id":"acfd6503-8d97-4349-ae2e-e7a19dd7b6bc"}
if ( error && error . message && error . message . indexOf ( '"invalid_client"' ) >= 0 ) {
reg . showErrorMessageBox ( _ ( 'Could not synchronize with OneDrive.\n\nThis error often happens when using OneDrive for Business, which unfortunately cannot be supported.\n\nPlease consider using a regular OneDrive account.' ) ) ;
}
}
reg . setupRecurrentSync ( ) ;
promiseResolve ( ) ;
2017-07-24 21:36:49 +01:00
2020-03-16 13:30:54 +11:00
} finally {
reg . timerCallbackCalls _ . pop ( ) ;
}
} ;
2017-11-28 00:22:38 +00:00
2020-03-16 13:30:54 +11:00
if ( delay === 0 ) {
timeoutCallback ( ) ;
} else {
reg . scheduleSyncId _ = setTimeout ( timeoutCallback , delay ) ;
}
return promise ;
2017-07-16 22:17:22 +01:00
2020-02-22 22:25:16 +11:00
} finally {
2020-03-16 13:30:54 +11:00
reg . schedSyncCalls _ . pop ( ) ;
2020-02-22 22:25:16 +11:00
}
} ;
2017-07-26 17:49:01 +00:00
reg . setupRecurrentSync = ( ) => {
2020-03-16 13:30:54 +11:00
reg . setupRecurrentCalls _ . push ( true ) ;
2017-07-26 17:49:01 +00:00
2020-03-16 13:30:54 +11:00
try {
if ( reg . recurrentSyncId _ ) {
shim . clearInterval ( reg . recurrentSyncId _ ) ;
reg . recurrentSyncId _ = null ;
2018-02-21 19:58:28 +00:00
}
2020-03-16 13:30:54 +11:00
if ( ! Setting . value ( 'sync.interval' ) ) {
reg . logger ( ) . debug ( 'Recurrent sync is disabled' ) ;
} else {
reg . logger ( ) . debug ( ` Setting up recurrent sync with interval ${ Setting . value ( 'sync.interval' ) } ` ) ;
if ( Setting . value ( 'env' ) === 'dev' ) {
reg . logger ( ) . info ( 'Recurrent sync operation DISABLED!!!' ) ;
return ;
}
reg . recurrentSyncId _ = shim . setInterval ( ( ) => {
reg . logger ( ) . info ( 'Running background sync on timer...' ) ;
reg . scheduleSync ( 0 ) ;
} , 1000 * Setting . value ( 'sync.interval' ) ) ;
}
} finally {
reg . setupRecurrentCalls _ . pop ( ) ;
2017-08-20 16:29:18 +02:00
}
2019-07-29 15:43:53 +02:00
} ;
2017-07-26 17:49:01 +00:00
2019-07-29 15:43:53 +02:00
reg . setDb = v => {
2017-07-06 19:48:17 +00:00
reg . db _ = v ;
2019-07-29 15:43:53 +02:00
} ;
2017-07-06 19:48:17 +00:00
reg . db = ( ) => {
return reg . db _ ;
2019-07-29 15:43:53 +02:00
} ;
2017-07-06 19:48:17 +00:00
2020-03-16 13:30:54 +11:00
reg . cancelTimers _ = ( ) => {
2020-02-28 05:25:42 +11:00
if ( this . recurrentSyncId _ ) {
2020-03-16 13:30:54 +11:00
shim . clearInterval ( reg . recurrentSyncId _ ) ;
2020-02-28 05:25:42 +11:00
this . recurrentSyncId _ = null ;
}
2020-03-16 13:30:54 +11:00
if ( reg . scheduleSyncId _ ) {
clearTimeout ( reg . scheduleSyncId _ ) ;
reg . scheduleSyncId _ = null ;
}
} ;
reg . cancelTimers = async ( ) => {
reg . logger ( ) . info ( 'Cancelling sync timers' ) ;
reg . cancelTimers _ ( ) ;
2020-02-22 22:25:16 +11:00
return new Promise ( ( resolve ) => {
2020-03-16 13:30:54 +11:00
setInterval ( ( ) => {
// ensure processing complete
if ( ! reg . setupRecurrentCalls _ . length && ! reg . schedSyncCalls _ . length && ! reg . timerCallbackCalls _ . length && ! reg . waitForReSyncCalls _ . length ) {
reg . cancelTimers _ ( ) ;
2020-02-22 22:25:16 +11:00
resolve ( ) ;
}
} , 100 ) ;
} ) ;
} ;
reg . syncCalls _ = [ ] ;
2020-03-16 13:30:54 +11:00
reg . schedSyncCalls _ = [ ] ;
reg . waitForReSyncCalls _ = [ ] ;
reg . setupRecurrentCalls _ = [ ] ;
reg . timerCallbackCalls _ = [ ] ;
2020-02-22 22:25:16 +11:00
2019-07-29 15:43:53 +02:00
module . exports = { reg } ;