* parserlib: fast section extraction, tweaks and speedups
* csslint: "simple-not" rule
* csslint: enable and fix "selector-newline" rule
* simplify db: resolve with result
* simplify download()
* remove noCode param as it wastes more time/memory on copying
* styleManager: switch style<->data names to reflect their actual contents
* inline method bodies to avoid indirection and enable better autocomplete/hint/jump support in IDE
* upgrade getEventKeyName to handle mouse clicks
* don't trust location.href as it hides text fragment
* getAllKeys is implemented since Chrome48, FF44
* allow recoverable css errors + async'ify usercss.js
* openManage: unminimize windows
* remove the obsolete Chrome pre-65 workaround
* fix temporal dead zone in apply.js
* ff bug workaround for simple editor window
* consistent window scrolling in scrollToEditor and jumpToPos
* rework waitForSelector and collapsible <details>
* blank paint frame workaround for new Chrome
* extract stuff from edit.js and load on demand
* simplify regexpTester::isShown
* move MozDocMapper to sections-util.js
* extract fitSelectBox()
* initialize router earlier
* use helpPopup.close()
* fix autofocus in popups, follow-up to 5bb1b5ef
* clone objects in prefs.get() + cosmetics
* reuse getAll result for INC
		
	
			
		
			
				
	
	
		
			226 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			226 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* global API msg */// msg.js
 | |
| /* global chromeLocal */// storage-util.js
 | |
| /* global compareRevision */// common.js
 | |
| /* global prefs */
 | |
| /* global tokenMan */
 | |
| 'use strict';
 | |
| 
 | |
| const syncMan = (() => {
 | |
|   //#region Init
 | |
| 
 | |
|   const SYNC_DELAY = 1; // minutes
 | |
|   const SYNC_INTERVAL = 30; // minutes
 | |
|   const STATES = Object.freeze({
 | |
|     connected: 'connected',
 | |
|     connecting: 'connecting',
 | |
|     disconnected: 'disconnected',
 | |
|     disconnecting: 'disconnecting',
 | |
|   });
 | |
|   const STORAGE_KEY = 'sync/state/';
 | |
|   const status = /** @namespace SyncManager.Status */ {
 | |
|     STATES,
 | |
|     state: STATES.disconnected,
 | |
|     syncing: false,
 | |
|     progress: null,
 | |
|     currentDriveName: null,
 | |
|     errorMessage: null,
 | |
|     login: false,
 | |
|   };
 | |
|   let ctrl;
 | |
|   let currentDrive;
 | |
|   /** @type {Promise|boolean} will be `true` to avoid wasting a microtask tick on each `await` */
 | |
|   let ready = prefs.ready.then(() => {
 | |
|     ready = true;
 | |
|     prefs.subscribe('sync.enabled',
 | |
|       (_, val) => val === 'none'
 | |
|         ? syncMan.stop()
 | |
|         : syncMan.start(val, true),
 | |
|       {runNow: true});
 | |
|   });
 | |
| 
 | |
|   chrome.alarms.onAlarm.addListener(info => {
 | |
|     if (info.name === 'syncNow') {
 | |
|       syncMan.syncNow();
 | |
|     }
 | |
|   });
 | |
| 
 | |
|   //#endregion
 | |
|   //#region Exports
 | |
| 
 | |
|   return {
 | |
| 
 | |
|     async delete(...args) {
 | |
|       if (ready.then) await ready;
 | |
|       if (!currentDrive) return;
 | |
|       schedule();
 | |
|       return ctrl.delete(...args);
 | |
|     },
 | |
| 
 | |
|     /** @returns {Promise<SyncManager.Status>} */
 | |
|     async getStatus() {
 | |
|       return status;
 | |
|     },
 | |
| 
 | |
|     async login(name = prefs.get('sync.enabled')) {
 | |
|       if (ready.then) await ready;
 | |
|       try {
 | |
|         await tokenMan.getToken(name, true);
 | |
|       } catch (err) {
 | |
|         if (/Authorization page could not be loaded/i.test(err.message)) {
 | |
|           // FIXME: Chrome always fails at the first login so we try again
 | |
|           await tokenMan.getToken(name);
 | |
|         }
 | |
|         throw err;
 | |
|       }
 | |
|       status.login = true;
 | |
|       emitStatusChange();
 | |
|     },
 | |
| 
 | |
|     async put(...args) {
 | |
|       if (ready.then) await ready;
 | |
|       if (!currentDrive) return;
 | |
|       schedule();
 | |
|       return ctrl.put(...args);
 | |
|     },
 | |
| 
 | |
|     async start(name, fromPref = false) {
 | |
|       if (ready.then) await ready;
 | |
|       if (!ctrl) await initController();
 | |
|       if (currentDrive) return;
 | |
|       currentDrive = getDrive(name);
 | |
|       ctrl.use(currentDrive);
 | |
|       status.state = STATES.connecting;
 | |
|       status.currentDriveName = currentDrive.name;
 | |
|       status.login = true;
 | |
|       emitStatusChange();
 | |
|       try {
 | |
|         if (!fromPref) {
 | |
|           await syncMan.login(name).catch(handle401Error);
 | |
|         }
 | |
|         await syncMan.syncNow();
 | |
|         status.errorMessage = null;
 | |
|       } catch (err) {
 | |
|         status.errorMessage = err.message;
 | |
|         // FIXME: should we move this logic to options.js?
 | |
|         if (!fromPref) {
 | |
|           console.error(err);
 | |
|           return syncMan.stop();
 | |
|         }
 | |
|       }
 | |
|       prefs.set('sync.enabled', name);
 | |
|       status.state = STATES.connected;
 | |
|       schedule(SYNC_INTERVAL);
 | |
|       emitStatusChange();
 | |
|     },
 | |
| 
 | |
|     async stop() {
 | |
|       if (ready.then) await ready;
 | |
|       if (!currentDrive) return;
 | |
|       chrome.alarms.clear('syncNow');
 | |
|       status.state = STATES.disconnecting;
 | |
|       emitStatusChange();
 | |
|       try {
 | |
|         await ctrl.stop();
 | |
|         await tokenMan.revokeToken(currentDrive.name);
 | |
|         await chromeLocal.remove(STORAGE_KEY + currentDrive.name);
 | |
|       } catch (e) {}
 | |
|       currentDrive = null;
 | |
|       prefs.set('sync.enabled', 'none');
 | |
|       status.state = STATES.disconnected;
 | |
|       status.currentDriveName = null;
 | |
|       status.login = false;
 | |
|       emitStatusChange();
 | |
|     },
 | |
| 
 | |
|     async syncNow() {
 | |
|       if (ready.then) await ready;
 | |
|       if (!currentDrive) throw new Error('cannot sync when disconnected');
 | |
|       try {
 | |
|         await (ctrl.isInit() ? ctrl.syncNow() : ctrl.start()).catch(handle401Error);
 | |
|         status.errorMessage = null;
 | |
|       } catch (err) {
 | |
|         status.errorMessage = err.message;
 | |
|       }
 | |
|       emitStatusChange();
 | |
|     },
 | |
|   };
 | |
| 
 | |
|   //#endregion
 | |
|   //#region Utils
 | |
| 
 | |
|   async function initController() {
 | |
|     await require(['/vendor/db-to-cloud/db-to-cloud.min']); /* global dbToCloud */
 | |
|     ctrl = dbToCloud.dbToCloud({
 | |
|       onGet(id) {
 | |
|         return API.styles.getByUUID(id);
 | |
|       },
 | |
|       onPut(doc) {
 | |
|         return API.styles.putByUUID(doc);
 | |
|       },
 | |
|       onDelete(id, rev) {
 | |
|         return API.styles.deleteByUUID(id, rev);
 | |
|       },
 | |
|       async onFirstSync() {
 | |
|         for (const i of await API.styles.getAll()) {
 | |
|           ctrl.put(i._id, i._rev);
 | |
|         }
 | |
|       },
 | |
|       onProgress(e) {
 | |
|         if (e.phase === 'start') {
 | |
|           status.syncing = true;
 | |
|         } else if (e.phase === 'end') {
 | |
|           status.syncing = false;
 | |
|           status.progress = null;
 | |
|         } else {
 | |
|           status.progress = e;
 | |
|         }
 | |
|         emitStatusChange();
 | |
|       },
 | |
|       compareRevision,
 | |
|       getState(drive) {
 | |
|         return chromeLocal.getValue(STORAGE_KEY + drive.name);
 | |
|       },
 | |
|       setState(drive, state) {
 | |
|         return chromeLocal.setValue(STORAGE_KEY + drive.name, state);
 | |
|       },
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   async function handle401Error(err) {
 | |
|     let emit;
 | |
|     if (err.code === 401) {
 | |
|       await tokenMan.revokeToken(currentDrive.name).catch(console.error);
 | |
|       emit = true;
 | |
|     } else if (/User interaction required|Requires user interaction/i.test(err.message)) {
 | |
|       emit = true;
 | |
|     }
 | |
|     if (emit) {
 | |
|       status.login = false;
 | |
|       emitStatusChange();
 | |
|     }
 | |
|     return Promise.reject(err);
 | |
|   }
 | |
| 
 | |
|   function emitStatusChange() {
 | |
|     msg.broadcastExtension({method: 'syncStatusUpdate', status});
 | |
|   }
 | |
| 
 | |
|   function getDrive(name) {
 | |
|     if (name === 'dropbox' || name === 'google' || name === 'onedrive') {
 | |
|       return dbToCloud.drive[name]({
 | |
|         getAccessToken: () => tokenMan.getToken(name),
 | |
|       });
 | |
|     }
 | |
|     throw new Error(`unknown cloud name: ${name}`);
 | |
|   }
 | |
| 
 | |
|   function schedule(delay = SYNC_DELAY) {
 | |
|     chrome.alarms.create('syncNow', {
 | |
|       delayInMinutes: delay,
 | |
|       periodInMinutes: SYNC_INTERVAL,
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   //#endregion
 | |
| })();
 |