forked from mirrors/gecko-dev
		
	 1ee73485b0
			
		
	
	
		1ee73485b0
		
	
	
	
	
		
			
			Remove now unused pref "browser.sessionstore.max_write_failures". Rename and simplify test_shutdown_cleanup.js to make it more clear that it is testing the isFinalWrite option. Differential Revision: https://phabricator.services.mozilla.com/D137930
		
			
				
	
	
		
			400 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			400 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* This Source Code Form is subject to the terms of the Mozilla Public
 | |
|  * License, v. 2.0. If a copy of the MPL was not distributed with this file,
 | |
|  * You can obtain one at http://mozilla.org/MPL/2.0/. */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| var EXPORTED_SYMBOLS = ["SessionWriter"];
 | |
| 
 | |
| /**
 | |
|  * We just started (we haven't written anything to disk yet) from
 | |
|  * `Paths.clean`. The backup directory may not exist.
 | |
|  */
 | |
| const STATE_CLEAN = "clean";
 | |
| /**
 | |
|  * We know that `Paths.recovery` is good, either because we just read
 | |
|  * it (we haven't written anything to disk yet) or because have
 | |
|  * already written once to `Paths.recovery` during this session.
 | |
|  * `Paths.clean` is absent or invalid. The backup directory exists.
 | |
|  */
 | |
| const STATE_RECOVERY = "recovery";
 | |
| /**
 | |
|  * We just started from `Paths.upgradeBackup` (we haven't written
 | |
|  * anything to disk yet). Both `Paths.clean`, `Paths.recovery` and
 | |
|  * `Paths.recoveryBackup` are absent or invalid. The backup directory
 | |
|  * exists.
 | |
|  */
 | |
| const STATE_UPGRADE_BACKUP = "upgradeBackup";
 | |
| /**
 | |
|  * We just started without a valid session store file (we haven't
 | |
|  * written anything to disk yet). The backup directory may not exist.
 | |
|  */
 | |
| const STATE_EMPTY = "empty";
 | |
| 
 | |
| var sessionFileIOMutex = Promise.resolve();
 | |
| // Ensure that we don't do concurrent I/O on the same file.
 | |
| // Example usage:
 | |
| // const unlock = await lockIOWithMutex();
 | |
| // try {
 | |
| //   ... (Do I/O work here.)
 | |
| // } finally { unlock(); }
 | |
| function lockIOWithMutex() {
 | |
|   // Return a Promise that resolves when the mutex is free.
 | |
|   return new Promise(unlock => {
 | |
|     // Overwrite the mutex variable with a chained-on, new Promise. The Promise
 | |
|     // we returned to the caller can be called to resolve that new Promise
 | |
|     // and unlock the mutex.
 | |
|     sessionFileIOMutex = sessionFileIOMutex.then(() => {
 | |
|       return new Promise(unlock);
 | |
|     });
 | |
|   });
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Interface dedicated to handling I/O for Session Store.
 | |
|  */
 | |
| const SessionWriter = {
 | |
|   init(origin, useOldExtension, paths, prefs = {}) {
 | |
|     return SessionWriterInternal.init(origin, useOldExtension, paths, prefs);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Write the contents of the session file.
 | |
|    * @param state - May get changed on shutdown.
 | |
|    */
 | |
|   async write(state, options = {}) {
 | |
|     const unlock = await lockIOWithMutex();
 | |
|     try {
 | |
|       return await SessionWriterInternal.write(state, options);
 | |
|     } finally {
 | |
|       unlock();
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   async wipe() {
 | |
|     const unlock = await lockIOWithMutex();
 | |
|     try {
 | |
|       return await SessionWriterInternal.wipe();
 | |
|     } finally {
 | |
|       unlock();
 | |
|     }
 | |
|   },
 | |
| };
 | |
| 
 | |
| const SessionWriterInternal = {
 | |
|   // Path to the files used by the SessionWriter
 | |
|   Paths: null,
 | |
| 
 | |
|   /**
 | |
|    * The current state of the session file, as one of the following strings:
 | |
|    * - "empty" if we have started without any sessionstore;
 | |
|    * - one of "clean", "recovery", "recoveryBackup", "cleanBackup",
 | |
|    *   "upgradeBackup" if we have started by loading the corresponding file.
 | |
|    */
 | |
|   state: null,
 | |
| 
 | |
|   /**
 | |
|    * A flag that indicates we loaded a session file with the deprecated .js extension.
 | |
|    */
 | |
|   useOldExtension: false,
 | |
| 
 | |
|   /**
 | |
|    * Number of old upgrade backups that are being kept
 | |
|    */
 | |
|   maxUpgradeBackups: null,
 | |
| 
 | |
|   /**
 | |
|    * Initialize (or reinitialize) the writer.
 | |
|    *
 | |
|    * @param {string} origin Which of sessionstore.js or its backups
 | |
|    *   was used. One of the `STATE_*` constants defined above.
 | |
|    * @param {boolean} a flag indicate whether we loaded a session file with ext .js
 | |
|    * @param {object} paths The paths at which to find the various files.
 | |
|    * @param {object} prefs The preferences the writer needs to know.
 | |
|    */
 | |
|   init(origin, useOldExtension, paths, prefs) {
 | |
|     if (!(origin in paths || origin == STATE_EMPTY)) {
 | |
|       throw new TypeError("Invalid origin: " + origin);
 | |
|     }
 | |
| 
 | |
|     // Check that all required preference values were passed.
 | |
|     for (let pref of [
 | |
|       "maxUpgradeBackups",
 | |
|       "maxSerializeBack",
 | |
|       "maxSerializeForward",
 | |
|     ]) {
 | |
|       if (!prefs.hasOwnProperty(pref)) {
 | |
|         throw new TypeError(`Missing preference value for ${pref}`);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     this.useOldExtension = useOldExtension;
 | |
|     this.state = origin;
 | |
|     this.Paths = paths;
 | |
|     this.maxUpgradeBackups = prefs.maxUpgradeBackups;
 | |
|     this.maxSerializeBack = prefs.maxSerializeBack;
 | |
|     this.maxSerializeForward = prefs.maxSerializeForward;
 | |
|     this.upgradeBackupNeeded = paths.nextUpgradeBackup != paths.upgradeBackup;
 | |
|     return { result: true };
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Write the session to disk.
 | |
|    * Write the session to disk, performing any necessary backup
 | |
|    * along the way.
 | |
|    *
 | |
|    * @param {object} state The state to write to disk.
 | |
|    * @param {object} options
 | |
|    *  - performShutdownCleanup If |true|, we should
 | |
|    *    perform shutdown-time cleanup to ensure that private data
 | |
|    *    is not left lying around;
 | |
|    *  - isFinalWrite If |true|, write to Paths.clean instead of
 | |
|    *    Paths.recovery
 | |
|    */
 | |
|   async write(state, options) {
 | |
|     let exn;
 | |
|     let telemetry = {};
 | |
| 
 | |
|     // Cap the number of backward and forward shistory entries on shutdown.
 | |
|     if (options.isFinalWrite) {
 | |
|       for (let window of state.windows) {
 | |
|         for (let tab of window.tabs) {
 | |
|           let lower = 0;
 | |
|           let upper = tab.entries.length;
 | |
| 
 | |
|           if (this.maxSerializeBack > -1) {
 | |
|             lower = Math.max(lower, tab.index - this.maxSerializeBack - 1);
 | |
|           }
 | |
|           if (this.maxSerializeForward > -1) {
 | |
|             upper = Math.min(upper, tab.index + this.maxSerializeForward);
 | |
|           }
 | |
| 
 | |
|           tab.entries = tab.entries.slice(lower, upper);
 | |
|           tab.index -= lower;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     try {
 | |
|       if (this.state == STATE_CLEAN || this.state == STATE_EMPTY) {
 | |
|         // The backups directory may not exist yet. In all other cases,
 | |
|         // we have either already read from or already written to this
 | |
|         // directory, so we are satisfied that it exists.
 | |
|         await IOUtils.makeDirectory(this.Paths.backups);
 | |
|       }
 | |
| 
 | |
|       if (this.state == STATE_CLEAN) {
 | |
|         // Move $Path.clean out of the way, to avoid any ambiguity as
 | |
|         // to which file is more recent.
 | |
|         if (!this.useOldExtension) {
 | |
|           await IOUtils.move(this.Paths.clean, this.Paths.cleanBackup);
 | |
|         } else {
 | |
|           // Since we are migrating from .js to .jsonlz4,
 | |
|           // we need to compress the deprecated $Path.clean
 | |
|           // and write it to $Path.cleanBackup.
 | |
|           let oldCleanPath = this.Paths.clean.replace("jsonlz4", "js");
 | |
|           let d = await IOUtils.read(oldCleanPath);
 | |
|           await IOUtils.write(this.Paths.cleanBackup, d, { compress: true });
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       let startWriteMs = Date.now();
 | |
|       let fileStat;
 | |
| 
 | |
|       if (options.isFinalWrite) {
 | |
|         // We are shutting down. At this stage, we know that
 | |
|         // $Paths.clean is either absent or corrupted. If it was
 | |
|         // originally present and valid, it has been moved to
 | |
|         // $Paths.cleanBackup a long time ago. We can therefore write
 | |
|         // with the guarantees that we erase no important data.
 | |
|         await IOUtils.writeJSON(this.Paths.clean, state, {
 | |
|           tmpPath: this.Paths.clean + ".tmp",
 | |
|           compress: true,
 | |
|         });
 | |
|         fileStat = await IOUtils.stat(this.Paths.clean);
 | |
|       } else if (this.state == STATE_RECOVERY) {
 | |
|         // At this stage, either $Paths.recovery was written >= 15
 | |
|         // seconds ago during this session or we have just started
 | |
|         // from $Paths.recovery left from the previous session. Either
 | |
|         // way, $Paths.recovery is good. We can move $Path.backup to
 | |
|         // $Path.recoveryBackup without erasing a good file with a bad
 | |
|         // file.
 | |
|         await IOUtils.writeJSON(this.Paths.recovery, state, {
 | |
|           tmpPath: this.Paths.recovery + ".tmp",
 | |
|           backupFile: this.Paths.recoveryBackup,
 | |
|           compress: true,
 | |
|         });
 | |
|         fileStat = await IOUtils.stat(this.Paths.recovery);
 | |
|       } else {
 | |
|         // In other cases, either $Path.recovery is not necessary, or
 | |
|         // it doesn't exist or it has been corrupted. Regardless,
 | |
|         // don't backup $Path.recovery.
 | |
|         await IOUtils.writeJSON(this.Paths.recovery, state, {
 | |
|           tmpPath: this.Paths.recovery + ".tmp",
 | |
|           compress: true,
 | |
|         });
 | |
|         fileStat = await IOUtils.stat(this.Paths.recovery);
 | |
|       }
 | |
| 
 | |
|       telemetry.FX_SESSION_RESTORE_WRITE_FILE_MS = Date.now() - startWriteMs;
 | |
|       telemetry.FX_SESSION_RESTORE_FILE_SIZE_BYTES = fileStat.size;
 | |
|     } catch (ex) {
 | |
|       // Don't throw immediately
 | |
|       exn = exn || ex;
 | |
|     }
 | |
| 
 | |
|     // If necessary, perform an upgrade backup
 | |
|     let upgradeBackupComplete = false;
 | |
|     if (
 | |
|       this.upgradeBackupNeeded &&
 | |
|       (this.state == STATE_CLEAN || this.state == STATE_UPGRADE_BACKUP)
 | |
|     ) {
 | |
|       try {
 | |
|         // If we loaded from `clean`, the file has since then been renamed to `cleanBackup`.
 | |
|         let path =
 | |
|           this.state == STATE_CLEAN
 | |
|             ? this.Paths.cleanBackup
 | |
|             : this.Paths.upgradeBackup;
 | |
|         await IOUtils.copy(path, this.Paths.nextUpgradeBackup);
 | |
|         this.upgradeBackupNeeded = false;
 | |
|         upgradeBackupComplete = true;
 | |
|       } catch (ex) {
 | |
|         // Don't throw immediately
 | |
|         exn = exn || ex;
 | |
|       }
 | |
| 
 | |
|       // Find all backups
 | |
|       let backups = [];
 | |
| 
 | |
|       try {
 | |
|         let children = await IOUtils.getChildren(this.Paths.backups);
 | |
|         backups = children.filter(path =>
 | |
|           path.startsWith(this.Paths.upgradeBackupPrefix)
 | |
|         );
 | |
|       } catch (ex) {
 | |
|         // Don't throw immediately
 | |
|         exn = exn || ex;
 | |
|       }
 | |
| 
 | |
|       // If too many backups exist, delete them
 | |
|       if (backups.length > this.maxUpgradeBackups) {
 | |
|         // Use alphanumerical sort since dates are in YYYYMMDDHHMMSS format
 | |
|         backups.sort();
 | |
|         // remove backup file if it is among the first (n-maxUpgradeBackups) files
 | |
|         for (let i = 0; i < backups.length - this.maxUpgradeBackups; i++) {
 | |
|           try {
 | |
|             await IOUtils.remove(backups[i]);
 | |
|           } catch (ex) {
 | |
|             exn = exn || ex;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (options.performShutdownCleanup && !exn) {
 | |
|       // During shutdown, if auto-restore is disabled, we need to
 | |
|       // remove possibly sensitive data that has been stored purely
 | |
|       // for crash recovery. Note that this slightly decreases our
 | |
|       // ability to recover from OS-level/hardware-level issue.
 | |
| 
 | |
|       // If an exception was raised, we assume that we still need
 | |
|       // these files.
 | |
|       await IOUtils.remove(this.Paths.recoveryBackup);
 | |
|       await IOUtils.remove(this.Paths.recovery);
 | |
|     }
 | |
| 
 | |
|     this.state = STATE_RECOVERY;
 | |
| 
 | |
|     if (exn) {
 | |
|       throw exn;
 | |
|     }
 | |
| 
 | |
|     return {
 | |
|       result: {
 | |
|         upgradeBackup: upgradeBackupComplete,
 | |
|       },
 | |
|       telemetry,
 | |
|     };
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Wipes all files holding session data from disk.
 | |
|    */
 | |
|   async wipe() {
 | |
|     // Don't stop immediately in case of error.
 | |
|     let exn = null;
 | |
| 
 | |
|     // Erase main session state file
 | |
|     try {
 | |
|       await IOUtils.remove(this.Paths.clean);
 | |
|       // Remove old extension ones.
 | |
|       let oldCleanPath = this.Paths.clean.replace("jsonlz4", "js");
 | |
|       await IOUtils.remove(oldCleanPath, {
 | |
|         ignoreAbsent: true,
 | |
|       });
 | |
|     } catch (ex) {
 | |
|       // Don't stop immediately.
 | |
|       exn = exn || ex;
 | |
|     }
 | |
| 
 | |
|     // Wipe the Session Restore directory
 | |
|     try {
 | |
|       await IOUtils.remove(this.Paths.backups, { recursive: true });
 | |
|     } catch (ex) {
 | |
|       exn = exn || ex;
 | |
|     }
 | |
| 
 | |
|     // Wipe legacy Session Restore files from the profile directory
 | |
|     try {
 | |
|       await this._wipeFromDir(PathUtils.profileDir, "sessionstore.bak");
 | |
|     } catch (ex) {
 | |
|       exn = exn || ex;
 | |
|     }
 | |
| 
 | |
|     this.state = STATE_EMPTY;
 | |
|     if (exn) {
 | |
|       throw exn;
 | |
|     }
 | |
| 
 | |
|     return { result: true };
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Wipe a number of files from a directory.
 | |
|    *
 | |
|    * @param {string} path The directory.
 | |
|    * @param {string} prefix Remove files whose
 | |
|    * name starts with the prefix.
 | |
|    */
 | |
|   async _wipeFromDir(path, prefix) {
 | |
|     // Sanity check
 | |
|     if (!prefix) {
 | |
|       throw new TypeError("Must supply prefix");
 | |
|     }
 | |
| 
 | |
|     let exn = null;
 | |
| 
 | |
|     let children = await IOUtils.getChildren(path, {
 | |
|       ignoreAbsent: true,
 | |
|     });
 | |
|     for (let entryPath of children) {
 | |
|       if (!PathUtils.filename(entryPath).startsWith(prefix)) {
 | |
|         continue;
 | |
|       }
 | |
|       try {
 | |
|         let { type } = await IOUtils.stat(entryPath);
 | |
|         if (type == "directory") {
 | |
|           continue;
 | |
|         }
 | |
|         await IOUtils.remove(entryPath);
 | |
|       } catch (ex) {
 | |
|         // Don't stop immediately
 | |
|         exn = exn || ex;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (exn) {
 | |
|       throw exn;
 | |
|     }
 | |
|   },
 | |
| };
 |