Blame dom/storage/StorageObserver.cpp

Packit f0b94e
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
Packit f0b94e
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
Packit f0b94e
/* This Source Code Form is subject to the terms of the Mozilla Public
Packit f0b94e
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit f0b94e
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
Packit f0b94e
Packit f0b94e
#include "StorageObserver.h"
Packit f0b94e
Packit f0b94e
#include "LocalStorageCache.h"
Packit f0b94e
#include "StorageDBThread.h"
Packit f0b94e
#include "StorageUtils.h"
Packit f0b94e
Packit f0b94e
#include "mozilla/BasePrincipal.h"
Packit f0b94e
#include "nsIObserverService.h"
Packit f0b94e
#include "nsIURI.h"
Packit f0b94e
#include "nsIURL.h"
Packit f0b94e
#include "nsIScriptSecurityManager.h"
Packit f0b94e
#include "nsIPermission.h"
Packit f0b94e
#include "nsIIDNService.h"
Packit f0b94e
#include "nsICookiePermission.h"
Packit f0b94e
Packit f0b94e
#include "nsPrintfCString.h"
Packit f0b94e
#include "nsXULAppAPI.h"
Packit f0b94e
#include "nsEscape.h"
Packit f0b94e
#include "nsNetCID.h"
Packit f0b94e
#include "mozilla/Preferences.h"
Packit f0b94e
#include "mozilla/Services.h"
Packit f0b94e
#include "nsServiceManagerUtils.h"
Packit f0b94e
Packit f0b94e
namespace mozilla {
Packit f0b94e
namespace dom {
Packit f0b94e
Packit f0b94e
using namespace StorageUtils;
Packit f0b94e
Packit f0b94e
static const char kStartupTopic[] = "sessionstore-windows-restored";
Packit f0b94e
static const uint32_t kStartupDelay = 0;
Packit f0b94e
Packit f0b94e
const char kTestingPref[] = "dom.storage.testing";
Packit f0b94e
Packit f0b94e
NS_IMPL_ISUPPORTS(StorageObserver, nsIObserver, nsISupportsWeakReference)
Packit f0b94e
Packit f0b94e
StorageObserver* StorageObserver::sSelf = nullptr;
Packit f0b94e
Packit f0b94e
// static
Packit f0b94e
nsresult StorageObserver::Init() {
Packit f0b94e
  if (sSelf) {
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
Packit f0b94e
  if (!obs) {
Packit f0b94e
    return NS_ERROR_UNEXPECTED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  sSelf = new StorageObserver();
Packit f0b94e
  NS_ADDREF(sSelf);
Packit f0b94e
Packit f0b94e
  // Chrome clear operations.
Packit f0b94e
  obs->AddObserver(sSelf, kStartupTopic, true);
Packit f0b94e
  obs->AddObserver(sSelf, "cookie-changed", true);
Packit f0b94e
  obs->AddObserver(sSelf, "perm-changed", true);
Packit f0b94e
  obs->AddObserver(sSelf, "browser:purge-domain-data", true);
Packit f0b94e
  obs->AddObserver(sSelf, "last-pb-context-exited", true);
Packit f0b94e
  obs->AddObserver(sSelf, "clear-origin-attributes-data", true);
Packit f0b94e
  obs->AddObserver(sSelf, "extension:purge-localStorage", true);
Packit f0b94e
Packit f0b94e
  // Shutdown
Packit f0b94e
  obs->AddObserver(sSelf, "profile-after-change", true);
Packit f0b94e
  if (XRE_IsParentProcess()) {
Packit f0b94e
    obs->AddObserver(sSelf, "profile-before-change", true);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Observe low device storage notifications.
Packit f0b94e
  obs->AddObserver(sSelf, "disk-space-watcher", true);
Packit f0b94e
Packit f0b94e
  // Testing
Packit f0b94e
#ifdef DOM_STORAGE_TESTS
Packit f0b94e
  Preferences::RegisterCallbackAndCall(TestingPrefChanged, kTestingPref);
Packit f0b94e
#endif
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
// static
Packit f0b94e
nsresult StorageObserver::Shutdown() {
Packit f0b94e
  if (!sSelf) {
Packit f0b94e
    return NS_ERROR_NOT_INITIALIZED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_RELEASE(sSelf);
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
// static
Packit f0b94e
void StorageObserver::TestingPrefChanged(const char* aPrefName,
Packit f0b94e
                                         void* aClosure) {
Packit f0b94e
  nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
Packit f0b94e
  if (!obs) {
Packit f0b94e
    return;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (Preferences::GetBool(kTestingPref)) {
Packit f0b94e
    obs->AddObserver(sSelf, "domstorage-test-flush-force", true);
Packit f0b94e
    if (XRE_IsParentProcess()) {
Packit f0b94e
      // Only to forward to child process.
Packit f0b94e
      obs->AddObserver(sSelf, "domstorage-test-flushed", true);
Packit f0b94e
    }
Packit f0b94e
    obs->AddObserver(sSelf, "domstorage-test-reload", true);
Packit f0b94e
  } else {
Packit f0b94e
    obs->RemoveObserver(sSelf, "domstorage-test-flush-force");
Packit f0b94e
    if (XRE_IsParentProcess()) {
Packit f0b94e
      // Only to forward to child process.
Packit f0b94e
      obs->RemoveObserver(sSelf, "domstorage-test-flushed");
Packit f0b94e
    }
Packit f0b94e
    obs->RemoveObserver(sSelf, "domstorage-test-reload");
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void StorageObserver::AddSink(StorageObserverSink* aObs) {
Packit f0b94e
  mSinks.AppendElement(aObs);
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void StorageObserver::RemoveSink(StorageObserverSink* aObs) {
Packit f0b94e
  mSinks.RemoveElement(aObs);
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void StorageObserver::Notify(const char* aTopic,
Packit f0b94e
                             const nsAString& aOriginAttributesPattern,
Packit f0b94e
                             const nsACString& aOriginScope) {
Packit f0b94e
  nsTObserverArray<StorageObserverSink*>::ForwardIterator iter(mSinks);
Packit f0b94e
  while (iter.HasMore()) {
Packit f0b94e
    StorageObserverSink* sink = iter.GetNext();
Packit f0b94e
    sink->Observe(aTopic, aOriginAttributesPattern, aOriginScope);
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void StorageObserver::NoteBackgroundThread(nsIEventTarget* aBackgroundThread) {
Packit f0b94e
  mBackgroundThread = aBackgroundThread;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult StorageObserver::ClearMatchingOrigin(const char16_t* aData,
Packit f0b94e
                                              nsACString& aOriginScope) {
Packit f0b94e
  nsresult rv;
Packit f0b94e
Packit f0b94e
  NS_ConvertUTF16toUTF8 domain(aData);
Packit f0b94e
Packit f0b94e
  nsAutoCString convertedDomain;
Packit f0b94e
  nsCOMPtr<nsIIDNService> converter = do_GetService(NS_IDNSERVICE_CONTRACTID);
Packit f0b94e
  if (converter) {
Packit f0b94e
    // Convert the domain name to the ACE format
Packit f0b94e
    rv = converter->ConvertUTF8toACE(domain, convertedDomain);
Packit f0b94e
  } else {
Packit f0b94e
    // In case the IDN service is not available, this is the best we can come
Packit f0b94e
    // up with!
Packit f0b94e
    rv = NS_EscapeURL(domain, esc_OnlyNonASCII | esc_AlwaysCopy,
Packit f0b94e
                      convertedDomain, fallible);
Packit f0b94e
  }
Packit f0b94e
  if (NS_WARN_IF(NS_FAILED(rv))) {
Packit f0b94e
    return rv;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  nsCString originScope;
Packit f0b94e
  rv = CreateReversedDomain(convertedDomain, originScope);
Packit f0b94e
  if (NS_WARN_IF(NS_FAILED(rv))) {
Packit f0b94e
    return rv;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (XRE_IsParentProcess()) {
Packit f0b94e
    StorageDBChild* storageChild = StorageDBChild::GetOrCreate();
Packit f0b94e
    if (NS_WARN_IF(!storageChild)) {
Packit f0b94e
      return NS_ERROR_FAILURE;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    storageChild->SendClearMatchingOrigin(originScope);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  aOriginScope = originScope;
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
NS_IMETHODIMP
Packit f0b94e
StorageObserver::Observe(nsISupports* aSubject, const char* aTopic,
Packit f0b94e
                         const char16_t* aData) {
Packit f0b94e
  nsresult rv;
Packit f0b94e
Packit f0b94e
  // Start the thread that opens the database.
Packit f0b94e
  if (!strcmp(aTopic, kStartupTopic)) {
Packit f0b94e
    MOZ_ASSERT(XRE_IsParentProcess());
Packit f0b94e
Packit f0b94e
    nsCOMPtr<nsIObserverService> obs = mozilla::services::GetObserverService();
Packit f0b94e
    obs->RemoveObserver(this, kStartupTopic);
Packit f0b94e
Packit f0b94e
    return NS_NewTimerWithObserver(getter_AddRefs(mDBThreadStartDelayTimer),
Packit f0b94e
                                   this, nsITimer::TYPE_ONE_SHOT,
Packit f0b94e
                                   kStartupDelay);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Timer callback used to start the database a short timer after startup
Packit f0b94e
  if (!strcmp(aTopic, NS_TIMER_CALLBACK_TOPIC)) {
Packit f0b94e
    MOZ_ASSERT(XRE_IsParentProcess());
Packit f0b94e
Packit f0b94e
    nsCOMPtr<nsITimer> timer = do_QueryInterface(aSubject);
Packit f0b94e
    if (!timer) {
Packit f0b94e
      return NS_ERROR_UNEXPECTED;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    if (timer == mDBThreadStartDelayTimer) {
Packit f0b94e
      mDBThreadStartDelayTimer = nullptr;
Packit f0b94e
Packit f0b94e
      StorageDBChild* storageChild = StorageDBChild::GetOrCreate();
Packit f0b94e
      if (NS_WARN_IF(!storageChild)) {
Packit f0b94e
        return NS_ERROR_FAILURE;
Packit f0b94e
      }
Packit f0b94e
Packit f0b94e
      storageChild->SendStartup();
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Clear everything, caches + database
Packit f0b94e
  if (!strcmp(aTopic, "cookie-changed")) {
Packit f0b94e
    if (!NS_LITERAL_STRING("cleared").Equals(aData)) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    StorageDBChild* storageChild = StorageDBChild::GetOrCreate();
Packit f0b94e
    if (NS_WARN_IF(!storageChild)) {
Packit f0b94e
      return NS_ERROR_FAILURE;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    storageChild->AsyncClearAll();
Packit f0b94e
Packit f0b94e
    if (XRE_IsParentProcess()) {
Packit f0b94e
      storageChild->SendClearAll();
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    Notify("cookie-cleared");
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Clear from caches everything that has been stored
Packit f0b94e
  // while in session-only mode
Packit f0b94e
  if (!strcmp(aTopic, "perm-changed")) {
Packit f0b94e
    // Check for cookie permission change
Packit f0b94e
    nsCOMPtr<nsIPermission> perm(do_QueryInterface(aSubject));
Packit f0b94e
    if (!perm) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    nsAutoCString type;
Packit f0b94e
    perm->GetType(type);
Packit f0b94e
    if (type != NS_LITERAL_CSTRING("cookie")) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    uint32_t cap = 0;
Packit f0b94e
    perm->GetCapability(&cap);
Packit f0b94e
    if (!(cap & nsICookiePermission::ACCESS_SESSION) ||
Packit f0b94e
        !NS_LITERAL_STRING("deleted").Equals(nsDependentString(aData))) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    nsCOMPtr<nsIPrincipal> principal;
Packit f0b94e
    perm->GetPrincipal(getter_AddRefs(principal));
Packit f0b94e
    if (!principal) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    nsAutoCString originSuffix;
Packit f0b94e
    BasePrincipal::Cast(principal)->OriginAttributesRef().CreateSuffix(
Packit f0b94e
        originSuffix);
Packit f0b94e
Packit f0b94e
    nsCOMPtr<nsIURI> origin;
Packit f0b94e
    principal->GetURI(getter_AddRefs(origin));
Packit f0b94e
    if (!origin) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    nsAutoCString host;
Packit f0b94e
    origin->GetHost(host);
Packit f0b94e
    if (host.IsEmpty()) {
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    nsAutoCString originScope;
Packit f0b94e
    rv = CreateReversedDomain(host, originScope);
Packit f0b94e
    NS_ENSURE_SUCCESS(rv, rv);
Packit f0b94e
Packit f0b94e
    Notify("session-only-cleared", NS_ConvertUTF8toUTF16(originSuffix),
Packit f0b94e
           originScope);
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (!strcmp(aTopic, "extension:purge-localStorage")) {
Packit f0b94e
    const char topic[] = "extension:purge-localStorage-caches";
Packit f0b94e
Packit f0b94e
    if (aData) {
Packit f0b94e
      nsCString originScope;
Packit f0b94e
      rv = ClearMatchingOrigin(aData, originScope);
Packit f0b94e
      if (NS_WARN_IF(NS_FAILED(rv))) {
Packit f0b94e
        return rv;
Packit f0b94e
      }
Packit f0b94e
Packit f0b94e
      Notify(topic, EmptyString(), originScope);
Packit f0b94e
    } else {
Packit f0b94e
      StorageDBChild* storageChild = StorageDBChild::GetOrCreate();
Packit f0b94e
      if (NS_WARN_IF(!storageChild)) {
Packit f0b94e
        return NS_ERROR_FAILURE;
Packit f0b94e
      }
Packit f0b94e
Packit f0b94e
      storageChild->AsyncClearAll();
Packit f0b94e
Packit f0b94e
      if (XRE_IsParentProcess()) {
Packit f0b94e
        storageChild->SendClearAll();
Packit f0b94e
      }
Packit f0b94e
Packit f0b94e
      Notify(topic);
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Clear everything (including so and pb data) from caches and database
Packit f0b94e
  // for the given domain and subdomains.
Packit f0b94e
  if (!strcmp(aTopic, "browser:purge-domain-data")) {
Packit f0b94e
    nsCString originScope;
Packit f0b94e
    rv = ClearMatchingOrigin(aData, originScope);
Packit f0b94e
    if (NS_WARN_IF(NS_FAILED(rv))) {
Packit f0b94e
      return rv;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    Notify("domain-data-cleared", EmptyString(), originScope);
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Clear all private-browsing caches
Packit f0b94e
  if (!strcmp(aTopic, "last-pb-context-exited")) {
Packit f0b94e
    Notify("private-browsing-data-cleared");
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Clear data of the origins whose prefixes will match the suffix.
Packit f0b94e
  if (!strcmp(aTopic, "clear-origin-attributes-data")) {
Packit f0b94e
    MOZ_ASSERT(XRE_IsParentProcess());
Packit f0b94e
Packit f0b94e
    OriginAttributesPattern pattern;
Packit f0b94e
    if (!pattern.Init(nsDependentString(aData))) {
Packit f0b94e
      NS_ERROR("Cannot parse origin attributes pattern");
Packit f0b94e
      return NS_ERROR_FAILURE;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    StorageDBChild* storageChild = StorageDBChild::GetOrCreate();
Packit f0b94e
    if (NS_WARN_IF(!storageChild)) {
Packit f0b94e
      return NS_ERROR_FAILURE;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    storageChild->SendClearMatchingOriginAttributes(pattern);
Packit f0b94e
Packit f0b94e
    Notify("origin-attr-pattern-cleared", nsDependentString(aData));
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (!strcmp(aTopic, "profile-after-change")) {
Packit f0b94e
    Notify("profile-change");
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (!strcmp(aTopic, "profile-before-change")) {
Packit f0b94e
    MOZ_ASSERT(XRE_IsParentProcess());
Packit f0b94e
Packit f0b94e
    if (mBackgroundThread) {
Packit f0b94e
      bool done = false;
Packit f0b94e
Packit f0b94e
      RefPtr<StorageDBThread::ShutdownRunnable> shutdownRunnable =
Packit f0b94e
          new StorageDBThread::ShutdownRunnable(done);
Packit f0b94e
      MOZ_ALWAYS_SUCCEEDS(
Packit f0b94e
          mBackgroundThread->Dispatch(shutdownRunnable, NS_DISPATCH_NORMAL));
Packit f0b94e
Packit f0b94e
      MOZ_ALWAYS_TRUE(SpinEventLoopUntil([&]() { return done; }));
Packit f0b94e
Packit f0b94e
      mBackgroundThread = nullptr;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (!strcmp(aTopic, "disk-space-watcher")) {
Packit f0b94e
    if (NS_LITERAL_STRING("full").Equals(aData)) {
Packit f0b94e
      Notify("low-disk-space");
Packit f0b94e
    } else if (NS_LITERAL_STRING("free").Equals(aData)) {
Packit f0b94e
      Notify("no-low-disk-space");
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
#ifdef DOM_STORAGE_TESTS
Packit f0b94e
  if (!strcmp(aTopic, "domstorage-test-flush-force")) {
Packit f0b94e
    StorageDBChild* storageChild = StorageDBChild::GetOrCreate();
Packit f0b94e
    if (NS_WARN_IF(!storageChild)) {
Packit f0b94e
      return NS_ERROR_FAILURE;
Packit f0b94e
    }
Packit f0b94e
Packit f0b94e
    storageChild->SendAsyncFlush();
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (!strcmp(aTopic, "domstorage-test-flushed")) {
Packit f0b94e
    // Only used to propagate to IPC children
Packit f0b94e
    Notify("test-flushed");
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  if (!strcmp(aTopic, "domstorage-test-reload")) {
Packit f0b94e
    Notify("test-reload");
Packit f0b94e
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
#endif
Packit f0b94e
Packit f0b94e
  NS_ERROR("Unexpected topic");
Packit f0b94e
  return NS_ERROR_UNEXPECTED;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
}  // namespace dom
Packit f0b94e
}  // namespace mozilla