forked from mirrors/gecko-dev
		
	The functionality has been moved to DatabaseOperationBase::BindKeyRangeToStatement resp. DatabaseOperationBase::GetBindingClauseForKeyRange as part of Bug 994190. Differential Revision: https://phabricator.services.mozilla.com/D41024 --HG-- extra : amend_source : 2c58011d334a35e808e93c1876080db5f83ee659
		
			
				
	
	
		
			352 lines
		
	
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			352 lines
		
	
	
	
		
			9.8 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 "IDBKeyRange.h"
 | 
						|
 | 
						|
#include "Key.h"
 | 
						|
#include "mozilla/ErrorResult.h"
 | 
						|
#include "mozilla/dom/BindingUtils.h"
 | 
						|
#include "mozilla/dom/IDBKeyRangeBinding.h"
 | 
						|
#include "mozilla/dom/indexedDB/PBackgroundIDBSharedTypes.h"
 | 
						|
 | 
						|
namespace mozilla {
 | 
						|
namespace dom {
 | 
						|
 | 
						|
using namespace mozilla::dom::indexedDB;
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
void GetKeyFromJSVal(JSContext* aCx, JS::Handle<JS::Value> aVal, Key& aKey,
 | 
						|
                     ErrorResult& aRv) {
 | 
						|
  auto result = aKey.SetFromJSVal(aCx, aVal, aRv);
 | 
						|
  if (!result.Is(Ok, aRv)) {
 | 
						|
    if (result.Is(Invalid, aRv)) {
 | 
						|
      aRv.Throw(NS_ERROR_DOM_INDEXEDDB_DATA_ERR);
 | 
						|
    }
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  if (aKey.IsUnset()) {
 | 
						|
    aRv.Throw(NS_ERROR_DOM_INDEXEDDB_DATA_ERR);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace
 | 
						|
 | 
						|
IDBKeyRange::IDBKeyRange(nsISupports* aGlobal, bool aLowerOpen, bool aUpperOpen,
 | 
						|
                         bool aIsOnly)
 | 
						|
    : mGlobal(aGlobal),
 | 
						|
      mCachedLowerVal(JS::UndefinedValue()),
 | 
						|
      mCachedUpperVal(JS::UndefinedValue()),
 | 
						|
      mLowerOpen(aLowerOpen),
 | 
						|
      mUpperOpen(aUpperOpen),
 | 
						|
      mIsOnly(aIsOnly),
 | 
						|
      mHaveCachedLowerVal(false),
 | 
						|
      mHaveCachedUpperVal(false),
 | 
						|
      mRooted(false) {
 | 
						|
  AssertIsOnOwningThread();
 | 
						|
}
 | 
						|
 | 
						|
IDBKeyRange::~IDBKeyRange() { DropJSObjects(); }
 | 
						|
 | 
						|
IDBLocaleAwareKeyRange::IDBLocaleAwareKeyRange(nsISupports* aGlobal,
 | 
						|
                                               bool aLowerOpen, bool aUpperOpen,
 | 
						|
                                               bool aIsOnly)
 | 
						|
    : IDBKeyRange(aGlobal, aLowerOpen, aUpperOpen, aIsOnly) {
 | 
						|
  AssertIsOnOwningThread();
 | 
						|
}
 | 
						|
 | 
						|
IDBLocaleAwareKeyRange::~IDBLocaleAwareKeyRange() { DropJSObjects(); }
 | 
						|
 | 
						|
// static
 | 
						|
void IDBKeyRange::FromJSVal(JSContext* aCx, JS::Handle<JS::Value> aVal,
 | 
						|
                            IDBKeyRange** aKeyRange, ErrorResult& aRv) {
 | 
						|
  MOZ_ASSERT_IF(!aCx, aVal.isUndefined());
 | 
						|
 | 
						|
  RefPtr<IDBKeyRange> keyRange;
 | 
						|
 | 
						|
  if (aVal.isNullOrUndefined()) {
 | 
						|
    // undefined and null returns no IDBKeyRange.
 | 
						|
    keyRange.forget(aKeyRange);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  JS::Rooted<JSObject*> obj(aCx, aVal.isObject() ? &aVal.toObject() : nullptr);
 | 
						|
 | 
						|
  // Unwrap an IDBKeyRange object if possible.
 | 
						|
  if (obj && NS_SUCCEEDED(UNWRAP_OBJECT(IDBKeyRange, obj, keyRange))) {
 | 
						|
    MOZ_ASSERT(keyRange);
 | 
						|
    keyRange.forget(aKeyRange);
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  // A valid key returns an 'only' IDBKeyRange.
 | 
						|
  keyRange = new IDBKeyRange(nullptr, false, false, true);
 | 
						|
  GetKeyFromJSVal(aCx, aVal, keyRange->Lower(), aRv);
 | 
						|
  if (!aRv.Failed()) {
 | 
						|
    keyRange.forget(aKeyRange);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
already_AddRefed<IDBKeyRange> IDBKeyRange::FromSerialized(
 | 
						|
    const SerializedKeyRange& aKeyRange) {
 | 
						|
  RefPtr<IDBKeyRange> keyRange =
 | 
						|
      new IDBKeyRange(nullptr, aKeyRange.lowerOpen(), aKeyRange.upperOpen(),
 | 
						|
                      aKeyRange.isOnly());
 | 
						|
  keyRange->Lower() = aKeyRange.lower();
 | 
						|
  if (!keyRange->IsOnly()) {
 | 
						|
    keyRange->Upper() = aKeyRange.upper();
 | 
						|
  }
 | 
						|
  return keyRange.forget();
 | 
						|
}
 | 
						|
 | 
						|
void IDBKeyRange::ToSerialized(SerializedKeyRange& aKeyRange) const {
 | 
						|
  aKeyRange.lowerOpen() = LowerOpen();
 | 
						|
  aKeyRange.upperOpen() = UpperOpen();
 | 
						|
  aKeyRange.isOnly() = IsOnly();
 | 
						|
 | 
						|
  aKeyRange.lower() = Lower();
 | 
						|
  if (!IsOnly()) {
 | 
						|
    aKeyRange.upper() = Upper();
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_CLASS(IDBKeyRange)
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(IDBKeyRange)
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mGlobal)
 | 
						|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(IDBKeyRange)
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRACE_JS_MEMBER_CALLBACK(mCachedLowerVal)
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_TRACE_JS_MEMBER_CALLBACK(mCachedUpperVal)
 | 
						|
NS_IMPL_CYCLE_COLLECTION_TRACE_END
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(IDBKeyRange)
 | 
						|
  NS_IMPL_CYCLE_COLLECTION_UNLINK(mGlobal)
 | 
						|
  tmp->DropJSObjects();
 | 
						|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
 | 
						|
 | 
						|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(IDBKeyRange)
 | 
						|
  NS_INTERFACE_MAP_ENTRY(nsISupports)
 | 
						|
NS_INTERFACE_MAP_END
 | 
						|
 | 
						|
NS_IMPL_CYCLE_COLLECTING_ADDREF(IDBKeyRange)
 | 
						|
NS_IMPL_CYCLE_COLLECTING_RELEASE(IDBKeyRange)
 | 
						|
 | 
						|
void IDBKeyRange::DropJSObjects() {
 | 
						|
  if (!mRooted) {
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  mCachedLowerVal.setUndefined();
 | 
						|
  mCachedUpperVal.setUndefined();
 | 
						|
  mHaveCachedLowerVal = false;
 | 
						|
  mHaveCachedUpperVal = false;
 | 
						|
  mRooted = false;
 | 
						|
  mozilla::DropJSObjects(this);
 | 
						|
}
 | 
						|
 | 
						|
bool IDBKeyRange::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto,
 | 
						|
                             JS::MutableHandle<JSObject*> aReflector) {
 | 
						|
  return IDBKeyRange_Binding::Wrap(aCx, this, aGivenProto, aReflector);
 | 
						|
}
 | 
						|
 | 
						|
bool IDBLocaleAwareKeyRange::WrapObject(
 | 
						|
    JSContext* aCx, JS::Handle<JSObject*> aGivenProto,
 | 
						|
    JS::MutableHandle<JSObject*> aReflector) {
 | 
						|
  return IDBLocaleAwareKeyRange_Binding::Wrap(aCx, this, aGivenProto,
 | 
						|
                                              aReflector);
 | 
						|
}
 | 
						|
 | 
						|
void IDBKeyRange::GetLower(JSContext* aCx, JS::MutableHandle<JS::Value> aResult,
 | 
						|
                           ErrorResult& aRv) {
 | 
						|
  AssertIsOnOwningThread();
 | 
						|
 | 
						|
  if (!mHaveCachedLowerVal) {
 | 
						|
    if (!mRooted) {
 | 
						|
      mozilla::HoldJSObjects(this);
 | 
						|
      mRooted = true;
 | 
						|
    }
 | 
						|
 | 
						|
    aRv = Lower().ToJSVal(aCx, mCachedLowerVal);
 | 
						|
    if (aRv.Failed()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    mHaveCachedLowerVal = true;
 | 
						|
  }
 | 
						|
 | 
						|
  aResult.set(mCachedLowerVal);
 | 
						|
}
 | 
						|
 | 
						|
void IDBKeyRange::GetUpper(JSContext* aCx, JS::MutableHandle<JS::Value> aResult,
 | 
						|
                           ErrorResult& aRv) {
 | 
						|
  AssertIsOnOwningThread();
 | 
						|
 | 
						|
  if (!mHaveCachedUpperVal) {
 | 
						|
    if (!mRooted) {
 | 
						|
      mozilla::HoldJSObjects(this);
 | 
						|
      mRooted = true;
 | 
						|
    }
 | 
						|
 | 
						|
    aRv = Upper().ToJSVal(aCx, mCachedUpperVal);
 | 
						|
    if (aRv.Failed()) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    mHaveCachedUpperVal = true;
 | 
						|
  }
 | 
						|
 | 
						|
  aResult.set(mCachedUpperVal);
 | 
						|
}
 | 
						|
 | 
						|
bool IDBKeyRange::Includes(JSContext* aCx, JS::Handle<JS::Value> aValue,
 | 
						|
                           ErrorResult& aRv) const {
 | 
						|
  Key key;
 | 
						|
  GetKeyFromJSVal(aCx, aValue, key, aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  MOZ_ASSERT(!(Lower().IsUnset() && Upper().IsUnset()));
 | 
						|
  MOZ_ASSERT_IF(IsOnly(), !Lower().IsUnset() && !LowerOpen() &&
 | 
						|
                              Lower() == Upper() && LowerOpen() == UpperOpen());
 | 
						|
 | 
						|
  if (!Lower().IsUnset()) {
 | 
						|
    switch (Key::CompareKeys(Lower(), key)) {
 | 
						|
      case 1:
 | 
						|
        return false;
 | 
						|
      case 0:
 | 
						|
        // Identical keys.
 | 
						|
        return !LowerOpen();
 | 
						|
      case -1:
 | 
						|
        if (IsOnly()) {
 | 
						|
          return false;
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      default:
 | 
						|
        MOZ_CRASH();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (!Upper().IsUnset()) {
 | 
						|
    switch (Key::CompareKeys(key, Upper())) {
 | 
						|
      case 1:
 | 
						|
        return false;
 | 
						|
      case 0:
 | 
						|
        // Identical keys.
 | 
						|
        return !UpperOpen();
 | 
						|
      case -1:
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
already_AddRefed<IDBKeyRange> IDBKeyRange::Only(const GlobalObject& aGlobal,
 | 
						|
                                                JS::Handle<JS::Value> aValue,
 | 
						|
                                                ErrorResult& aRv) {
 | 
						|
  RefPtr<IDBKeyRange> keyRange =
 | 
						|
      new IDBKeyRange(aGlobal.GetAsSupports(), false, false, true);
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aValue, keyRange->Lower(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  return keyRange.forget();
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
already_AddRefed<IDBKeyRange> IDBKeyRange::LowerBound(
 | 
						|
    const GlobalObject& aGlobal, JS::Handle<JS::Value> aValue, bool aOpen,
 | 
						|
    ErrorResult& aRv) {
 | 
						|
  RefPtr<IDBKeyRange> keyRange =
 | 
						|
      new IDBKeyRange(aGlobal.GetAsSupports(), aOpen, true, false);
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aValue, keyRange->Lower(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  return keyRange.forget();
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
already_AddRefed<IDBKeyRange> IDBKeyRange::UpperBound(
 | 
						|
    const GlobalObject& aGlobal, JS::Handle<JS::Value> aValue, bool aOpen,
 | 
						|
    ErrorResult& aRv) {
 | 
						|
  RefPtr<IDBKeyRange> keyRange =
 | 
						|
      new IDBKeyRange(aGlobal.GetAsSupports(), true, aOpen, false);
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aValue, keyRange->Upper(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  return keyRange.forget();
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
already_AddRefed<IDBKeyRange> IDBKeyRange::Bound(const GlobalObject& aGlobal,
 | 
						|
                                                 JS::Handle<JS::Value> aLower,
 | 
						|
                                                 JS::Handle<JS::Value> aUpper,
 | 
						|
                                                 bool aLowerOpen,
 | 
						|
                                                 bool aUpperOpen,
 | 
						|
                                                 ErrorResult& aRv) {
 | 
						|
  RefPtr<IDBKeyRange> keyRange =
 | 
						|
      new IDBKeyRange(aGlobal.GetAsSupports(), aLowerOpen, aUpperOpen, false);
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aLower, keyRange->Lower(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aUpper, keyRange->Upper(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  if (keyRange->Lower() > keyRange->Upper() ||
 | 
						|
      (keyRange->Lower() == keyRange->Upper() && (aLowerOpen || aUpperOpen))) {
 | 
						|
    aRv.Throw(NS_ERROR_DOM_INDEXEDDB_DATA_ERR);
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  return keyRange.forget();
 | 
						|
}
 | 
						|
 | 
						|
// static
 | 
						|
already_AddRefed<IDBLocaleAwareKeyRange> IDBLocaleAwareKeyRange::Bound(
 | 
						|
    const GlobalObject& aGlobal, JS::Handle<JS::Value> aLower,
 | 
						|
    JS::Handle<JS::Value> aUpper, bool aLowerOpen, bool aUpperOpen,
 | 
						|
    ErrorResult& aRv) {
 | 
						|
  RefPtr<IDBLocaleAwareKeyRange> keyRange = new IDBLocaleAwareKeyRange(
 | 
						|
      aGlobal.GetAsSupports(), aLowerOpen, aUpperOpen, false);
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aLower, keyRange->Lower(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  GetKeyFromJSVal(aGlobal.Context(), aUpper, keyRange->Upper(), aRv);
 | 
						|
  if (aRv.Failed()) {
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  if (keyRange->Lower() == keyRange->Upper() && (aLowerOpen || aUpperOpen)) {
 | 
						|
    aRv.Throw(NS_ERROR_DOM_INDEXEDDB_DATA_ERR);
 | 
						|
    return nullptr;
 | 
						|
  }
 | 
						|
 | 
						|
  return keyRange.forget();
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace dom
 | 
						|
}  // namespace mozilla
 |