/* -*- 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 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; } JS::Rooted result(aCx, holder->GetJSObject()); if (!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!"); JS::Rooted val(aCx); if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "v", val.address())) { return false; } nsRefPtr 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!"); JS::Rooted val(aCx); JSBool open = false; if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "v/b", val.address(), &open)) { return false; } nsRefPtr 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!"); JS::Rooted val(aCx); JSBool open = false; if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "v/b", val.address(), &open)) { return false; } nsRefPtr 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!"); JS::Rooted lowerVal(aCx), upperVal(aCx); JSBool lowerOpen = false, upperOpen = false; if (!JS_ConvertArguments(aCx, aArgc, JS_ARGV(aCx, aVp), "vv/bb", lowerVal.address(), upperVal.address(), &lowerOpen, &upperOpen)) { return false; } nsRefPtr 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, gKeyRangeConstructors); } // static nsresult IDBKeyRange::FromJSVal(JSContext* aCx, const jsval& aVal, IDBKeyRange** aKeyRange) { nsresult rv; nsRefPtr 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 wrapper; rv = xpc->GetWrappedNativeOfJSObject(aCx, JSVAL_TO_OBJECT(aVal), getter_AddRefs(wrapper)); if (NS_FAILED(rv)) { return NS_ERROR_DOM_INDEXEDDB_DATA_ERR; } nsCOMPtr iface; if (!wrapper || !(iface = do_QueryInterface(wrapper->Native()))) { // Some random JS object? return NS_ERROR_DOM_INDEXEDDB_DATA_ERR; } keyRange = static_cast(iface.get()); } keyRange.forget(aKeyRange); return NS_OK; } // static template already_AddRefed IDBKeyRange::FromSerializedKeyRange(const T& aKeyRange) { nsRefPtr keyRange = new IDBKeyRange(aKeyRange.lowerOpen(), aKeyRange.upperOpen(), aKeyRange.isOnly()); keyRange->Lower() = aKeyRange.lower(); if (!keyRange->IsOnly()) { keyRange->Upper() = aKeyRange.upper(); } return keyRange.forget(); } template 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::FromSerializedKeyRange (const FIXME_Bug_521898_objectstore::KeyRange& aKeyRange); template already_AddRefed IDBKeyRange::FromSerializedKeyRange (const FIXME_Bug_521898_index::KeyRange& aKeyRange); template void IDBKeyRange::ToSerializedKeyRange (FIXME_Bug_521898_objectstore::KeyRange& aKeyRange); template void IDBKeyRange::ToSerializedKeyRange (FIXME_Bug_521898_index::KeyRange& aKeyRange);