421 lines
13 KiB
JavaScript
421 lines
13 KiB
JavaScript
/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*- */
|
|
/* vim: set sw=2 ts=2 sts=2 et */
|
|
/* 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/. */
|
|
|
|
/*
|
|
* Migrates from a Firefox profile in a lossy manner in order to clean up a
|
|
* user's profile. Data is only migrated where the benefits outweigh the
|
|
* potential problems caused by importing undesired/invalid configurations
|
|
* from the source profile.
|
|
*/
|
|
|
|
import { MigrationUtils } from "resource:///modules/MigrationUtils.sys.mjs";
|
|
|
|
import { MigratorBase } from "resource:///modules/MigratorBase.sys.mjs";
|
|
|
|
const lazy = {};
|
|
|
|
ChromeUtils.defineESModuleGetters(lazy, {
|
|
FileUtils: "resource://gre/modules/FileUtils.sys.mjs",
|
|
PlacesBackups: "resource://gre/modules/PlacesBackups.sys.mjs",
|
|
ProfileAge: "resource://gre/modules/ProfileAge.sys.mjs",
|
|
SessionMigration: "resource:///modules/sessionstore/SessionMigration.sys.mjs",
|
|
});
|
|
|
|
/**
|
|
* Firefox profile migrator. Currently, this class only does "pave over"
|
|
* migrations, where various parts of an old profile overwrite a new
|
|
* profile. This is distinct from other migrators which attempt to import
|
|
* old profile data into the existing profile.
|
|
*
|
|
* This migrator is what powers the "Profile Refresh" mechanism.
|
|
*/
|
|
export class FirefoxProfileMigrator extends MigratorBase {
|
|
static get key() {
|
|
return "firefox";
|
|
}
|
|
|
|
static get displayNameL10nID() {
|
|
return "migration-wizard-migrator-display-name-firefox";
|
|
}
|
|
|
|
static get brandImage() {
|
|
return "chrome://branding/content/icon128.png";
|
|
}
|
|
|
|
_getAllProfiles() {
|
|
let allProfiles = new Map();
|
|
let profileService = Cc[
|
|
"@mozilla.org/toolkit/profile-service;1"
|
|
].getService(Ci.nsIToolkitProfileService);
|
|
for (let profile of profileService.profiles) {
|
|
let rootDir = profile.rootDir;
|
|
|
|
if (
|
|
rootDir.exists() &&
|
|
rootDir.isReadable() &&
|
|
!rootDir.equals(MigrationUtils.profileStartup.directory)
|
|
) {
|
|
allProfiles.set(profile.name, rootDir);
|
|
}
|
|
}
|
|
return allProfiles;
|
|
}
|
|
|
|
getSourceProfiles() {
|
|
let sorter = (a, b) => {
|
|
return a.id.toLocaleLowerCase().localeCompare(b.id.toLocaleLowerCase());
|
|
};
|
|
|
|
return [...this._getAllProfiles().keys()]
|
|
.map(x => ({ id: x, name: x }))
|
|
.sort(sorter);
|
|
}
|
|
|
|
_getFileObject(dir, fileName) {
|
|
let file = dir.clone();
|
|
file.append(fileName);
|
|
|
|
// File resources are monolithic. We don't make partial copies since
|
|
// they are not expected to work alone. Return null to avoid trying to
|
|
// copy non-existing files.
|
|
return file.exists() ? file : null;
|
|
}
|
|
|
|
getResources(aProfile) {
|
|
let sourceProfileDir = aProfile
|
|
? this._getAllProfiles().get(aProfile.id)
|
|
: Cc["@mozilla.org/toolkit/profile-service;1"].getService(
|
|
Ci.nsIToolkitProfileService
|
|
).defaultProfile.rootDir;
|
|
if (
|
|
!sourceProfileDir ||
|
|
!sourceProfileDir.exists() ||
|
|
!sourceProfileDir.isReadable()
|
|
) {
|
|
return null;
|
|
}
|
|
|
|
// Being a startup-only migrator, we can rely on
|
|
// MigrationUtils.profileStartup being set.
|
|
let currentProfileDir = MigrationUtils.profileStartup.directory;
|
|
|
|
// Surely data cannot be imported from the current profile.
|
|
if (sourceProfileDir.equals(currentProfileDir)) {
|
|
return null;
|
|
}
|
|
|
|
return this._getResourcesInternal(sourceProfileDir, currentProfileDir);
|
|
}
|
|
|
|
getLastUsedDate() {
|
|
// We always pretend we're really old, so that we don't mess
|
|
// up the determination of which browser is the most 'recent'
|
|
// to import from.
|
|
return Promise.resolve(new Date(0));
|
|
}
|
|
|
|
_getResourcesInternal(sourceProfileDir, currentProfileDir) {
|
|
let getFileResource = (aMigrationType, aFileNames) => {
|
|
let files = [];
|
|
for (let fileName of aFileNames) {
|
|
let file = this._getFileObject(sourceProfileDir, fileName);
|
|
if (file) {
|
|
files.push(file);
|
|
}
|
|
}
|
|
if (!files.length) {
|
|
return null;
|
|
}
|
|
return {
|
|
type: aMigrationType,
|
|
migrate(aCallback) {
|
|
for (let file of files) {
|
|
file.copyTo(currentProfileDir, "");
|
|
}
|
|
aCallback(true);
|
|
},
|
|
};
|
|
};
|
|
|
|
let _oldRawPrefsMemoized = null;
|
|
async function readOldPrefs() {
|
|
if (!_oldRawPrefsMemoized) {
|
|
let prefsPath = PathUtils.join(sourceProfileDir.path, "prefs.js");
|
|
if (await IOUtils.exists(prefsPath)) {
|
|
_oldRawPrefsMemoized = await IOUtils.readUTF8(prefsPath, {
|
|
encoding: "utf-8",
|
|
});
|
|
}
|
|
}
|
|
|
|
return _oldRawPrefsMemoized;
|
|
}
|
|
|
|
function savePrefs() {
|
|
// If we've used the pref service to write prefs for the new profile, it's too
|
|
// early in startup for the service to have a profile directory, so we have to
|
|
// manually tell it where to save the prefs file.
|
|
let newPrefsFile = currentProfileDir.clone();
|
|
newPrefsFile.append("prefs.js");
|
|
Services.prefs.savePrefFile(newPrefsFile);
|
|
}
|
|
|
|
function configureHomepage(resetSession) {
|
|
// We just refreshed the profile, so don't show the profile reset prompt
|
|
// on the homepage.
|
|
Services.prefs.setBoolPref("browser.disableResetPrompt", true);
|
|
if (resetSession) {
|
|
// We're resetting the user's session, not creating a new one. Set the
|
|
// homepage_override prefs so that the browser doesn't override our
|
|
// session with an unwanted homepage.
|
|
let buildID = Services.appinfo.platformBuildID;
|
|
let mstone = Services.appinfo.platformVersion;
|
|
Services.prefs.setCharPref(
|
|
"browser.startup.homepage_override.mstone",
|
|
mstone
|
|
);
|
|
Services.prefs.setCharPref(
|
|
"browser.startup.homepage_override.buildID",
|
|
buildID
|
|
);
|
|
}
|
|
}
|
|
|
|
let types = MigrationUtils.resourceTypes;
|
|
let places = getFileResource(types.HISTORY, [
|
|
"places.sqlite",
|
|
"places.sqlite-wal",
|
|
]);
|
|
let favicons = getFileResource(types.HISTORY, [
|
|
"favicons.sqlite",
|
|
"favicons.sqlite-wal",
|
|
]);
|
|
let cookies = getFileResource(types.COOKIES, [
|
|
"cookies.sqlite",
|
|
"cookies.sqlite-wal",
|
|
]);
|
|
let passwords = getFileResource(types.PASSWORDS, [
|
|
"logins.json",
|
|
"key3.db",
|
|
"key4.db",
|
|
]);
|
|
let formData = getFileResource(types.FORMDATA, [
|
|
"formhistory.sqlite",
|
|
"autofill-profiles.json",
|
|
]);
|
|
let bookmarksBackups = getFileResource(types.OTHERDATA, [
|
|
lazy.PlacesBackups.profileRelativeFolderPath,
|
|
]);
|
|
let dictionary = getFileResource(types.OTHERDATA, ["persdict.dat"]);
|
|
|
|
// Determine if we want to restore the previous session or start a new one
|
|
const NEW_SESSION = "0";
|
|
const RESTORE_SESSION = "1";
|
|
let resetSession = Services.env.get("MOZ_RESET_PROFILE_SESSION");
|
|
Services.env.set("MOZ_RESET_PROFILE_SESSION", "");
|
|
|
|
let session;
|
|
if (resetSession === RESTORE_SESSION) {
|
|
// We only want to restore the previous firefox session if the profile
|
|
// refresh was triggered by the user, such as through about:support. In
|
|
// these cases, MOZ_RESET_PROFILE_SESSION is set to restore, signaling
|
|
// that session data migration is required.
|
|
let sessionCheckpoints = this._getFileObject(
|
|
sourceProfileDir,
|
|
"sessionCheckpoints.json"
|
|
);
|
|
let sessionFile = this._getFileObject(
|
|
sourceProfileDir,
|
|
"sessionstore.jsonlz4"
|
|
);
|
|
if (sessionFile) {
|
|
session = {
|
|
type: types.SESSION,
|
|
migrate(aCallback) {
|
|
sessionCheckpoints.copyTo(
|
|
currentProfileDir,
|
|
"sessionCheckpoints.json"
|
|
);
|
|
let newSessionFile = currentProfileDir.clone();
|
|
newSessionFile.append("sessionstore.jsonlz4");
|
|
let migrationPromise = lazy.SessionMigration.migrate(
|
|
sessionFile.path,
|
|
newSessionFile.path
|
|
);
|
|
migrationPromise.then(
|
|
function () {
|
|
// Force the browser to one-off resume the session that we give it:
|
|
Services.prefs.setBoolPref(
|
|
"browser.sessionstore.resume_session_once",
|
|
true
|
|
);
|
|
configureHomepage(true);
|
|
savePrefs();
|
|
aCallback(true);
|
|
},
|
|
function () {
|
|
aCallback(false);
|
|
}
|
|
);
|
|
},
|
|
};
|
|
}
|
|
} else if (resetSession === NEW_SESSION) {
|
|
// If this is first startup and the profile refresh was triggered via the
|
|
// command line, such as through the stub installer, we do not restore the
|
|
// previous session.
|
|
configureHomepage();
|
|
savePrefs();
|
|
}
|
|
|
|
// Sync/FxA related data
|
|
let sync = {
|
|
name: "sync", // name is used only by tests.
|
|
type: types.OTHERDATA,
|
|
migrate: async aCallback => {
|
|
// Try and parse a signedInUser.json file from the source directory and
|
|
// if we can, copy it to the new profile and set sync's username pref
|
|
// (which acts as a de-facto flag to indicate if sync is configured)
|
|
try {
|
|
let oldPath = PathUtils.join(
|
|
sourceProfileDir.path,
|
|
"signedInUser.json"
|
|
);
|
|
let exists = await IOUtils.exists(oldPath);
|
|
if (exists) {
|
|
let data = await IOUtils.readJSON(oldPath);
|
|
if (data && data.accountData && data.accountData.email) {
|
|
let username = data.accountData.email;
|
|
// copy the file itself.
|
|
await IOUtils.copy(
|
|
oldPath,
|
|
PathUtils.join(currentProfileDir.path, "signedInUser.json")
|
|
);
|
|
// Now we need to know whether Sync is actually configured for this
|
|
// user. The only way we know is by looking at the prefs file from
|
|
// the old profile. We avoid trying to do a full parse of the prefs
|
|
// file and even avoid parsing the single string value we care
|
|
// about.
|
|
let oldRawPrefs = await readOldPrefs();
|
|
if (/^user_pref\("services\.sync\.username"/m.test(oldRawPrefs)) {
|
|
// sync's configured in the source profile - ensure it is in the
|
|
// new profile too.
|
|
// Write it to prefs.js and flush the file.
|
|
Services.prefs.setStringPref(
|
|
"services.sync.username",
|
|
username
|
|
);
|
|
savePrefs();
|
|
}
|
|
}
|
|
}
|
|
} catch (ex) {
|
|
aCallback(false);
|
|
return;
|
|
}
|
|
aCallback(true);
|
|
},
|
|
};
|
|
|
|
// Telemetry related migrations.
|
|
let times = {
|
|
name: "times", // name is used only by tests.
|
|
type: types.OTHERDATA,
|
|
migrate: aCallback => {
|
|
let file = this._getFileObject(sourceProfileDir, "times.json");
|
|
if (file) {
|
|
file.copyTo(currentProfileDir, "");
|
|
}
|
|
// And record the fact a migration (ie, a reset) happened.
|
|
let recordMigration = async () => {
|
|
try {
|
|
let profileTimes = await lazy.ProfileAge(currentProfileDir.path);
|
|
await profileTimes.recordProfileReset();
|
|
aCallback(true);
|
|
} catch (e) {
|
|
aCallback(false);
|
|
}
|
|
};
|
|
|
|
recordMigration();
|
|
},
|
|
};
|
|
let telemetry = {
|
|
name: "telemetry", // name is used only by tests...
|
|
type: types.OTHERDATA,
|
|
migrate: async aCallback => {
|
|
let createSubDir = name => {
|
|
let dir = currentProfileDir.clone();
|
|
dir.append(name);
|
|
dir.create(Ci.nsIFile.DIRECTORY_TYPE, lazy.FileUtils.PERMS_DIRECTORY);
|
|
return dir;
|
|
};
|
|
|
|
// If the 'datareporting' directory exists we migrate files from it.
|
|
let dataReportingDir = this._getFileObject(
|
|
sourceProfileDir,
|
|
"datareporting"
|
|
);
|
|
if (dataReportingDir && dataReportingDir.isDirectory()) {
|
|
// Copy only specific files.
|
|
let toCopy = ["state.json", "session-state.json"];
|
|
|
|
let dest = createSubDir("datareporting");
|
|
let enumerator = dataReportingDir.directoryEntries;
|
|
while (enumerator.hasMoreElements()) {
|
|
let file = enumerator.nextFile;
|
|
if (file.isDirectory() || !toCopy.includes(file.leafName)) {
|
|
continue;
|
|
}
|
|
file.copyTo(dest, "");
|
|
}
|
|
}
|
|
|
|
try {
|
|
let oldRawPrefs = await readOldPrefs();
|
|
let writePrefs = false;
|
|
const PREFS = ["bookmarks", "csvpasswords", "history", "passwords"];
|
|
|
|
for (let pref of PREFS) {
|
|
let fullPref = `browser\.migrate\.interactions\.${pref}`;
|
|
let regex = new RegExp('^user_pref\\("' + fullPref, "m");
|
|
if (regex.test(oldRawPrefs)) {
|
|
Services.prefs.setBoolPref(fullPref, true);
|
|
writePrefs = true;
|
|
}
|
|
}
|
|
|
|
if (writePrefs) {
|
|
savePrefs();
|
|
}
|
|
} catch (e) {
|
|
aCallback(false);
|
|
return;
|
|
}
|
|
|
|
aCallback(true);
|
|
},
|
|
};
|
|
|
|
return [
|
|
places,
|
|
cookies,
|
|
passwords,
|
|
formData,
|
|
dictionary,
|
|
bookmarksBackups,
|
|
session,
|
|
sync,
|
|
times,
|
|
telemetry,
|
|
favicons,
|
|
].filter(r => r);
|
|
}
|
|
|
|
get startupOnlyMigrator() {
|
|
return true;
|
|
}
|
|
}
|