/* * Copyright (C) 2011 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #pragma once #if ENABLE(INDEXED_DATABASE) #include "IndexedDB.h" #include "ThreadSafeDataBuffer.h" #include #include #include #include #include using WebCore::IndexedDB::KeyType; namespace JSC { class JSArrayBuffer; class JSArrayBufferView; } namespace WebCore { class IDBKey : public RefCounted { public: static Ref createInvalid() { return adoptRef(*new IDBKey()); } static Ref createNumber(double number) { return adoptRef(*new IDBKey(KeyType::Number, number)); } static Ref createString(const String& string) { return adoptRef(*new IDBKey(string)); } static Ref createDate(double date) { return adoptRef(*new IDBKey(KeyType::Date, date)); } static Ref createMultiEntryArray(const Vector>& array) { Vector> result; size_t sizeEstimate = 0; for (auto& key : array) { if (!key->isValid()) continue; bool skip = false; for (auto& resultKey : result) { if (key->isEqual(*resultKey)) { skip = true; break; } } if (!skip) { result.append(key); sizeEstimate += key->m_sizeEstimate; } } Ref idbKey = adoptRef(*new IDBKey(result, sizeEstimate)); ASSERT(idbKey->isValid()); return idbKey; } static Ref createArray(const Vector>& array) { size_t sizeEstimate = 0; for (auto& key : array) sizeEstimate += key->m_sizeEstimate; return adoptRef(*new IDBKey(array, sizeEstimate)); } static Ref createBinary(const ThreadSafeDataBuffer&); static Ref createBinary(JSC::JSArrayBuffer&); static Ref createBinary(JSC::JSArrayBufferView&); WEBCORE_EXPORT ~IDBKey(); KeyType type() const { return m_type; } WEBCORE_EXPORT bool isValid() const; const Vector>& array() const { ASSERT(m_type == KeyType::Array); return WTF::get>>(m_value); } const String& string() const { ASSERT(m_type == KeyType::String); return WTF::get(m_value); } double date() const { ASSERT(m_type == KeyType::Date); return WTF::get(m_value); } double number() const { ASSERT(m_type == KeyType::Number); return WTF::get(m_value); } const ThreadSafeDataBuffer& binary() const { ASSERT(m_type == KeyType::Binary); return WTF::get(m_value); } int compare(const IDBKey& other) const; bool isLessThan(const IDBKey& other) const; bool isEqual(const IDBKey& other) const; size_t sizeEstimate() const { return m_sizeEstimate; } static int compareTypes(KeyType a, KeyType b) { return b - a; } using RefCounted::ref; using RefCounted::deref; #if !LOG_DISABLED String loggingString() const; #endif private: IDBKey() : m_type(KeyType::Invalid) , m_sizeEstimate(OverheadSize) { } IDBKey(KeyType, double number); explicit IDBKey(const String& value); IDBKey(const Vector>& keyArray, size_t arraySize); explicit IDBKey(const ThreadSafeDataBuffer&); const KeyType m_type; Variant>, String, double, ThreadSafeDataBuffer> m_value; const size_t m_sizeEstimate; // Very rough estimate of minimum key size overhead. enum { OverheadSize = 16 }; }; inline int compareBinaryKeyData(const Vector& a, const Vector& b) { size_t length = std::min(a.size(), b.size()); for (size_t i = 0; i < length; ++i) { if (a[i] > b[i]) return 1; if (a[i] < b[i]) return -1; } if (a.size() == b.size()) return 0; if (a.size() > b.size()) return 1; return -1; } inline int compareBinaryKeyData(const ThreadSafeDataBuffer& a, const ThreadSafeDataBuffer& b) { auto* aData = a.data(); auto* bData = b.data(); // Covers the cases where both pointers are null as well as both pointing to the same buffer. if (aData == bData) return 0; if (aData && !bData) return 1; if (!aData && bData) return -1; return compareBinaryKeyData(*aData, *bData); } } // namespace WebCore #endif // ENABLE(INDEXED_DATABASE)