diff options
Diffstat (limited to 'toolkit/components/backgroundtasks/BackgroundTask_removeDirectory.sys.mjs')
-rw-r--r-- | toolkit/components/backgroundtasks/BackgroundTask_removeDirectory.sys.mjs | 336 |
1 files changed, 336 insertions, 0 deletions
diff --git a/toolkit/components/backgroundtasks/BackgroundTask_removeDirectory.sys.mjs b/toolkit/components/backgroundtasks/BackgroundTask_removeDirectory.sys.mjs new file mode 100644 index 0000000000..e49f925145 --- /dev/null +++ b/toolkit/components/backgroundtasks/BackgroundTask_removeDirectory.sys.mjs @@ -0,0 +1,336 @@ +/* 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/. */ + +const lazy = {}; +ChromeUtils.defineESModuleGetters(lazy, { + setTimeout: "resource://gre/modules/Timer.sys.mjs", +}); + +import { AppConstants } from "resource://gre/modules/AppConstants.sys.mjs"; +import { EXIT_CODE } from "resource://gre/modules/BackgroundTasksManager.sys.mjs"; + +class Metrics { + /** + * @param {string} metricsId + */ + constructor(metricsId) { + this.metricsId = metricsId; + this.startedTime = new Date(); + + this.wasFirst = true; + this.retryCount = 0; + this.removalCountObj = { value: 0 }; + this.succeeded = true; + + this.suffixRemovalCountObj = { value: 0 }; + this.suffixEverFailed = false; + } + + async report() { + if (!this.metricsId) { + console.warn(`Skipping Glean as no metrics id is passed`); + return; + } + if (AppConstants.MOZ_APP_NAME !== "firefox") { + console.warn( + `Skipping Glean as the app is not Firefox: ${AppConstants.MOZ_APP_NAME}` + ); + return; + } + + const elapsedMs = new Date().valueOf() - this.startedTime.valueOf(); + + // Note(krosylight): This FOG initialization happens within a unique + // temporary directory created for each background task, which will + // be removed after each run. + // That means any failed submission will be lost, but we are fine with + // that as we only have a single submission. + Services.fog.initializeFOG(undefined, "firefox.desktop.background.tasks"); + + const gleanMetrics = Glean[`backgroundTasksRmdir${this.metricsId}`]; + if (!gleanMetrics) { + throw new Error( + `The metrics id "${this.metricsId}" is not available in toolkit/components/backgroundtasks/metrics.yaml. ` + + `Make sure that the id has no typo and is in PascalCase. ` + + `Note that you can omit the id for testing.` + ); + } + + gleanMetrics.elapsedMs.set(elapsedMs); + gleanMetrics.wasFirst.set(this.wasFirst); + gleanMetrics.retryCount.set(this.retryCount); + gleanMetrics.removalCount.set(this.removalCountObj.value); + gleanMetrics.succeeded.set(this.succeeded); + gleanMetrics.suffixRemovalCount.set(this.suffixRemovalCountObj.value); + gleanMetrics.suffixEverFailed.set(this.suffixEverFailed); + + GleanPings.backgroundTasks.submit(); + + // XXX: We wait for arbitrary time for Glean to submit telemetry. + // Bug 1790702 should add a better way. + console.error("Pinged glean, waiting for submission."); + await new Promise(resolve => lazy.setTimeout(resolve, 5000)); + } +} + +// Recursively removes a directory. +// Returns true if it succeeds, false otherwise. +function tryRemoveDir(aFile, countObj) { + try { + aFile.remove(true, countObj); + } catch (e) { + return false; + } + + return true; +} + +const FILE_CHECK_ITERATION_TIMEOUT_MS = 1000; + +function cleanupDirLockFile(aLock, aProfileName) { + let lockFile = aLock.getLockFile(aProfileName); + try { + // Try to clean up the lock file + lockFile.remove(false); + } catch (ex) {} +} + +async function deleteChildDirectory( + parentDirPath, + childDirName, + secondsToWait, + metrics +) { + if (!childDirName || !childDirName.length) { + return; + } + + let targetFile = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile); + targetFile.initWithPath(parentDirPath); + targetFile.append(childDirName); + + // We create the lock before the file is actually there so this task + // is the first one to acquire the lock. Otherwise a different task + // could be cleaning suffixes and start deleting the folder while this + // task is waiting for it to show up. + let dirLock = Cc["@mozilla.org/net/CachePurgeLock;1"].createInstance( + Ci.nsICachePurgeLock + ); + + let locked = false; + try { + dirLock.lock(childDirName); + locked = true; + metrics.wasFirst = !dirLock.isOtherInstanceRunning(); + } catch (e) { + console.error("Failed to check dirLock"); + } + + if (!metrics.wasFirst) { + if (locked) { + dirLock.unlock(); + locked = false; + } + console.error("Another instance is already purging this directory"); + return; + } + + // This backgroundtask process is spawned by the call to + // PR_CreateProcessDetached in CacheFileIOManager::SyncRemoveAllCacheFiles + // Only if spawning the process is successful is the cache folder renamed, + // so we need to wait until that is done. + while (!targetFile.exists()) { + if ( + metrics.retryCount * FILE_CHECK_ITERATION_TIMEOUT_MS > + secondsToWait * 1000 + ) { + // We don't know for sure if the folder was renamed or if a different + // task removed it already. The second variant is more likely but to + // be sure we'd have to consult a log file, which introduces + // more complexity. + console.error(`couldn't find cache folder ${targetFile.path}`); + if (locked) { + dirLock.unlock(); + locked = false; + } + return; + } + await new Promise(resolve => + lazy.setTimeout(resolve, FILE_CHECK_ITERATION_TIMEOUT_MS) + ); + metrics.retryCount++; + console.error(`Cache folder attempt no ${metrics.retryCount}`); + } + + if (!targetFile.isDirectory()) { + if (locked) { + dirLock.unlock(); + locked = false; + } + throw new Error("Path was not a directory"); + } + + console.error(`started removing ${targetFile.path}`); + try { + targetFile.remove(true, metrics.removalCountObj); + } catch (err) { + console.error( + `failed removing ${targetFile.path}. removed ${metrics.removalCountObj.value} entries.` + ); + throw err; + } finally { + console.error( + `done removing ${targetFile.path}. removed ${metrics.removalCountObj.value} entries.` + ); + if (locked) { + dirLock.unlock(); + locked = false; + cleanupDirLockFile(dirLock, childDirName); + } + } +} + +async function cleanupOtherDirectories( + parentDirPath, + otherFoldersSuffix, + metrics +) { + if (!otherFoldersSuffix || !otherFoldersSuffix.length) { + return; + } + + let targetFile = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile); + targetFile.initWithPath(parentDirPath); + + let entries = targetFile.directoryEntries; + while (entries.hasMoreElements()) { + let entry = entries.nextFile; + + if (!entry.leafName.endsWith(otherFoldersSuffix)) { + continue; + } + + let shouldProcessEntry = false; + // The folder could already be gone, so isDirectory could throw + try { + shouldProcessEntry = entry.isDirectory(); + } catch (e) {} + + if (!shouldProcessEntry) { + continue; + } + + let dirLock = Cc["@mozilla.org/net/CachePurgeLock;1"].createInstance( + Ci.nsICachePurgeLock + ); + let wasFirst = false; + + try { + dirLock.lock(entry.leafName); + wasFirst = !dirLock.isOtherInstanceRunning(); + } catch (e) { + console.error("Failed to check dirlock. Skipping folder"); + dirLock.unlock(); + continue; + } + + if (!wasFirst) { + dirLock.unlock(); + continue; + } + + // Remove directory recursively. + let removedDir = tryRemoveDir(entry, metrics.suffixRemovalCountObj); + if (!removedDir && entry.exists()) { + // If first deletion of the directory failed, then we try again once more + // just in case. + metrics.suffixEverFailed = true; + removedDir = tryRemoveDir(entry, metrics.suffixRemovalCountObj); + } + console.error( + `Deletion of folder ${entry.leafName} - success=${removedDir}` + ); + dirLock.unlock(); + cleanupDirLockFile(dirLock, entry.leafName); + } +} + +// Usage: +// removeDirectory parentDirPath childDirName secondsToWait [otherFoldersSuffix] +// arg0 arg1 arg2 arg3 +// [--test-sleep testSleep] +// [--metrics-id metricsId] +// parentDirPath - The path to the parent directory that includes the target directory +// childDirName - The "leaf name" of the moved cache directory +// If empty, the background task will only purge folders that have the "otherFoldersSuffix". +// secondsToWait - String representing the number of seconds to wait for the cacheDir to be moved +// otherFoldersSuffix - [optional] The suffix of directories that should be removed +// When not empty, this task will also attempt to remove all directories in +// the parent dir that end with this suffix +// testSleep - [optional] A test-only argument to sleep for a given milliseconds before removal. +// This exists to test whether a long-running task can survive. +// metricsId - [optional] The identifier for Glean metrics, in PascalCase. +// It'll be submitted only when the matching identifier exists in +// toolkit/components/backgroundtasks/metrics.yaml. +export async function runBackgroundTask(commandLine) { + const testSleep = Number.parseInt( + commandLine.handleFlagWithParam("test-sleep", false) + ); + const metricsId = commandLine.handleFlagWithParam("metrics-id", false) || ""; + + if (commandLine.length < 3) { + throw new Error("Insufficient arguments"); + } + + const parentDirPath = commandLine.getArgument(0); + const childDirName = commandLine.getArgument(1); + let secondsToWait = parseInt(commandLine.getArgument(2)); + if (isNaN(secondsToWait)) { + secondsToWait = 10; + } + commandLine.removeArguments(0, 2); + + let otherFoldersSuffix = ""; + if (commandLine.length) { + otherFoldersSuffix = commandLine.getArgument(0); + commandLine.removeArguments(0, 0); + } + + if (commandLine.length) { + throw new Error( + `${commandLine.length} unknown command args exist, closing.` + ); + } + + console.error( + parentDirPath, + childDirName, + secondsToWait, + otherFoldersSuffix, + metricsId + ); + + if (!Number.isNaN(testSleep)) { + await new Promise(resolve => lazy.setTimeout(resolve, testSleep)); + } + + const metrics = new Metrics(metricsId); + + try { + await deleteChildDirectory( + parentDirPath, + childDirName, + secondsToWait, + metrics + ); + await cleanupOtherDirectories(parentDirPath, otherFoldersSuffix, metrics); + } catch (err) { + metrics.succeeded = false; + throw err; + } finally { + await metrics.report(); + } + + return EXIT_CODE.SUCCESS; +} |