mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
59e31c9007
- Bug 1182987 - Part 1: Remove unreferenced files immediately; r=baku (3ced56ca25) - Bug 1182987 - Part 2: Add getQuotaObjects() to mozIStorageConnection; r=mak (64b4dc418c) - Bug 1182987 - Part 3: Add "cleanup" transaction with disabled quota checks and vacuuming/checkpointing after commit; r=baku (7c19f28ae2) - Bug 1182987 - Part 4: Add a test for QuotaExceededError recovery and the new "cleanup" transaction type; r=baku (f91935e737) - Bug 1182987 - Part 5: Change mode of "readwrite" transaction to "cleanup" after QuotaExceeded is fired; r=baku (79d709970d) - Bug 1257725 part 3. Get rid of ThreadsafeAutoJSContext usage in Promise code. r=bholley (405d3c03d4) - Bug 1257725 part 1. Get rid of ThreadsafeAutoJSContext usage in JSEventHandler::HandleEvent. r=smaug (3222a73565) - Bug 1257725 part 4. Get rid of ThreadsafeAutoJSContext usage in IndexedDB code, except for IDBRequest::CaptureCaller. r=khuey (8ad88560f0) - Bug 1257725 part 5. Get rid of ThreadsafeAutoJSContext usage in IDBRequest::CaptureCaller. r=khuey (50a1f05ec9) - Bug 1257725 part 6. Get rid of ThreadsafeAutoJSContext. r=bholley (8968c69fcc) - Bug 1247420 - part1: removeContentState. r=smaug (6c7a54b58e) - Bug 1247420 - part2: IPC hover state management for select. r=Felipc (c7809aec7c) - Bug 1223533 - Don't hide the select popup on irrelevant pagehide events. r=mconley (0cf218515a) - Bug 1253486, [e10s only] hide select popups when the select element is removed, r=mconley (8a7049b6f1) - Bug 1180827 - Fix reuse of previous search results. r=MattN (6b4e65d318) - Bug 1242208 - Fix cached form history results with a datalist present. r=MattN (10078ada31) - Bug 1252074 - test_form_autocomplete.html and test_form_autocomplete_with_list.html should pass on e10s. r=paolo (8a9cf4a5f1) - Bug 1260441 - Never pass a null js context to OpenCursor() r=bz (8d818b0257) - Bug 1170045 - part 2 - use SegmentedVector in the DeferredFinalize implementation; r=mccr8 (3954a5e390) - Bug 1265770. Don't try to get a prototype for the interface object for an interface that's [NoInterfaceObject], since it's just unnecessary work that can't even be done at all in some cases (e.g. when the parent interface is also [NoInterfaceObject]). r=peterv (53d3077e31) - Bug 1264187 - check for a ProtoAndIfaceCache before blindly destroying it; r=bz (97536e815b) - Bug 1104955 part 3. Pass our unscopable names to CreateInterfaceObjects and have it define the right thing on the prototype. r=khuey (48386ab6b5) - Bug 934889: Use JS_InitStandardClasses everywhere now that it works. r=bz (01d545259a) - Bug 1258585. Remove some remaining vestiges of WebIDL quickstubs. r=peterv (3fa02388f1)
443 lines
11 KiB
C++
443 lines
11 KiB
C++
/* -*- 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 "DataStoreDB.h"
|
|
|
|
#include "DataStoreCallbacks.h"
|
|
#include "jsapi.h"
|
|
#include "mozilla/dom/IDBDatabaseBinding.h"
|
|
#include "mozilla/dom/IDBDatabase.h"
|
|
#include "mozilla/dom/IDBEvents.h"
|
|
#include "mozilla/dom/IDBFactory.h"
|
|
#include "mozilla/dom/IDBFactoryBinding.h"
|
|
#include "mozilla/dom/IDBIndex.h"
|
|
#include "mozilla/dom/IDBObjectStore.h"
|
|
#include "mozilla/dom/IDBObjectStoreBinding.h"
|
|
#include "mozilla/dom/IDBRequest.h"
|
|
#include "mozilla/dom/IDBTransaction.h"
|
|
#include "nsComponentManagerUtils.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsIDOMEvent.h"
|
|
#include "nsIPrincipal.h"
|
|
#include "nsIXPConnect.h"
|
|
#include "nsNullPrincipal.h"
|
|
|
|
#define DATASTOREDB_VERSION 1
|
|
#define DATASTOREDB_NAME "DataStoreDB"
|
|
#define DATASTOREDB_REVISION_INDEX "revisionIndex"
|
|
|
|
using namespace mozilla::dom::indexedDB;
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
class VersionChangeListener final : public nsIDOMEventListener
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
|
|
explicit VersionChangeListener(IDBDatabase* aDatabase)
|
|
: mDatabase(aDatabase)
|
|
{}
|
|
|
|
// nsIDOMEventListener
|
|
NS_IMETHOD HandleEvent(nsIDOMEvent* aEvent) override
|
|
{
|
|
nsString type;
|
|
nsresult rv = aEvent->GetType(type);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (!type.EqualsASCII("versionchange")) {
|
|
MOZ_ASSERT_UNREACHABLE("Expected a versionchange event");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
rv = mDatabase->RemoveEventListener(NS_LITERAL_STRING("versionchange"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
nsCOMPtr<IDBVersionChangeEvent> event = do_QueryInterface(aEvent);
|
|
MOZ_ASSERT(event);
|
|
|
|
Nullable<uint64_t> version = event->GetNewVersion();
|
|
MOZ_ASSERT(version.IsNull());
|
|
#endif
|
|
|
|
mDatabase->Close();
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
IDBDatabase* mDatabase;
|
|
|
|
~VersionChangeListener() {}
|
|
};
|
|
|
|
NS_IMPL_ISUPPORTS(VersionChangeListener, nsIDOMEventListener)
|
|
|
|
NS_IMPL_ISUPPORTS(DataStoreDB, nsIDOMEventListener)
|
|
|
|
DataStoreDB::DataStoreDB(const nsAString& aManifestURL, const nsAString& aName)
|
|
: mState(Inactive)
|
|
, mCreatedSchema(false)
|
|
{
|
|
mDatabaseName.Assign(aName);
|
|
mDatabaseName.Append('|');
|
|
mDatabaseName.Append(aManifestURL);
|
|
}
|
|
|
|
DataStoreDB::~DataStoreDB()
|
|
{
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::CreateFactoryIfNeeded()
|
|
{
|
|
if (!mFactory) {
|
|
nsresult rv;
|
|
nsCOMPtr<nsIPrincipal> principal = nsNullPrincipal::Create();
|
|
if (!principal) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsIXPConnect* xpc = nsContentUtils::XPConnect();
|
|
MOZ_ASSERT(xpc);
|
|
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
JSContext* cx = jsapi.cx();
|
|
JS::Rooted<JSObject*> global(cx);
|
|
rv = xpc->CreateSandbox(cx, principal, global.address());
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
// The CreateSandbox call returns a proxy to the actual sandbox object. We
|
|
// don't need a proxy here.
|
|
global = js::UncheckedUnwrap(global);
|
|
|
|
JSAutoCompartment ac(cx, global);
|
|
|
|
rv = IDBFactory::CreateForDatastore(cx, global, getter_AddRefs(mFactory));
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::Open(IDBTransactionMode aMode, const Sequence<nsString>& aDbs,
|
|
DataStoreDBCallback* aCallback)
|
|
{
|
|
MOZ_ASSERT(mState == Inactive);
|
|
|
|
nsresult rv = CreateFactoryIfNeeded();
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
// We only need a JSContext here to get a stack from, so just init our
|
|
// AutoJSAPI without a global.
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
ErrorResult error;
|
|
mRequest = mFactory->Open(jsapi.cx(), mDatabaseName, DATASTOREDB_VERSION,
|
|
error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
|
|
rv = AddEventListeners();
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
mState = Active;
|
|
mTransactionMode = aMode;
|
|
mObjectStores = aDbs;
|
|
mCallback = aCallback;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
DataStoreDB::HandleEvent(nsIDOMEvent* aEvent)
|
|
{
|
|
nsString type;
|
|
nsresult rv = aEvent->GetType(type);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (type.EqualsASCII("success")) {
|
|
RemoveEventListeners();
|
|
mState = Inactive;
|
|
|
|
rv = DatabaseOpened();
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
mCallback->Run(this, DataStoreDBCallback::Error);
|
|
} else {
|
|
mCallback->Run(this, mCreatedSchema
|
|
? DataStoreDBCallback::CreatedSchema :
|
|
DataStoreDBCallback::Success);
|
|
}
|
|
|
|
mRequest = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (type.EqualsASCII("upgradeneeded")) {
|
|
return UpgradeSchema(aEvent);
|
|
}
|
|
|
|
if (type.EqualsASCII("error") || type.EqualsASCII("blocked")) {
|
|
RemoveEventListeners();
|
|
mState = Inactive;
|
|
mCallback->Run(this, DataStoreDBCallback::Error);
|
|
mRequest = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
MOZ_CRASH("This should not happen");
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::UpgradeSchema(nsIDOMEvent* aEvent)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
// This DB has been just created and we have to inform the callback about
|
|
// this.
|
|
mCreatedSchema = true;
|
|
|
|
#ifdef DEBUG
|
|
nsCOMPtr<IDBVersionChangeEvent> event = do_QueryInterface(aEvent);
|
|
MOZ_ASSERT(event);
|
|
|
|
Nullable<uint64_t> version = event->GetNewVersion();
|
|
MOZ_ASSERT(!version.IsNull());
|
|
MOZ_ASSERT(version.Value() == DATASTOREDB_VERSION);
|
|
#endif
|
|
|
|
ErrorResult error;
|
|
JS::Rooted<JS::Value> result(nsContentUtils::RootingCx());
|
|
mRequest->GetResult(&result, error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
|
|
MOZ_ASSERT(result.isObject());
|
|
|
|
IDBDatabase* database = nullptr;
|
|
nsresult rv = UNWRAP_OBJECT(IDBDatabase, &result.toObject(), database);
|
|
if (NS_FAILED(rv)) {
|
|
NS_WARNING("Didn't get the object we expected!");
|
|
return rv;
|
|
}
|
|
|
|
{
|
|
IDBObjectStoreParameters params;
|
|
params.Init(NS_LITERAL_STRING("{ \"autoIncrement\": true }"));
|
|
RefPtr<IDBObjectStore> store =
|
|
database->CreateObjectStore(NS_LITERAL_STRING(DATASTOREDB_NAME),
|
|
params, error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
}
|
|
|
|
RefPtr<IDBObjectStore> store;
|
|
|
|
{
|
|
IDBObjectStoreParameters params;
|
|
params.Init(NS_LITERAL_STRING("{ \"autoIncrement\": true, \"keyPath\": \"internalRevisionId\" }"));
|
|
|
|
store =
|
|
database->CreateObjectStore(NS_LITERAL_STRING(DATASTOREDB_REVISION),
|
|
params, error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
}
|
|
|
|
{
|
|
IDBIndexParameters params;
|
|
params.Init(NS_LITERAL_STRING("{ \"unique\": true }"));
|
|
RefPtr<IDBIndex> index =
|
|
store->CreateIndex(NS_LITERAL_STRING(DATASTOREDB_REVISION_INDEX),
|
|
NS_LITERAL_STRING("revisionId"), params, error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::DatabaseOpened()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
ErrorResult error;
|
|
JS::Rooted<JS::Value> result(nsContentUtils::RootingCx());
|
|
mRequest->GetResult(&result, error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
|
|
MOZ_ASSERT(result.isObject());
|
|
|
|
nsresult rv = UNWRAP_OBJECT(IDBDatabase, &result.toObject(), mDatabase);
|
|
if (NS_FAILED(rv)) {
|
|
NS_WARNING("Didn't get the object we expected!");
|
|
return rv;
|
|
}
|
|
|
|
RefPtr<VersionChangeListener> listener =
|
|
new VersionChangeListener(mDatabase);
|
|
rv = mDatabase->EventTarget::AddEventListener(
|
|
NS_LITERAL_STRING("versionchange"), listener, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
StringOrStringSequence objectStores;
|
|
if (!objectStores.RawSetAsStringSequence().AppendElements(mObjectStores,
|
|
fallible)) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
|
|
// We init with the global of our result, just for consistency.
|
|
AutoJSAPI jsapi;
|
|
if (!jsapi.Init(&result.toObject())) {
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
RefPtr<IDBTransaction> txn;
|
|
error = mDatabase->Transaction(jsapi.cx(),
|
|
objectStores,
|
|
mTransactionMode,
|
|
getter_AddRefs(txn));
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
|
|
mTransaction = txn.forget();
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::Delete()
|
|
{
|
|
MOZ_ASSERT(mState == Inactive);
|
|
|
|
nsresult rv = CreateFactoryIfNeeded();
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
mTransaction = nullptr;
|
|
|
|
if (mDatabase) {
|
|
mDatabase->Close();
|
|
mDatabase = nullptr;
|
|
}
|
|
|
|
// We only need a JSContext here to get a stack from, so just init our
|
|
// AutoJSAPI without a global.
|
|
AutoJSAPI jsapi;
|
|
jsapi.Init();
|
|
ErrorResult error;
|
|
RefPtr<IDBOpenDBRequest> request =
|
|
mFactory->DeleteDatabase(jsapi.cx(), mDatabaseName, IDBOpenDBOptions(),
|
|
error);
|
|
if (NS_WARN_IF(error.Failed())) {
|
|
return error.StealNSResult();
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
IDBTransaction*
|
|
DataStoreDB::Transaction() const
|
|
{
|
|
MOZ_ASSERT(mTransaction);
|
|
MOZ_ASSERT(mTransaction->IsOpen());
|
|
return mTransaction;
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::AddEventListeners()
|
|
{
|
|
nsresult rv;
|
|
rv = mRequest->EventTarget::AddEventListener(NS_LITERAL_STRING("success"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = mRequest->EventTarget::AddEventListener(NS_LITERAL_STRING("upgradeneeded"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = mRequest->EventTarget::AddEventListener(NS_LITERAL_STRING("error"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = mRequest->EventTarget::AddEventListener(NS_LITERAL_STRING("blocked"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
DataStoreDB::RemoveEventListeners()
|
|
{
|
|
nsresult rv;
|
|
rv = mRequest->RemoveEventListener(NS_LITERAL_STRING("success"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = mRequest->RemoveEventListener(NS_LITERAL_STRING("upgradeneeded"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = mRequest->RemoveEventListener(NS_LITERAL_STRING("error"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
rv = mRequest->RemoveEventListener(NS_LITERAL_STRING("blocked"),
|
|
this, false);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|