2019-01-19 01:52:06 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
|
|
/* 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/. */
|
|
|
|
|
|
|
|
#include "QuotaCommon.h"
|
|
|
|
|
2020-12-04 17:56:18 +03:00
|
|
|
#include "mozIStorageConnection.h"
|
|
|
|
#include "mozIStorageStatement.h"
|
2020-12-21 12:32:12 +03:00
|
|
|
#include "mozilla/ErrorNames.h"
|
2020-10-21 16:16:19 +03:00
|
|
|
#include "mozilla/Logging.h"
|
2020-12-09 23:01:07 +03:00
|
|
|
#include "mozilla/Telemetry.h"
|
|
|
|
#include "mozilla/TelemetryComms.h"
|
|
|
|
#include "mozilla/TelemetryEventEnums.h"
|
2020-10-21 16:16:19 +03:00
|
|
|
#include "mozilla/TextUtils.h"
|
2020-09-08 10:40:46 +03:00
|
|
|
#include "nsIConsoleService.h"
|
2020-04-01 10:26:27 +03:00
|
|
|
#include "nsIFile.h"
|
2020-10-21 16:16:19 +03:00
|
|
|
#include "nsServiceManagerUtils.h"
|
|
|
|
#include "nsStringFlags.h"
|
|
|
|
#include "nsTStringRepr.h"
|
|
|
|
#include "nsUnicharUtils.h"
|
2020-04-06 12:28:35 +03:00
|
|
|
#include "nsXPCOM.h"
|
|
|
|
#include "nsXULAppAPI.h"
|
2020-10-21 16:16:19 +03:00
|
|
|
|
2020-04-01 10:26:27 +03:00
|
|
|
#ifdef XP_WIN
|
2020-10-21 16:16:19 +03:00
|
|
|
# include "mozilla/Atomics.h"
|
2020-04-06 12:28:35 +03:00
|
|
|
# include "mozilla/ipc/BackgroundParent.h"
|
|
|
|
# include "mozilla/StaticPrefs_dom.h"
|
2020-04-01 10:26:27 +03:00
|
|
|
# include "nsILocalFileWin.h"
|
|
|
|
#endif
|
|
|
|
|
2020-11-04 20:04:01 +03:00
|
|
|
namespace mozilla::dom::quota {
|
2019-03-21 23:08:00 +03:00
|
|
|
|
2020-12-09 23:01:07 +03:00
|
|
|
using namespace mozilla::Telemetry;
|
|
|
|
|
2019-03-21 23:08:00 +03:00
|
|
|
namespace {
|
|
|
|
|
2020-06-25 10:16:47 +03:00
|
|
|
#ifdef DEBUG
|
|
|
|
constexpr auto kDSStoreFileName = u".DS_Store"_ns;
|
|
|
|
constexpr auto kDesktopFileName = u".desktop"_ns;
|
|
|
|
constexpr auto kDesktopIniFileName = u"desktop.ini"_ns;
|
|
|
|
constexpr auto kThumbsDbFileName = u"thumbs.db"_ns;
|
|
|
|
#endif
|
|
|
|
|
2020-04-06 12:28:35 +03:00
|
|
|
#ifdef XP_WIN
|
|
|
|
Atomic<int32_t> gUseDOSDevicePathSyntax(-1);
|
|
|
|
#endif
|
|
|
|
|
2019-03-21 23:08:00 +03:00
|
|
|
LazyLogModule gLogger("QuotaManager");
|
|
|
|
|
2019-10-02 07:28:23 +03:00
|
|
|
void AnonymizeCString(nsACString& aCString, uint32_t aStart) {
|
|
|
|
MOZ_ASSERT(!aCString.IsEmpty());
|
|
|
|
MOZ_ASSERT(aStart < aCString.Length());
|
2019-01-19 01:52:06 +03:00
|
|
|
|
2019-10-02 07:28:23 +03:00
|
|
|
char* iter = aCString.BeginWriting() + aStart;
|
|
|
|
char* end = aCString.EndWriting();
|
2019-06-09 22:46:34 +03:00
|
|
|
|
|
|
|
while (iter != end) {
|
|
|
|
char c = *iter;
|
|
|
|
|
|
|
|
if (IsAsciiAlpha(c)) {
|
|
|
|
*iter = 'a';
|
|
|
|
} else if (IsAsciiDigit(c)) {
|
|
|
|
*iter = 'D';
|
|
|
|
}
|
|
|
|
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-02 07:28:23 +03:00
|
|
|
} // namespace
|
2019-06-09 22:46:34 +03:00
|
|
|
|
2019-10-02 07:28:23 +03:00
|
|
|
const char kQuotaGenericDelimiter = '|';
|
2019-06-09 22:46:34 +03:00
|
|
|
|
2019-10-02 07:28:23 +03:00
|
|
|
#ifdef NIGHTLY_BUILD
|
|
|
|
const nsLiteralCString kQuotaInternalError = "internal"_ns;
|
|
|
|
const nsLiteralCString kQuotaExternalError = "external"_ns;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
LogModule* GetQuotaManagerLogger() { return gLogger; }
|
|
|
|
|
|
|
|
void AnonymizeCString(nsACString& aCString) {
|
|
|
|
if (aCString.IsEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
AnonymizeCString(aCString, /* aStart */ 0);
|
|
|
|
}
|
2019-06-09 22:46:34 +03:00
|
|
|
|
2019-10-02 07:28:23 +03:00
|
|
|
void AnonymizeOriginString(nsACString& aOriginString) {
|
|
|
|
if (aOriginString.IsEmpty()) {
|
|
|
|
return;
|
2019-06-09 22:46:34 +03:00
|
|
|
}
|
2019-10-02 07:28:23 +03:00
|
|
|
|
|
|
|
int32_t start = aOriginString.FindChar(':');
|
|
|
|
if (start < 0) {
|
|
|
|
start = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
AnonymizeCString(aOriginString, start);
|
2019-06-09 22:46:34 +03:00
|
|
|
}
|
|
|
|
|
2020-04-06 12:28:35 +03:00
|
|
|
#ifdef XP_WIN
|
|
|
|
void CacheUseDOSDevicePathSyntaxPrefValue() {
|
|
|
|
MOZ_ASSERT(XRE_IsParentProcess());
|
|
|
|
AssertIsOnBackgroundThread();
|
|
|
|
|
|
|
|
if (gUseDOSDevicePathSyntax == -1) {
|
|
|
|
bool useDOSDevicePathSyntax =
|
|
|
|
StaticPrefs::dom_quotaManager_useDOSDevicePathSyntax_DoNotUseDirectly();
|
|
|
|
gUseDOSDevicePathSyntax = useDOSDevicePathSyntax ? 1 : 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-04-01 10:26:27 +03:00
|
|
|
Result<nsCOMPtr<nsIFile>, nsresult> QM_NewLocalFile(const nsAString& aPath) {
|
2020-11-17 14:37:15 +03:00
|
|
|
QM_TRY_UNWRAP(auto file,
|
|
|
|
ToResultInvoke<nsCOMPtr<nsIFile>>(NS_NewLocalFile, aPath,
|
|
|
|
/* aFollowLinks */ false),
|
|
|
|
QM_PROPAGATE, [&aPath](const nsresult rv) {
|
|
|
|
QM_WARNING("Failed to construct a file for path (%s)",
|
|
|
|
NS_ConvertUTF16toUTF8(aPath).get());
|
|
|
|
});
|
2020-04-01 10:26:27 +03:00
|
|
|
|
|
|
|
#ifdef XP_WIN
|
2020-04-06 12:28:35 +03:00
|
|
|
MOZ_ASSERT(gUseDOSDevicePathSyntax != -1);
|
2020-04-01 10:26:27 +03:00
|
|
|
|
2020-04-06 12:28:35 +03:00
|
|
|
if (gUseDOSDevicePathSyntax) {
|
2020-11-17 14:37:15 +03:00
|
|
|
QM_TRY_INSPECT(const auto& winFile,
|
|
|
|
ToResultGet<nsCOMPtr<nsILocalFileWin>>(
|
|
|
|
MOZ_SELECT_OVERLOAD(do_QueryInterface), file));
|
2020-04-06 12:28:35 +03:00
|
|
|
|
|
|
|
MOZ_ASSERT(winFile);
|
|
|
|
winFile->SetUseDOSDevicePathSyntax(true);
|
|
|
|
}
|
2020-04-01 10:26:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
2020-09-08 10:40:46 +03:00
|
|
|
nsDependentCSubstring GetLeafName(const nsACString& aPath) {
|
|
|
|
nsACString::const_iterator start, end;
|
|
|
|
aPath.BeginReading(start);
|
|
|
|
aPath.EndReading(end);
|
|
|
|
|
|
|
|
bool found = RFindInReadable("/"_ns, start, end);
|
|
|
|
if (found) {
|
|
|
|
start = end;
|
|
|
|
}
|
|
|
|
|
|
|
|
aPath.EndReading(end);
|
|
|
|
|
|
|
|
return nsDependentCSubstring(start.get(), end.get());
|
|
|
|
}
|
|
|
|
|
2020-10-26 17:51:32 +03:00
|
|
|
Result<nsCOMPtr<nsIFile>, nsresult> CloneFileAndAppend(
|
|
|
|
nsIFile& aDirectory, const nsAString& aPathElement) {
|
|
|
|
QM_TRY_UNWRAP(auto resultFile, MOZ_TO_RESULT_INVOKE_TYPED(nsCOMPtr<nsIFile>,
|
|
|
|
aDirectory, Clone));
|
|
|
|
|
|
|
|
QM_TRY(resultFile->Append(aPathElement));
|
|
|
|
|
|
|
|
return resultFile;
|
|
|
|
}
|
|
|
|
|
2021-01-21 17:38:30 +03:00
|
|
|
Result<nsIFileKind, nsresult> GetDirEntryKind(nsIFile& aFile) {
|
|
|
|
QM_TRY_RETURN(
|
|
|
|
MOZ_TO_RESULT_INVOKE(aFile, IsDirectory)
|
|
|
|
.map([](const bool isDirectory) {
|
|
|
|
return isDirectory ? nsIFileKind::ExistsAsDirectory
|
|
|
|
: nsIFileKind::ExistsAsFile;
|
|
|
|
})
|
|
|
|
.orElse([](const nsresult rv) -> Result<nsIFileKind, nsresult> {
|
|
|
|
if (rv == NS_ERROR_FILE_NOT_FOUND ||
|
2021-02-04 15:14:45 +03:00
|
|
|
rv == NS_ERROR_FILE_TARGET_DOES_NOT_EXIST
|
|
|
|
#ifdef WIN32
|
|
|
|
// We treat ERROR_FILE_CORRUPT as if the file did not exist at
|
|
|
|
// all.
|
|
|
|
|| (NS_ERROR_GET_MODULE(rv) == NS_ERROR_MODULE_WIN32 &&
|
|
|
|
NS_ERROR_GET_CODE(rv) == ERROR_FILE_CORRUPT)
|
|
|
|
#endif
|
|
|
|
) {
|
2021-01-21 17:38:30 +03:00
|
|
|
return nsIFileKind::DoesNotExist;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Err(rv);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2020-12-04 18:15:31 +03:00
|
|
|
Result<nsCOMPtr<mozIStorageStatement>, nsresult> CreateStatement(
|
|
|
|
mozIStorageConnection& aConnection, const nsACString& aStatementString) {
|
|
|
|
QM_TRY_RETURN(MOZ_TO_RESULT_INVOKE_TYPED(nsCOMPtr<mozIStorageStatement>,
|
|
|
|
aConnection, CreateStatement,
|
|
|
|
aStatementString));
|
|
|
|
}
|
|
|
|
|
|
|
|
template <SingleStepResult ResultHandling>
|
|
|
|
Result<SingleStepSuccessType<ResultHandling>, nsresult> ExecuteSingleStep(
|
|
|
|
nsCOMPtr<mozIStorageStatement>&& aStatement) {
|
|
|
|
QM_TRY_INSPECT(const bool& hasResult,
|
|
|
|
MOZ_TO_RESULT_INVOKE(aStatement, ExecuteStep));
|
|
|
|
|
|
|
|
if constexpr (ResultHandling == SingleStepResult::AssertHasResult) {
|
|
|
|
MOZ_ASSERT(hasResult);
|
|
|
|
(void)hasResult;
|
|
|
|
|
|
|
|
return WrapNotNullUnchecked(std::move(aStatement));
|
|
|
|
} else {
|
|
|
|
return hasResult ? std::move(aStatement) : nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template Result<SingleStepSuccessType<SingleStepResult::AssertHasResult>,
|
|
|
|
nsresult>
|
|
|
|
ExecuteSingleStep<SingleStepResult::AssertHasResult>(
|
|
|
|
nsCOMPtr<mozIStorageStatement>&&);
|
|
|
|
|
|
|
|
template Result<SingleStepSuccessType<SingleStepResult::ReturnNullIfNoResult>,
|
|
|
|
nsresult>
|
|
|
|
ExecuteSingleStep<SingleStepResult::ReturnNullIfNoResult>(
|
|
|
|
nsCOMPtr<mozIStorageStatement>&&);
|
|
|
|
|
2020-12-04 18:10:07 +03:00
|
|
|
template <SingleStepResult ResultHandling>
|
|
|
|
Result<SingleStepSuccessType<ResultHandling>, nsresult>
|
2020-12-04 17:56:18 +03:00
|
|
|
CreateAndExecuteSingleStepStatement(mozIStorageConnection& aConnection,
|
|
|
|
const nsACString& aStatementString) {
|
|
|
|
QM_TRY_UNWRAP(auto stmt, MOZ_TO_RESULT_INVOKE_TYPED(
|
|
|
|
nsCOMPtr<mozIStorageStatement>, aConnection,
|
|
|
|
CreateStatement, aStatementString));
|
|
|
|
|
2020-12-04 18:15:31 +03:00
|
|
|
return ExecuteSingleStep<ResultHandling>(std::move(stmt));
|
2020-12-04 17:56:18 +03:00
|
|
|
}
|
|
|
|
|
2020-12-04 18:10:07 +03:00
|
|
|
template Result<SingleStepSuccessType<SingleStepResult::AssertHasResult>,
|
|
|
|
nsresult>
|
|
|
|
CreateAndExecuteSingleStepStatement<SingleStepResult::AssertHasResult>(
|
|
|
|
mozIStorageConnection& aConnection, const nsACString& aStatementString);
|
|
|
|
|
|
|
|
template Result<SingleStepSuccessType<SingleStepResult::ReturnNullIfNoResult>,
|
|
|
|
nsresult>
|
|
|
|
CreateAndExecuteSingleStepStatement<SingleStepResult::ReturnNullIfNoResult>(
|
|
|
|
mozIStorageConnection& aConnection, const nsACString& aStatementString);
|
|
|
|
|
2020-09-30 11:24:18 +03:00
|
|
|
#ifdef QM_ENABLE_SCOPED_LOG_EXTRA_INFO
|
|
|
|
MOZ_THREAD_LOCAL(const nsACString*) ScopedLogExtraInfo::sQueryValue;
|
2020-12-09 23:01:07 +03:00
|
|
|
MOZ_THREAD_LOCAL(const nsACString*) ScopedLogExtraInfo::sContextValue;
|
2020-09-30 11:24:18 +03:00
|
|
|
|
|
|
|
/* static */
|
|
|
|
auto ScopedLogExtraInfo::FindSlot(const char* aTag) {
|
|
|
|
// XXX For now, don't use a real map but just allow the known tag values.
|
|
|
|
|
|
|
|
if (aTag == kTagQuery) {
|
|
|
|
return &sQueryValue;
|
|
|
|
}
|
|
|
|
|
2020-12-09 23:01:07 +03:00
|
|
|
if (aTag == kTagContext) {
|
|
|
|
return &sContextValue;
|
|
|
|
}
|
|
|
|
|
2020-09-30 11:24:18 +03:00
|
|
|
MOZ_CRASH("Unknown tag!");
|
|
|
|
}
|
|
|
|
|
|
|
|
ScopedLogExtraInfo::~ScopedLogExtraInfo() {
|
|
|
|
if (mTag) {
|
|
|
|
MOZ_ASSERT(&mCurrentValue == FindSlot(mTag)->get(),
|
|
|
|
"Bad scoping of ScopedLogExtraInfo, must not be interleaved!");
|
|
|
|
|
|
|
|
FindSlot(mTag)->set(mPreviousValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ScopedLogExtraInfo::ScopedLogExtraInfo(ScopedLogExtraInfo&& aOther)
|
|
|
|
: mTag(aOther.mTag),
|
|
|
|
mPreviousValue(aOther.mPreviousValue),
|
|
|
|
mCurrentValue(std::move(aOther.mCurrentValue)) {
|
|
|
|
aOther.mTag = nullptr;
|
|
|
|
FindSlot(mTag)->set(&mCurrentValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ ScopedLogExtraInfo::ScopedLogExtraInfoMap
|
|
|
|
ScopedLogExtraInfo::GetExtraInfoMap() {
|
|
|
|
// This could be done in a cheaper way, but this is never called on a hot
|
|
|
|
// path, so we anticipate using a real map inside here to make use simpler for
|
|
|
|
// the caller(s).
|
|
|
|
|
|
|
|
ScopedLogExtraInfoMap map;
|
|
|
|
if (XRE_IsParentProcess()) {
|
|
|
|
if (sQueryValue.get()) {
|
|
|
|
map.emplace(kTagQuery, sQueryValue.get());
|
|
|
|
}
|
2020-12-09 23:01:07 +03:00
|
|
|
|
|
|
|
if (sContextValue.get()) {
|
|
|
|
map.emplace(kTagContext, sContextValue.get());
|
|
|
|
}
|
2020-09-30 11:24:18 +03:00
|
|
|
}
|
|
|
|
return map;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ void ScopedLogExtraInfo::Initialize() {
|
|
|
|
MOZ_ALWAYS_TRUE(sQueryValue.init());
|
2020-12-09 23:01:07 +03:00
|
|
|
MOZ_ALWAYS_TRUE(sContextValue.init());
|
2020-09-30 11:24:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScopedLogExtraInfo::AddInfo() {
|
|
|
|
auto* slot = FindSlot(mTag);
|
|
|
|
MOZ_ASSERT(slot);
|
|
|
|
mPreviousValue = slot->get();
|
|
|
|
|
|
|
|
slot->set(&mCurrentValue);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-09-08 10:40:30 +03:00
|
|
|
void LogError(const nsLiteralCString& aModule, const nsACString& aExpr,
|
2020-12-21 12:32:12 +03:00
|
|
|
const nsACString& aSourceFile, int32_t aSourceLine,
|
|
|
|
Maybe<nsresult> aRv) {
|
2020-09-21 13:32:55 +03:00
|
|
|
nsAutoCString extraInfosString;
|
|
|
|
|
2021-01-22 11:39:17 +03:00
|
|
|
nsAutoCString rvName;
|
2020-12-21 12:32:12 +03:00
|
|
|
if (aRv) {
|
2021-01-22 11:39:17 +03:00
|
|
|
if (NS_ERROR_GET_MODULE(*aRv) == NS_ERROR_MODULE_WIN32) {
|
|
|
|
// XXX We could also try to get the Win32 error name here.
|
|
|
|
rvName = nsPrintfCString("WIN32(0x%" PRIX16 ")", NS_ERROR_GET_CODE(*aRv));
|
|
|
|
} else {
|
|
|
|
rvName = mozilla::GetStaticErrorName(*aRv);
|
|
|
|
}
|
2020-12-21 12:32:12 +03:00
|
|
|
extraInfosString.AppendPrintf(
|
2021-01-22 11:39:17 +03:00
|
|
|
" failed with "
|
2020-12-21 12:32:12 +03:00
|
|
|
"result 0x%" PRIX32 "%s%s%s",
|
2021-01-22 11:39:17 +03:00
|
|
|
static_cast<uint32_t>(*aRv), !rvName.IsEmpty() ? " (" : "",
|
|
|
|
!rvName.IsEmpty() ? rvName.get() : "", !rvName.IsEmpty() ? ")" : "");
|
2020-12-21 12:32:12 +03:00
|
|
|
}
|
|
|
|
|
2020-09-21 13:32:55 +03:00
|
|
|
#ifdef QM_ENABLE_SCOPED_LOG_EXTRA_INFO
|
2020-09-30 11:24:18 +03:00
|
|
|
const auto& extraInfos = ScopedLogExtraInfo::GetExtraInfoMap();
|
|
|
|
for (const auto& item : extraInfos) {
|
|
|
|
extraInfosString.Append(", "_ns + nsDependentCString(item.first) + "="_ns +
|
|
|
|
*item.second);
|
2020-09-21 13:32:55 +03:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-09-08 10:40:46 +03:00
|
|
|
#ifdef DEBUG
|
2020-10-23 03:57:07 +03:00
|
|
|
NS_DebugBreak(NS_DEBUG_WARNING, nsAutoCString(aModule + " failure"_ns).get(),
|
|
|
|
(extraInfosString.IsEmpty()
|
|
|
|
? nsPromiseFlatCString(aExpr)
|
|
|
|
: static_cast<const nsCString&>(
|
|
|
|
nsAutoCString(aExpr + extraInfosString)))
|
|
|
|
.get(),
|
|
|
|
nsPromiseFlatCString(aSourceFile).get(), aSourceLine);
|
2020-09-08 10:40:46 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(EARLY_BETA_OR_EARLIER) || defined(DEBUG)
|
|
|
|
nsCOMPtr<nsIConsoleService> console =
|
|
|
|
do_GetService(NS_CONSOLESERVICE_CONTRACTID);
|
|
|
|
if (console) {
|
|
|
|
NS_ConvertUTF8toUTF16 message(aModule + " failure: '"_ns + aExpr +
|
|
|
|
"', file "_ns + GetLeafName(aSourceFile) +
|
2020-10-14 18:45:50 +03:00
|
|
|
", line "_ns + IntToCString(aSourceLine) +
|
2020-09-21 13:32:55 +03:00
|
|
|
extraInfosString);
|
2020-09-08 10:40:46 +03:00
|
|
|
|
|
|
|
// The concatenation above results in a message like:
|
|
|
|
// QuotaManager failure: 'EXP', file XYZ, line N)
|
|
|
|
|
|
|
|
console->LogStringMessage(message.get());
|
|
|
|
}
|
2020-12-09 23:01:07 +03:00
|
|
|
|
|
|
|
# ifdef QM_ENABLE_SCOPED_LOG_EXTRA_INFO
|
|
|
|
if (const auto contextIt = extraInfos.find(ScopedLogExtraInfo::kTagContext);
|
|
|
|
contextIt != extraInfos.cend()) {
|
|
|
|
// For now, we don't include aExpr in the telemetry event. It might help to
|
|
|
|
// match locations across versions, but they might be large.
|
|
|
|
auto extra = Some([&] {
|
|
|
|
auto res = CopyableTArray<EventExtraEntry>{};
|
|
|
|
res.SetCapacity(5);
|
|
|
|
res.AppendElement(EventExtraEntry{"module"_ns, aModule});
|
|
|
|
res.AppendElement(EventExtraEntry{"source_file"_ns,
|
|
|
|
nsCString(GetLeafName(aSourceFile))});
|
|
|
|
res.AppendElement(
|
|
|
|
EventExtraEntry{"source_line"_ns, IntToCString(aSourceLine)});
|
|
|
|
res.AppendElement(EventExtraEntry{
|
|
|
|
"context"_ns, nsPromiseFlatCString{*contextIt->second}});
|
|
|
|
|
2021-01-22 11:39:17 +03:00
|
|
|
if (!rvName.IsEmpty()) {
|
2020-12-21 12:32:12 +03:00
|
|
|
res.AppendElement(EventExtraEntry{"result"_ns, nsCString{rvName}});
|
|
|
|
}
|
|
|
|
|
2021-03-04 12:13:21 +03:00
|
|
|
// The sequence number is currently per-process, and we don't record the
|
|
|
|
// thread id. This is ok as long as we only record during storage
|
|
|
|
// initialization, and that happens (mostly) from a single thread. It's
|
|
|
|
// safe even if errors from multiple threads are interleaved, but the data
|
|
|
|
// will be hard to analyze then. In that case, we should record a pair of
|
|
|
|
// thread id and thread-local sequence number.
|
2021-01-07 19:18:32 +03:00
|
|
|
static Atomic<int32_t> sSequenceNumber{0};
|
|
|
|
|
|
|
|
res.AppendElement(
|
|
|
|
EventExtraEntry{"seq"_ns, IntToCString(++sSequenceNumber)});
|
|
|
|
|
2020-12-09 23:01:07 +03:00
|
|
|
return res;
|
|
|
|
}());
|
|
|
|
|
|
|
|
Telemetry::RecordEvent(Telemetry::EventID::DomQuotaTry_Error_Step,
|
|
|
|
Nothing(), extra);
|
|
|
|
}
|
|
|
|
# endif
|
2020-09-08 10:40:46 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-06-25 10:16:47 +03:00
|
|
|
#ifdef DEBUG
|
|
|
|
Result<bool, nsresult> WarnIfFileIsUnknown(nsIFile& aFile,
|
|
|
|
const char* aSourceFile,
|
|
|
|
const int32_t aSourceLine) {
|
|
|
|
nsString leafName;
|
|
|
|
nsresult rv = aFile.GetLeafName(leafName);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return Err(rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isDirectory;
|
|
|
|
rv = aFile.IsDirectory(&isDirectory);
|
|
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
|
|
return Err(rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isDirectory) {
|
|
|
|
// Don't warn about OS metadata files. These files are only used in
|
|
|
|
// different platforms, but the profile can be shared across different
|
|
|
|
// operating systems, so we check it on all platforms.
|
|
|
|
if (leafName.Equals(kDSStoreFileName) ||
|
|
|
|
leafName.Equals(kDesktopFileName) ||
|
|
|
|
leafName.Equals(kDesktopIniFileName,
|
|
|
|
nsCaseInsensitiveStringComparator) ||
|
|
|
|
leafName.Equals(kThumbsDbFileName, nsCaseInsensitiveStringComparator)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't warn about files starting with ".".
|
|
|
|
if (leafName.First() == char16_t('.')) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_DebugBreak(
|
|
|
|
NS_DEBUG_WARNING,
|
|
|
|
nsPrintfCString("Something (%s) in the directory that doesn't belong!",
|
|
|
|
NS_ConvertUTF16toUTF8(leafName).get())
|
|
|
|
.get(),
|
|
|
|
nullptr, aSourceFile, aSourceLine);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-11-04 20:04:01 +03:00
|
|
|
} // namespace mozilla::dom::quota
|