mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-02 07:05:24 +00:00
431 lines
11 KiB
C++
431 lines
11 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=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 "base/basictypes.h"
|
|
|
|
#include "IDBKeyRange.h"
|
|
|
|
#include "nsIXPConnect.h"
|
|
|
|
#include "nsJSUtils.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsDOMClassInfoID.h"
|
|
#include "Key.h"
|
|
|
|
#include "mozilla/dom/indexedDB/PIndexedDBIndex.h"
|
|
#include "mozilla/dom/indexedDB/PIndexedDBObjectStore.h"
|
|
|
|
USING_INDEXEDDB_NAMESPACE
|
|
using namespace mozilla::dom::indexedDB::ipc;
|
|
|
|
namespace {
|
|
|
|
inline
|
|
bool
|
|
ReturnKeyRange(JSContext* aCx,
|
|
jsval* aVp,
|
|
IDBKeyRange* aKeyRange)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
NS_ASSERTION(aCx, "Null pointer!");
|
|
NS_ASSERTION(aVp, "Null pointer!");
|
|
NS_ASSERTION(aKeyRange, "Null pointer!");
|
|
|
|
nsIXPConnect* xpc = nsContentUtils::XPConnect();
|
|
NS_ASSERTION(xpc, "This should never be null!");
|
|
|
|
JSObject* global = JS_GetGlobalForScopeChain(aCx);
|
|
if (!global) {
|
|
NS_WARNING("Couldn't get global object!");
|
|
return false;
|
|
}
|
|
|
|
nsCOMPtr<nsIXPConnectJSObjectHolder> holder;
|
|
if (NS_FAILED(xpc->WrapNative(aCx, global, aKeyRange,
|
|
NS_GET_IID(nsIIDBKeyRange),
|
|
getter_AddRefs(holder)))) {
|
|
JS_ReportError(aCx, "Couldn't wrap IDBKeyRange object.");
|
|
return false;
|
|
}
|
|
|
|
JSObject* result;
|
|
if (NS_FAILED(holder->GetJSObject(&result))) {
|
|
JS_ReportError(aCx, "Couldn't get JSObject from wrapper.");
|
|
return false;
|
|
}
|
|
|
|
JS_SET_RVAL(aCx, aVp, OBJECT_TO_JSVAL(result));
|
|
return true;
|
|
}
|
|
|
|
inline
|
|
nsresult
|
|
GetKeyFromJSVal(JSContext* aCx,
|
|
jsval aVal,
|
|
Key& aKey,
|
|
bool aAllowUnset = false)
|
|
{
|
|
nsresult rv = aKey.SetFromJSVal(aCx, aVal);
|
|
if (NS_FAILED(rv)) {
|
|
NS_ASSERTION(NS_ERROR_GET_MODULE(rv) == NS_ERROR_MODULE_DOM_INDEXEDDB,
|
|
"Bad error code!");
|
|
return rv;
|
|
}
|
|
|
|
if (aKey.IsUnset() && !aAllowUnset) {
|
|
return NS_ERROR_DOM_INDEXEDDB_DATA_ERR;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
inline
|
|
void
|
|
ThrowException(JSContext* aCx,
|
|
nsresult aErrorCode)
|
|
{
|
|
NS_ASSERTION(NS_FAILED(aErrorCode), "Not an error code!");
|
|
xpc::Throw(aCx, aErrorCode);
|
|
}
|
|
|
|
inline
|
|
bool
|
|
GetKeyFromJSValOrThrow(JSContext* aCx,
|
|
jsval aVal,
|
|
Key& aKey)
|
|
{
|
|
nsresult rv = GetKeyFromJSVal(aCx, aVal, aKey);
|
|
if (NS_FAILED(rv)) {
|
|
ThrowException(aCx, rv);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
JSBool
|
|
MakeOnlyKeyRange(JSContext* aCx,
|
|
unsigned aArgc,
|
|
jsval* aVp)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
jsval val;
|
|
if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "v", &val)) {
|
|
return false;
|
|
}
|
|
|
|
nsRefPtr<IDBKeyRange> keyRange = new IDBKeyRange(false, false, true);
|
|
|
|
if (!GetKeyFromJSValOrThrow(aCx, val, keyRange->Lower())) {
|
|
return false;
|
|
}
|
|
|
|
return ReturnKeyRange(aCx, aVp, keyRange);
|
|
}
|
|
|
|
JSBool
|
|
MakeLowerBoundKeyRange(JSContext* aCx,
|
|
unsigned aArgc,
|
|
jsval* aVp)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
jsval val;
|
|
JSBool open = false;
|
|
if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "v/b", &val, &open)) {
|
|
return false;
|
|
}
|
|
|
|
nsRefPtr<IDBKeyRange> keyRange = new IDBKeyRange(open, true, false);
|
|
|
|
if (!GetKeyFromJSValOrThrow(aCx, val, keyRange->Lower())) {
|
|
return false;
|
|
}
|
|
|
|
return ReturnKeyRange(aCx, aVp, keyRange);
|
|
}
|
|
|
|
JSBool
|
|
MakeUpperBoundKeyRange(JSContext* aCx,
|
|
unsigned aArgc,
|
|
jsval* aVp)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
jsval val;
|
|
JSBool open = false;
|
|
if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "v/b", &val, &open)) {
|
|
return false;
|
|
}
|
|
|
|
nsRefPtr<IDBKeyRange> keyRange = new IDBKeyRange(true, open, false);
|
|
|
|
if (!GetKeyFromJSValOrThrow(aCx, val, keyRange->Upper())) {
|
|
return false;
|
|
}
|
|
|
|
return ReturnKeyRange(aCx, aVp, keyRange);
|
|
}
|
|
|
|
JSBool
|
|
MakeBoundKeyRange(JSContext* aCx,
|
|
unsigned aArgc,
|
|
jsval* aVp)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
jsval lowerVal, upperVal;
|
|
JSBool lowerOpen = false, upperOpen = false;
|
|
if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "vv/bb", &lowerVal,
|
|
&upperVal, &lowerOpen, &upperOpen)) {
|
|
return false;
|
|
}
|
|
|
|
nsRefPtr<IDBKeyRange> keyRange = new IDBKeyRange(lowerOpen, upperOpen, false);
|
|
|
|
if (!GetKeyFromJSValOrThrow(aCx, lowerVal, keyRange->Lower()) ||
|
|
!GetKeyFromJSValOrThrow(aCx, upperVal, keyRange->Upper())) {
|
|
return false;
|
|
}
|
|
|
|
if (keyRange->Lower() > keyRange->Upper() ||
|
|
(keyRange->Lower() == keyRange->Upper() && (lowerOpen || upperOpen))) {
|
|
ThrowException(aCx, NS_ERROR_DOM_INDEXEDDB_DATA_ERR);
|
|
return false;
|
|
}
|
|
|
|
return ReturnKeyRange(aCx, aVp, keyRange);
|
|
}
|
|
|
|
#define KEYRANGE_FUNCTION_FLAGS (JSPROP_ENUMERATE | JSPROP_PERMANENT)
|
|
|
|
const JSFunctionSpec gKeyRangeConstructors[] = {
|
|
JS_FN("only", MakeOnlyKeyRange, 1, KEYRANGE_FUNCTION_FLAGS),
|
|
JS_FN("lowerBound", MakeLowerBoundKeyRange, 1, KEYRANGE_FUNCTION_FLAGS),
|
|
JS_FN("upperBound", MakeUpperBoundKeyRange, 1, KEYRANGE_FUNCTION_FLAGS),
|
|
JS_FN("bound", MakeBoundKeyRange, 2, KEYRANGE_FUNCTION_FLAGS),
|
|
JS_FS_END
|
|
};
|
|
|
|
#undef KEYRANGE_FUNCTION_FLAGS
|
|
|
|
} // anonymous namespace
|
|
|
|
// static
|
|
JSBool
|
|
IDBKeyRange::DefineConstructors(JSContext* aCx,
|
|
JSObject* aObject)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
NS_ASSERTION(aCx, "Null pointer!");
|
|
NS_ASSERTION(aObject, "Null pointer!");
|
|
|
|
// Add the constructor methods for key ranges.
|
|
return JS_DefineFunctions(aCx, aObject,
|
|
const_cast<JSFunctionSpec*>(gKeyRangeConstructors));
|
|
}
|
|
|
|
// static
|
|
nsresult
|
|
IDBKeyRange::FromJSVal(JSContext* aCx,
|
|
const jsval& aVal,
|
|
IDBKeyRange** aKeyRange)
|
|
{
|
|
nsresult rv;
|
|
nsRefPtr<IDBKeyRange> keyRange;
|
|
|
|
if (JSVAL_IS_VOID(aVal) || JSVAL_IS_NULL(aVal)) {
|
|
// undefined and null returns no IDBKeyRange.
|
|
}
|
|
else if (JSVAL_IS_PRIMITIVE(aVal) ||
|
|
JS_IsArrayObject(aCx, JSVAL_TO_OBJECT(aVal)) ||
|
|
JS_ObjectIsDate(aCx, JSVAL_TO_OBJECT(aVal))) {
|
|
// A valid key returns an 'only' IDBKeyRange.
|
|
keyRange = new IDBKeyRange(false, false, true);
|
|
|
|
rv = GetKeyFromJSVal(aCx, aVal, keyRange->Lower());
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
}
|
|
else {
|
|
// An object is not permitted unless it's another IDBKeyRange.
|
|
nsIXPConnect* xpc = nsContentUtils::XPConnect();
|
|
NS_ASSERTION(xpc, "This should never be null!");
|
|
|
|
nsCOMPtr<nsIXPConnectWrappedNative> wrapper;
|
|
rv = xpc->GetWrappedNativeOfJSObject(aCx, JSVAL_TO_OBJECT(aVal),
|
|
getter_AddRefs(wrapper));
|
|
if (NS_FAILED(rv)) {
|
|
return NS_ERROR_DOM_INDEXEDDB_DATA_ERR;
|
|
}
|
|
|
|
nsCOMPtr<nsIIDBKeyRange> iface;
|
|
if (!wrapper || !(iface = do_QueryInterface(wrapper->Native()))) {
|
|
// Some random JS object?
|
|
return NS_ERROR_DOM_INDEXEDDB_DATA_ERR;
|
|
}
|
|
|
|
keyRange = static_cast<IDBKeyRange*>(iface.get());
|
|
}
|
|
|
|
keyRange.forget(aKeyRange);
|
|
return NS_OK;
|
|
}
|
|
|
|
// static
|
|
template <class T>
|
|
already_AddRefed<IDBKeyRange>
|
|
IDBKeyRange::FromSerializedKeyRange(const T& aKeyRange)
|
|
{
|
|
nsRefPtr<IDBKeyRange> keyRange =
|
|
new IDBKeyRange(aKeyRange.lowerOpen(), aKeyRange.upperOpen(),
|
|
aKeyRange.isOnly());
|
|
keyRange->Lower() = aKeyRange.lower();
|
|
if (!keyRange->IsOnly()) {
|
|
keyRange->Upper() = aKeyRange.upper();
|
|
}
|
|
return keyRange.forget();
|
|
}
|
|
|
|
template <class T>
|
|
void
|
|
IDBKeyRange::ToSerializedKeyRange(T& aKeyRange)
|
|
{
|
|
aKeyRange.lowerOpen() = IsLowerOpen();
|
|
aKeyRange.upperOpen() = IsUpperOpen();
|
|
aKeyRange.isOnly() = IsOnly();
|
|
|
|
aKeyRange.lower() = Lower();
|
|
if (!IsOnly()) {
|
|
aKeyRange.upper() = Upper();
|
|
}
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(IDBKeyRange)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(IDBKeyRange)
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_JSVAL_MEMBER_CALLBACK(mCachedLowerVal)
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_JSVAL_MEMBER_CALLBACK(mCachedUpperVal)
|
|
NS_IMPL_CYCLE_COLLECTION_TRACE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(IDBKeyRange)
|
|
tmp->DropJSObjects();
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(IDBKeyRange)
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_ENTRY(nsIIDBKeyRange)
|
|
NS_DOM_INTERFACE_MAP_ENTRY_CLASSINFO(IDBKeyRange)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(IDBKeyRange)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(IDBKeyRange)
|
|
|
|
DOMCI_DATA(IDBKeyRange, IDBKeyRange)
|
|
|
|
void
|
|
IDBKeyRange::DropJSObjects()
|
|
{
|
|
if (!mRooted) {
|
|
return;
|
|
}
|
|
mCachedLowerVal = JSVAL_VOID;
|
|
mCachedUpperVal = JSVAL_VOID;
|
|
mHaveCachedLowerVal = false;
|
|
mHaveCachedUpperVal = false;
|
|
mRooted = false;
|
|
NS_DROP_JS_OBJECTS(this, IDBKeyRange);
|
|
}
|
|
|
|
IDBKeyRange::~IDBKeyRange()
|
|
{
|
|
DropJSObjects();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
IDBKeyRange::GetLower(JSContext* aCx,
|
|
jsval* aLower)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
if (!mHaveCachedLowerVal) {
|
|
if (!mRooted) {
|
|
NS_HOLD_JS_OBJECTS(this, IDBKeyRange);
|
|
mRooted = true;
|
|
}
|
|
|
|
nsresult rv = Lower().ToJSVal(aCx, &mCachedLowerVal);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mHaveCachedLowerVal = true;
|
|
}
|
|
|
|
*aLower = mCachedLowerVal;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
IDBKeyRange::GetUpper(JSContext* aCx,
|
|
jsval* aUpper)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
if (!mHaveCachedUpperVal) {
|
|
if (!mRooted) {
|
|
NS_HOLD_JS_OBJECTS(this, IDBKeyRange);
|
|
mRooted = true;
|
|
}
|
|
|
|
nsresult rv = Upper().ToJSVal(aCx, &mCachedUpperVal);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mHaveCachedUpperVal = true;
|
|
}
|
|
|
|
*aUpper = mCachedUpperVal;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
IDBKeyRange::GetLowerOpen(bool* aLowerOpen)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
*aLowerOpen = mLowerOpen;
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
IDBKeyRange::GetUpperOpen(bool* aUpperOpen)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
*aUpperOpen = mUpperOpen;
|
|
return NS_OK;
|
|
}
|
|
|
|
// Explicitly instantiate for all our key range types... Grumble.
|
|
template already_AddRefed<IDBKeyRange>
|
|
IDBKeyRange::FromSerializedKeyRange<FIXME_Bug_521898_objectstore::KeyRange>
|
|
(const FIXME_Bug_521898_objectstore::KeyRange& aKeyRange);
|
|
|
|
template already_AddRefed<IDBKeyRange>
|
|
IDBKeyRange::FromSerializedKeyRange<FIXME_Bug_521898_index::KeyRange>
|
|
(const FIXME_Bug_521898_index::KeyRange& aKeyRange);
|
|
|
|
template void
|
|
IDBKeyRange::ToSerializedKeyRange<FIXME_Bug_521898_objectstore::KeyRange>
|
|
(FIXME_Bug_521898_objectstore::KeyRange& aKeyRange);
|
|
|
|
template void
|
|
IDBKeyRange::ToSerializedKeyRange<FIXME_Bug_521898_index::KeyRange>
|
|
(FIXME_Bug_521898_index::KeyRange& aKeyRange);
|