[tor-commits] [tor-messenger-build/master] Add patch for #14631
arlo at torproject.org
arlo at torproject.org
Thu Jul 28 23:52:35 UTC 2016
commit d4f35015683bedf07194f3982e080641eb0ff27f
Author: Sukhbir Singh <sukhbir at torproject.org>
Date: Mon Jul 18 12:40:15 2016 -0400
Add patch for #14631
---
...Improve-profile-access-bug-14631-first.mozpatch | 246 +++++++++++++++++++++
...mprove-profile-access-bug-14631-second.mozpatch | 182 +++++++++++++++
2 files changed, 428 insertions(+)
diff --git a/projects/instantbird/Improve-profile-access-bug-14631-first.mozpatch b/projects/instantbird/Improve-profile-access-bug-14631-first.mozpatch
new file mode 100644
index 0000000..7a9ad68
--- /dev/null
+++ b/projects/instantbird/Improve-profile-access-bug-14631-first.mozpatch
@@ -0,0 +1,246 @@
+commit 5aae044b37579dc1c95b195084bcfcdaed352545
+Author: Kathy Brade <brade at pearlcrescent.com>
+Date: Tue Feb 24 13:50:23 2015 -0500
+
+ Bug 14631: Improve profile access error messages.
+
+ Instead of always reporting that the profile is locked, display specific
+ messages for "access denied" and "read-only file system".
+
+diff --git a/toolkit/locales/en-US/chrome/mozapps/profile/profileSelection.properties b/toolkit/locales/en-US/chrome/mozapps/profile/profileSelection.properties
+index adac95a..3cf48ff 100644
+--- a/toolkit/locales/en-US/chrome/mozapps/profile/profileSelection.properties
++++ b/toolkit/locales/en-US/chrome/mozapps/profile/profileSelection.properties
+@@ -12,6 +12,11 @@ restartMessageUnlocker=%S is already running, but is not responding. The old %S
+ restartMessageNoUnlockerMac=A copy of %S is already open. Only one copy of %S can be open at a time.
+ restartMessageUnlockerMac=A copy of %S is already open. The running copy of %S will quit in order to open this one.
+
++# LOCALIZATION NOTE (profileProblemTitle, profileReadOnly, profileReadOnlyMac, profileAccessDenied): Messages displayed when the browser profile cannot be accessed or written to. %S is the application name.
++profileProblemTitle=%S Profile Problem
++profileReadOnly=You cannot run %S from a read-only file system. Please copy %S to another location before trying to use it.
++profileReadOnlyMac=You cannot run %S from a read-only file system. Please copy %S to your Desktop or Applications folder before trying to use it.
++profileAccessDenied=%S does not have permission to access the profile. Please adjust your file system permissions and try again.
+ # Profile manager
+ # LOCALIZATION NOTE (profileTooltip): First %S is the profile name, second %S is the path to the profile folder.
+ profileTooltip=Profile: '%S' - Path: '%S'
+diff --git a/toolkit/xre/nsAppRunner.cpp b/toolkit/xre/nsAppRunner.cpp
+index da73cd5..d82f303 100644
+--- a/toolkit/xre/nsAppRunner.cpp
++++ b/toolkit/xre/nsAppRunner.cpp
+@@ -1913,6 +1913,14 @@ static nsresult LaunchChild(nsINativeAppSupport* aNative,
+ return NS_ERROR_LAUNCHED_CHILD_PROCESS;
+ }
+
++enum ProfileStatus {
++ PROFILE_STATUS_OK,
++ PROFILE_STATUS_ACCESS_DENIED,
++ PROFILE_STATUS_READ_ONLY,
++ PROFILE_STATUS_IS_LOCKED,
++ PROFILE_STATUS_OTHER_ERROR
++};
++
+ static const char kProfileProperties[] =
+ "chrome://mozapps/locale/profile/profileSelection.properties";
+
+@@ -1951,9 +1959,9 @@ private:
+ } // namespace
+
+ static ReturnAbortOnError
+-ProfileLockedDialog(nsIFile* aProfileDir, nsIFile* aProfileLocalDir,
+- nsIProfileUnlocker* aUnlocker,
+- nsINativeAppSupport* aNative, nsIProfileLock* *aResult)
++ProfileErrorDialog(nsIFile* aProfileDir, nsIFile* aProfileLocalDir,
++ ProfileStatus aStatus, nsIProfileUnlocker* aUnlocker,
++ nsINativeAppSupport* aNative, nsIProfileLock* *aResult)
+ {
+ nsresult rv;
+
+@@ -1980,18 +1988,31 @@ ProfileLockedDialog(nsIFile* aProfileDir, nsIFile* aProfileLocalDir,
+
+ nsXPIDLString killMessage;
+ #ifndef XP_MACOSX
+- sb->FormatStringFromName(aUnlocker ? MOZ_UTF16("restartMessageUnlocker")
+- : MOZ_UTF16("restartMessageNoUnlocker"),
+- params, 2, getter_Copies(killMessage));
++ static const char16_t kRestartUnlocker[] = MOZ_UTF16("restartMessageUnlocker");
++ static const char16_t kRestartNoUnlocker[] = MOZ_UTF16("restartMessageNoUnlocker");
++ static const char16_t kReadOnly[] = MOZ_UTF16("profileReadOnly");
+ #else
+- sb->FormatStringFromName(aUnlocker ? MOZ_UTF16("restartMessageUnlockerMac")
+- : MOZ_UTF16("restartMessageNoUnlockerMac"),
+- params, 2, getter_Copies(killMessage));
++ static const char16_t kRestartUnlocker[] = MOZ_UTF16("restartMessageUnlockerMac");
++ static const char16_t kRestartNoUnlocker[] = MOZ_UTF16("restartMessageNoUnlockerMac");
++ static const char16_t kReadOnly[] = MOZ_UTF16("profileReadOnlyMac");
+ #endif
++ static const char16_t kAccessDenied[] = MOZ_UTF16("profileAccessDenied");
++
++ const char16_t *errorKey = aUnlocker ? kRestartUnlocker
++ : kRestartNoUnlocker;
++ if (PROFILE_STATUS_READ_ONLY == aStatus)
++ errorKey = kReadOnly;
++ else if (PROFILE_STATUS_ACCESS_DENIED == aStatus)
++ errorKey = kAccessDenied;
++ sb->FormatStringFromName(errorKey, params, 2, getter_Copies(killMessage));
++
++ const char16_t *titleKey = ((PROFILE_STATUS_READ_ONLY == aStatus) ||
++ (PROFILE_STATUS_ACCESS_DENIED == aStatus))
++ ? MOZ_UTF16("profileProblemTitle")
++ : MOZ_UTF16("restartTitle");
+
+ nsXPIDLString killTitle;
+- sb->FormatStringFromName(MOZ_UTF16("restartTitle"),
+- params, 1, getter_Copies(killTitle));
++ sb->FormatStringFromName(titleKey, params, 1, getter_Copies(killTitle));
+
+ if (!killMessage || !killTitle)
+ return NS_ERROR_FAILURE;
+@@ -2092,8 +2113,9 @@ ProfileMissingDialog(nsINativeAppSupport* aNative)
+ }
+
+ static nsresult
+-ProfileLockedDialog(nsIToolkitProfile* aProfile, nsIProfileUnlocker* aUnlocker,
+- nsINativeAppSupport* aNative, nsIProfileLock* *aResult)
++ProfileErrorDialog(nsIToolkitProfile* aProfile, ProfileStatus aStatus,
++ nsIProfileUnlocker* aUnlocker, nsINativeAppSupport* aNative,
++ nsIProfileLock* *aResult)
+ {
+ nsCOMPtr<nsIFile> profileDir;
+ nsresult rv = aProfile->GetRootDir(getter_AddRefs(profileDir));
+@@ -2109,8 +2131,8 @@ ProfileLockedDialog(nsIToolkitProfile* aProfile, nsIProfileUnlocker* aUnlocker,
+ rv = aProfile->GetLocalDir(getter_AddRefs(profileLocalDir));
+ if (NS_FAILED(rv)) return rv;
+
+- return ProfileLockedDialog(profileDir, profileLocalDir, aUnlocker, aNative,
+- aResult);
++ return ProfileErrorDialog(profileDir, profileLocalDir, aStatus, aUnlocker,
++ aNative, aResult);
+ }
+
+ static const char kProfileManagerURL[] =
+@@ -2275,6 +2297,53 @@ SetCurrentProfileAsDefault(nsIToolkitProfileService* aProfileSvc,
+ return rv;
+ }
+
++// Check for write permission to the profile directory by trying to create a
++// new file (after ensuring that no file with the same name exists).
++static ProfileStatus CheckProfileWriteAccess(nsIFile* aProfileDir)
++{
++#if defined(XP_UNIX)
++ NS_NAMED_LITERAL_STRING(writeTestFileName, ".parentwritetest");
++#else
++ NS_NAMED_LITERAL_STRING(writeTestFileName, "parent.writetest");
++#endif
++
++ nsCOMPtr<nsIFile> writeTestFile;
++ nsresult rv = aProfileDir->Clone(getter_AddRefs(writeTestFile));
++ if (NS_SUCCEEDED(rv))
++ rv = writeTestFile->Append(writeTestFileName);
++
++ if (NS_SUCCEEDED(rv)) {
++ bool doesExist = false;
++ rv = writeTestFile->Exists(&doesExist);
++ if (NS_SUCCEEDED(rv) && doesExist)
++ rv = writeTestFile->Remove(true);
++ }
++
++ if (NS_SUCCEEDED(rv)) {
++ rv = writeTestFile->Create(nsIFile::NORMAL_FILE_TYPE, 0666);
++ (void)writeTestFile->Remove(true);
++ }
++
++ ProfileStatus status = NS_SUCCEEDED(rv) ? PROFILE_STATUS_OK
++ : PROFILE_STATUS_OTHER_ERROR;
++ if (NS_ERROR_FILE_ACCESS_DENIED == rv)
++ status = PROFILE_STATUS_ACCESS_DENIED;
++ else if (NS_ERROR_FILE_READ_ONLY == rv)
++ status = PROFILE_STATUS_READ_ONLY;
++
++ return status;
++}
++
++static ProfileStatus CheckProfileWriteAccess(nsIToolkitProfile* aProfile)
++{
++ nsCOMPtr<nsIFile> profileDir;
++ nsresult rv = aProfile->GetRootDir(getter_AddRefs(profileDir));
++ if (NS_FAILED(rv))
++ return PROFILE_STATUS_OTHER_ERROR;
++
++ return CheckProfileWriteAccess(profileDir);
++}
++
+ static bool gDoMigration = false;
+ static bool gDoProfileReset = false;
+
+@@ -2410,13 +2479,18 @@ SelectProfile(nsIProfileLock* *aResult, nsIToolkitProfileService* aProfileSvc, n
+ NS_ENSURE_SUCCESS(rv, rv);
+ }
+
++ ProfileStatus status = CheckProfileWriteAccess(lf);
++ if (PROFILE_STATUS_OK != status)
++ return ProfileErrorDialog(lf, lf, status, nullptr, aNative, aResult);
++
+ // If a profile path is specified directory on the command line, then
+ // assume that the temp directory is the same as the given directory.
+ rv = NS_LockProfilePath(lf, lf, getter_AddRefs(unlocker), aResult);
+ if (NS_SUCCEEDED(rv))
+ return rv;
+
+- return ProfileLockedDialog(lf, lf, unlocker, aNative, aResult);
++ return ProfileErrorDialog(lf, lf, PROFILE_STATUS_IS_LOCKED, unlocker,
++ aNative, aResult);
+ }
+
+ ar = CheckArg("createprofile", true, &arg);
+@@ -2501,6 +2575,10 @@ SelectProfile(nsIProfileLock* *aResult, nsIToolkitProfileService* aProfileSvc, n
+ gDoProfileReset = false;
+ }
+
++ ProfileStatus status = CheckProfileWriteAccess(profile);
++ if (PROFILE_STATUS_OK != status)
++ return ProfileErrorDialog(profile, status, nullptr, aNative, aResult);
++
+ nsCOMPtr<nsIProfileUnlocker> unlocker;
+ rv = profile->Lock(getter_AddRefs(unlocker), aResult);
+ if (NS_SUCCEEDED(rv)) {
+@@ -2509,7 +2587,8 @@ SelectProfile(nsIProfileLock* *aResult, nsIToolkitProfileService* aProfileSvc, n
+ return NS_OK;
+ }
+
+- return ProfileLockedDialog(profile, unlocker, aNative, aResult);
++ return ProfileErrorDialog(profile, PROFILE_STATUS_IS_LOCKED, unlocker,
++ aNative, aResult);
+ }
+
+ if (CanShowProfileManager()) {
+@@ -2589,7 +2668,8 @@ SelectProfile(nsIProfileLock* *aResult, nsIToolkitProfileService* aProfileSvc, n
+ nsCOMPtr<nsIProfileUnlocker> unlocker;
+ rv = profile->Lock(getter_AddRefs(unlocker), &tempProfileLock);
+ if (NS_FAILED(rv))
+- return ProfileLockedDialog(profile, unlocker, aNative, &tempProfileLock);
++ return ProfileErrorDialog(profile, PROFILE_STATUS_IS_LOCKED,
++ unlocker, aNative, &tempProfileLock);
+ }
+
+ nsCOMPtr<nsIToolkitProfile> newProfile;
+@@ -2600,6 +2680,10 @@ SelectProfile(nsIProfileLock* *aResult, nsIToolkitProfileService* aProfileSvc, n
+ gDoProfileReset = false;
+ }
+
++ ProfileStatus status = CheckProfileWriteAccess(profile);
++ if (PROFILE_STATUS_OK != status)
++ return ProfileErrorDialog(profile, status, nullptr, aNative, aResult);
++
+ // If you close Firefox and very quickly reopen it, the old Firefox may
+ // still be closing down. Rather than immediately showing the
+ // "Firefox is running but is not responding" message, we spend a few
+@@ -2625,7 +2709,8 @@ SelectProfile(nsIProfileLock* *aResult, nsIToolkitProfileService* aProfileSvc, n
+ PR_Sleep(kLockRetrySleepMS);
+ } while (TimeStamp::Now() - start < TimeDuration::FromSeconds(kLockRetrySeconds));
+
+- return ProfileLockedDialog(profile, unlocker, aNative, aResult);
++ return ProfileErrorDialog(profile, PROFILE_STATUS_IS_LOCKED, unlocker,
++ aNative, aResult);
+ }
+ }
+
diff --git a/projects/instantbird/Improve-profile-access-bug-14631-second.mozpatch b/projects/instantbird/Improve-profile-access-bug-14631-second.mozpatch
new file mode 100644
index 0000000..3e59cd5
--- /dev/null
+++ b/projects/instantbird/Improve-profile-access-bug-14631-second.mozpatch
@@ -0,0 +1,182 @@
+commit 303992b0684036f6f23d1ca7f76b360930f510db
+Author: Kathy Brade <brade at pearlcrescent.com>
+Date: Fri Feb 27 10:38:40 2015 -0500
+
+ Bug 14631: Improve profile access error msgs (strings).
+
+ To allow for localization, get profile-related error strings from Torbutton.
+ Use app display name ("Tor Browser") in profile-related error alerts.
+
+diff --git a/toolkit/xre/moz.build b/toolkit/xre/moz.build
+index d8127a6..3e6a3b6 100644
+--- a/toolkit/xre/moz.build
++++ b/toolkit/xre/moz.build
+@@ -116,8 +116,8 @@ FINAL_LIBRARY = 'xul'
+ if CONFIG['MOZ_GL_DEFAULT_PROVIDER'] == 'GLX':
+ DEFINES['USE_GLX_TEST'] = True
+
+-for var in ('MOZ_APP_NAME', 'MOZ_APP_BASENAME', 'MOZ_APP_VERSION', 'OS_TARGET',
+- 'MOZ_WIDGET_TOOLKIT'):
++for var in ('MOZ_APP_NAME', 'MOZ_APP_BASENAME', 'MOZ_APP_DISPLAYNAME',
++ 'MOZ_APP_VERSION', 'OS_TARGET', 'MOZ_WIDGET_TOOLKIT'):
+ DEFINES[var] = '"%s"' % CONFIG[var]
+
+ if CONFIG['MOZ_UPDATER'] and CONFIG['MOZ_WIDGET_TOOLKIT'] != 'android':
+diff --git a/toolkit/xre/nsAppRunner.cpp b/toolkit/xre/nsAppRunner.cpp
+index d82f303..a7c7d75 100644
+--- a/toolkit/xre/nsAppRunner.cpp
++++ b/toolkit/xre/nsAppRunner.cpp
+@@ -1913,6 +1913,104 @@ static nsresult LaunchChild(nsINativeAppSupport* aNative,
+ return NS_ERROR_LAUNCHED_CHILD_PROCESS;
+ }
+
++static nsresult
++GetOverrideStringBundleForLocale(nsIStringBundleService* aSBS,
++ const char* aTorbuttonURI, const char* aLocale,
++ nsIStringBundle* *aResult)
++{
++ NS_ENSURE_ARG(aSBS);
++ NS_ENSURE_ARG(aTorbuttonURI);
++ NS_ENSURE_ARG(aLocale);
++ NS_ENSURE_ARG(aResult);
++
++ const char* kFormatStr = "jar:%s!/chrome/locale/%s/torbutton.properties";
++ nsPrintfCString strBundleURL(kFormatStr, aTorbuttonURI, aLocale);
++ nsresult rv = aSBS->CreateBundle(strBundleURL.get(), aResult);
++ NS_ENSURE_SUCCESS(rv, rv);
++
++ // To ensure that we have a valid string bundle, try to retrieve a string
++ // that we know exists.
++ nsXPIDLString val;
++ rv = (*aResult)->GetStringFromName(MOZ_UTF16("profileProblemTitle"),
++ getter_Copies(val));
++ if (!NS_SUCCEEDED(rv))
++ *aResult = nullptr; // No good. Discard it.
++
++ return rv;
++}
++
++static void
++GetOverrideStringBundle(nsIStringBundleService* aSBS, nsIStringBundle* *aResult)
++{
++ if (!aSBS || !aResult)
++ return;
++
++ *aResult = nullptr;
++
++ // Build Torbutton file URI string by starting from the profiles directory.
++ nsXREDirProvider* dirProvider = nsXREDirProvider::GetSingleton();
++ if (!dirProvider)
++ return;
++
++ bool persistent = false; // ignored
++ nsCOMPtr<nsIFile> profilesDir;
++ nsresult rv = dirProvider->GetFile(NS_APP_USER_PROFILES_ROOT_DIR, &persistent,
++ getter_AddRefs(profilesDir));
++ if (NS_FAILED(rv))
++ return;
++
++ // Create file URI, extract as string, and append Torbutton xpi relative path.
++ nsCOMPtr<nsIURI> uri;
++ nsAutoCString uriString;
++ if (NS_FAILED(NS_NewFileURI(getter_AddRefs(uri), profilesDir)) ||
++ NS_FAILED(uri->GetSpec(uriString))) {
++ return;
++ }
++
++ uriString.Append("profile.default/extensions/torbutton at torproject.org.xpi");
++
++ nsCString userAgentLocale;
++ if (!NS_SUCCEEDED(Preferences::GetCString("general.useragent.locale",
++ &userAgentLocale))) {
++ return;
++ }
++
++ rv = GetOverrideStringBundleForLocale(aSBS, uriString.get(),
++ userAgentLocale.get(), aResult);
++ if (NS_FAILED(rv)) {
++ // Try again using base locale, e.g., "en" vs. "en-US".
++ int16_t offset = userAgentLocale.FindChar('-', 1);
++ if (offset > 0) {
++ nsAutoCString shortLocale(Substring(userAgentLocale, 0, offset));
++ rv = GetOverrideStringBundleForLocale(aSBS, uriString.get(),
++ shortLocale.get(), aResult);
++ }
++ }
++}
++
++static nsresult
++GetFormattedString(nsIStringBundle* aOverrideBundle,
++ nsIStringBundle* aMainBundle,
++ const char16_t* aName,
++ const char16_t** aParams, uint32_t aLength,
++ char16_t* *aResult)
++{
++ NS_ENSURE_ARG(aName);
++ NS_ENSURE_ARG(aResult);
++
++ nsresult rv = NS_ERROR_FAILURE;
++ if (aOverrideBundle) {
++ rv = aOverrideBundle->FormatStringFromName(aName, aParams, aLength,
++ aResult);
++ }
++
++ // If string was not found in override bundle, use main (browser) bundle.
++ if (NS_FAILED(rv) && aMainBundle)
++ rv = aMainBundle->FormatStringFromName(aName, aParams, aLength, aResult);
++
++ return rv;
++}
++
+ enum ProfileStatus {
+ PROFILE_STATUS_OK,
+ PROFILE_STATUS_ACCESS_DENIED,
+@@ -1983,7 +2081,10 @@ ProfileErrorDialog(nsIFile* aProfileDir, nsIFile* aProfileLocalDir,
+ sbs->CreateBundle(kProfileProperties, getter_AddRefs(sb));
+ NS_ENSURE_TRUE_LOG(sbs, NS_ERROR_FAILURE);
+
+- NS_ConvertUTF8toUTF16 appName(gAppData->name);
++ nsCOMPtr<nsIStringBundle> overrideSB;
++ GetOverrideStringBundle(sbs, getter_AddRefs(overrideSB));
++
++ NS_ConvertUTF8toUTF16 appName(MOZ_APP_DISPLAYNAME);
+ const char16_t* params[] = {appName.get(), appName.get()};
+
+ nsXPIDLString killMessage;
+@@ -1998,21 +2099,23 @@ ProfileErrorDialog(nsIFile* aProfileDir, nsIFile* aProfileLocalDir,
+ #endif
+ static const char16_t kAccessDenied[] = MOZ_UTF16("profileAccessDenied");
+
+- const char16_t *errorKey = aUnlocker ? kRestartUnlocker
++ const char16_t* errorKey = aUnlocker ? kRestartUnlocker
+ : kRestartNoUnlocker;
+ if (PROFILE_STATUS_READ_ONLY == aStatus)
+ errorKey = kReadOnly;
+ else if (PROFILE_STATUS_ACCESS_DENIED == aStatus)
+ errorKey = kAccessDenied;
+- sb->FormatStringFromName(errorKey, params, 2, getter_Copies(killMessage));
++ GetFormattedString(overrideSB, sb, errorKey, params, 2,
++ getter_Copies(killMessage));
+
+- const char16_t *titleKey = ((PROFILE_STATUS_READ_ONLY == aStatus) ||
++ const char16_t* titleKey = ((PROFILE_STATUS_READ_ONLY == aStatus) ||
+ (PROFILE_STATUS_ACCESS_DENIED == aStatus))
+ ? MOZ_UTF16("profileProblemTitle")
+ : MOZ_UTF16("restartTitle");
+
+ nsXPIDLString killTitle;
+- sb->FormatStringFromName(titleKey, params, 1, getter_Copies(killTitle));
++ GetFormattedString(overrideSB, sb, titleKey, params, 1,
++ getter_Copies(killTitle));
+
+ if (!killMessage || !killTitle)
+ return NS_ERROR_FAILURE;
+@@ -2088,7 +2191,7 @@ ProfileMissingDialog(nsINativeAppSupport* aNative)
+ sbs->CreateBundle(kProfileProperties, getter_AddRefs(sb));
+ NS_ENSURE_TRUE_LOG(sbs, NS_ERROR_FAILURE);
+
+- NS_ConvertUTF8toUTF16 appName(gAppData->name);
++ NS_ConvertUTF8toUTF16 appName(MOZ_APP_DISPLAYNAME);
+ const char16_t* params[] = {appName.get(), appName.get()};
+
+ nsXPIDLString missingMessage;
More information about the tor-commits
mailing list