blob: 2de29d451c79b437f1e22cb7a64fec3677c74d20 [file] [log] [blame]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_LEVELDB_CODING_H_
#define CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_LEVELDB_CODING_H_
#include <stddef.h>
#include <stdint.h>
#include <memory>
#include <string>
#include <string_view>
#include <utility>
#include <vector>
#include "content/common/content_export.h"
#include "third_party/blink/public/common/indexeddb/indexeddb_key.h"
#include "third_party/blink/public/common/indexeddb/indexeddb_key_path.h"
namespace content::indexed_db {
// 0 - Initial version.
// 1 - Adds UserIntVersion to DatabaseMetaData.
// 2 - Adds DataVersion to to global metadata.
// 3 - Adds metadata needed for blob support.
// 4 - Adds size & last_modified to 'file' blob_info encodings.
// 5 - One time verification that blob files exist on disk.
inline constexpr int64_t kLatestKnownSchemaVersion = 5;
// Migration from version 2 to 3 occurred in 2014, and migration to version 4
// began in early 2020, so we currently continue to support schema that are as
// old as 2014.
inline constexpr int64_t kEarliestSupportedSchemaVersion = 3;
inline constexpr unsigned char kMinimumIndexId = 30;
CONTENT_EXPORT std::string MaxIDBKey();
CONTENT_EXPORT std::string MinIDBKey();
// DatabaseId, BlobNumber
using BlobJournalEntryType = std::pair<int64_t, int64_t>;
using BlobJournalType = std::vector<BlobJournalEntryType>;
CONTENT_EXPORT void EncodeByte(unsigned char value, std::string* into);
CONTENT_EXPORT void EncodeBool(bool value, std::string* into);
// Unlike EncodeVarInt, this is a 'dumb' implementation of a variable int
// encoder. It writes, little-endian', until there are no more '1' bits in the
// number. The Decoder must know how to calculate the size of the encoded int,
// typically by having this reside at the end of the value or key.
CONTENT_EXPORT void EncodeInt(int64_t value, std::string* into);
CONTENT_EXPORT void EncodeString(const std::u16string& value,
std::string* into);
CONTENT_EXPORT void EncodeStringWithLength(const std::u16string& value,
std::string* into);
CONTENT_EXPORT void EncodeBinary(const std::string& value, std::string* into);
CONTENT_EXPORT void EncodeBinary(base::span<const uint8_t> value,
std::string* into);
CONTENT_EXPORT void EncodeDouble(double value, std::string* into);
// This version will CHECK if encoding fails. This is generally preferred to
// handling an error.
CONTENT_EXPORT void EncodeIDBKey(const blink::IndexedDBKey& value,
std::string* into);
// This version will return `true` on success. It exists so tests can run
// without crashing.
[[nodiscard]] CONTENT_EXPORT bool MaybeEncodeIDBKey(
const blink::IndexedDBKey& value,
std::string* into);
// This function creates a byte stream that can be directly compared to other
// byte streams on a byte-by-byte basis and retain semantic ordering. This
// enables the value to be stored as a SQLite blob without a specialized
// collation operation. Unlike `EncodeIDBKey`, which makes use of length bytes,
// this operation re-encodes variable-length values in a way that supports
// sentinels.
CONTENT_EXPORT std::string EncodeSortableIDBKey(
const blink::IndexedDBKey& value);
CONTENT_EXPORT void EncodeIDBKeyPath(const blink::IndexedDBKeyPath& value,
std::string* into);
CONTENT_EXPORT void EncodeBlobJournal(const BlobJournalType& journal,
std::string* into);
[[nodiscard]] CONTENT_EXPORT bool DecodeByte(std::string_view* slice,
unsigned char* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeBool(std::string_view* slice,
bool* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeInt(std::string_view* slice,
int64_t* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeString(std::string_view* slice,
std::u16string* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeStringWithLength(
std::string_view* slice,
std::u16string* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeBinary(std::string_view* slice,
std::string* value);
// The returned span is only valid as long as the date behind |slice| is
// still valid.
[[nodiscard]] CONTENT_EXPORT bool DecodeBinary(
std::string_view* slice,
base::span<const uint8_t>* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeDouble(std::string_view* slice,
double* value);
// Will return an invalid key if deserialization fails.
[[nodiscard]] CONTENT_EXPORT blink::IndexedDBKey DecodeIDBKey(
std::string_view* slice);
// Will return an invalid key if deserialization fails.
[[nodiscard]] CONTENT_EXPORT blink::IndexedDBKey DecodeSortableIDBKey(
std::string_view serialized);
[[nodiscard]] CONTENT_EXPORT bool DecodeIDBKeyPath(
std::string_view* slice,
blink::IndexedDBKeyPath* value);
[[nodiscard]] CONTENT_EXPORT bool DecodeBlobJournal(std::string_view* slice,
BlobJournalType* journal);
CONTENT_EXPORT int CompareEncodedStringsWithLength(std::string_view* slice1,
std::string_view* slice2,
bool* ok);
[[nodiscard]] CONTENT_EXPORT bool ExtractEncodedIDBKey(std::string_view* slice,
std::string* result);
CONTENT_EXPORT int CompareEncodedIDBKeys(std::string_view* slice1,
std::string_view* slice2,
bool* ok);
CONTENT_EXPORT int Compare(std::string_view a,
std::string_view b,
bool index_keys);
CONTENT_EXPORT int CompareKeys(std::string_view a, std::string_view b);
CONTENT_EXPORT int CompareIndexKeys(std::string_view a, std::string_view b);
// Logging support.
std::string IndexedDBKeyToDebugString(std::string_view key);
// TODO(dmurph): Modify all decoding methods to return something more sensible,
// as it is not obvious that they modify the input slice to remove the decoded
// bit. https://crbug.com/922225
class KeyPrefix {
public:
// These are serialized to disk; any new items must be appended, and none can
// be deleted.
enum Type {
GLOBAL_METADATA = 0,
DATABASE_METADATA = 1,
OBJECT_STORE_DATA = 2,
EXISTS_ENTRY = 3,
INDEX_DATA = 4,
INVALID_TYPE = 5,
BLOB_ENTRY = 6
};
static const size_t kMaxDatabaseIdSizeBits = 3;
static const size_t kMaxObjectStoreIdSizeBits = 3;
static const size_t kMaxIndexIdSizeBits = 2;
static const size_t kMaxDatabaseIdSizeBytes = 1ULL
<< kMaxDatabaseIdSizeBits; // 8
static const size_t kMaxObjectStoreIdSizeBytes =
1ULL << kMaxObjectStoreIdSizeBits; // 8
static const size_t kMaxIndexIdSizeBytes = 1ULL << kMaxIndexIdSizeBits; // 4
static const size_t kMaxDatabaseIdBits =
kMaxDatabaseIdSizeBytes * 8 - 1; // 63
static const size_t kMaxObjectStoreIdBits =
kMaxObjectStoreIdSizeBytes * 8 - 1; // 63
static const size_t kMaxIndexIdBits = kMaxIndexIdSizeBytes * 8 - 1; // 31
static const int64_t kMaxDatabaseId =
(1ULL << kMaxDatabaseIdBits) - 1; // max signed int64_t
static const int64_t kMaxObjectStoreId =
(1ULL << kMaxObjectStoreIdBits) - 1; // max signed int64_t
static const int64_t kMaxIndexId =
(1ULL << kMaxIndexIdBits) - 1; // max signed int32_t
CONTENT_EXPORT static const int64_t kInvalidId;
KeyPrefix();
explicit KeyPrefix(int64_t database_id);
KeyPrefix(int64_t database_id, int64_t object_store_id);
KeyPrefix(int64_t database_id, int64_t object_store_id, int64_t index_id);
static KeyPrefix CreateWithSpecialIndex(int64_t database_id,
int64_t object_store_id,
int64_t index_id);
static bool Decode(std::string_view* slice, KeyPrefix* result);
std::string Encode() const;
static std::string EncodeEmpty();
int Compare(const KeyPrefix& other) const;
CONTENT_EXPORT static bool IsValidDatabaseId(int64_t database_id);
static bool IsValidObjectStoreId(int64_t index_id);
static bool IsValidIndexId(int64_t index_id);
static bool ValidIds(int64_t database_id,
int64_t object_store_id,
int64_t index_id) {
return IsValidDatabaseId(database_id) &&
IsValidObjectStoreId(object_store_id) && IsValidIndexId(index_id);
}
static bool ValidIds(int64_t database_id, int64_t object_store_id) {
return IsValidDatabaseId(database_id) &&
IsValidObjectStoreId(object_store_id);
}
std::string DebugString();
Type type() const;
int64_t database_id_;
int64_t object_store_id_;
int64_t index_id_;
private:
// Special constructor for CreateWithSpecialIndex()
KeyPrefix(enum Type,
int64_t database_id,
int64_t object_store_id,
int64_t index_id);
static std::string EncodeInternal(int64_t database_id,
int64_t object_store_id,
int64_t index_id);
};
class SchemaVersionKey {
public:
CONTENT_EXPORT static std::string Encode();
};
class MaxDatabaseIdKey {
public:
CONTENT_EXPORT static std::string Encode();
};
class DataVersionKey {
public:
CONTENT_EXPORT static std::string Encode();
};
class RecoveryBlobJournalKey {
public:
static std::string Encode();
};
class ActiveBlobJournalKey {
public:
static std::string Encode();
};
class EarliestSweepKey {
public:
static std::string Encode();
};
class EarliestCompactionKey {
public:
static std::string Encode();
};
class ScopesPrefix {
public:
CONTENT_EXPORT static std::vector<uint8_t> Encode();
};
class DatabaseFreeListKey {
public:
DatabaseFreeListKey();
static bool Decode(std::string_view* slice, DatabaseFreeListKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id);
static CONTENT_EXPORT std::string EncodeMaxKey();
int64_t DatabaseId() const;
int Compare(const DatabaseFreeListKey& other) const;
std::string DebugString() const;
private:
int64_t database_id_;
};
class DatabaseNameKey {
public:
static bool Decode(std::string_view* slice, DatabaseNameKey* result);
CONTENT_EXPORT static std::string Encode(const std::string& origin_identifier,
const std::u16string& database_name);
static std::string EncodeMinKeyForOrigin(
const std::string& origin_identifier);
static std::string EncodeStopKeyForOrigin(
const std::string& origin_identifier);
std::u16string origin() const { return origin_; }
std::u16string database_name() const { return database_name_; }
int Compare(const DatabaseNameKey& other);
std::string DebugString() const;
private:
std::u16string origin_; // TODO(jsbell): Store encoded strings, or just
// pointers.
std::u16string database_name_;
};
class DatabaseMetaDataKey {
public:
enum MetaDataType {
ORIGIN_NAME = 0,
DATABASE_NAME = 1,
USER_STRING_VERSION = 2, // Obsolete
MAX_OBJECT_STORE_ID = 3,
USER_VERSION = 4,
BLOB_KEY_GENERATOR_CURRENT_NUMBER = 5,
MAX_SIMPLE_METADATA_TYPE = 6
};
CONTENT_EXPORT static const int64_t kAllBlobsNumber;
CONTENT_EXPORT static const int64_t kBlobNumberGeneratorInitialNumber;
// All keys <= 0 are invalid. This one's just a convenient example.
static const int64_t kInvalidBlobNumber;
CONTENT_EXPORT static bool IsValidBlobNumber(int64_t blob_number);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
MetaDataType type);
};
class ObjectStoreMetaDataKey {
public:
enum MetaDataType {
NAME = 0,
KEY_PATH = 1,
AUTO_INCREMENT = 2,
EVICTABLE = 3,
LAST_VERSION = 4,
MAX_INDEX_ID = 5,
HAS_KEY_PATH = 6,
KEY_GENERATOR_CURRENT_NUMBER = 7
};
// From the IndexedDB specification.
static const int64_t kKeyGeneratorInitialNumber;
ObjectStoreMetaDataKey();
static bool Decode(std::string_view* slice, ObjectStoreMetaDataKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
unsigned char meta_data_type);
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id);
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id,
int64_t object_store_id);
int64_t ObjectStoreId() const;
unsigned char MetaDataType() const;
int Compare(const ObjectStoreMetaDataKey& other);
std::string DebugString() const;
private:
int64_t object_store_id_;
unsigned char meta_data_type_;
};
class IndexMetaDataKey {
public:
enum MetaDataType { NAME = 0, UNIQUE = 1, KEY_PATH = 2, MULTI_ENTRY = 3 };
IndexMetaDataKey();
static bool Decode(std::string_view* slice, IndexMetaDataKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
int64_t index_id,
unsigned char meta_data_type);
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id,
int64_t object_store_id);
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id,
int64_t object_store_id,
int64_t index_id);
int Compare(const IndexMetaDataKey& other);
std::string DebugString() const;
int64_t IndexId() const;
unsigned char meta_data_type() const { return meta_data_type_; }
private:
int64_t object_store_id_;
int64_t index_id_;
unsigned char meta_data_type_;
};
class ObjectStoreFreeListKey {
public:
ObjectStoreFreeListKey();
static bool Decode(std::string_view* slice, ObjectStoreFreeListKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id);
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id);
int64_t ObjectStoreId() const;
int Compare(const ObjectStoreFreeListKey& other);
std::string DebugString() const;
private:
int64_t object_store_id_;
};
class IndexFreeListKey {
public:
IndexFreeListKey();
static bool Decode(std::string_view* slice, IndexFreeListKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
int64_t index_id);
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id,
int64_t object_store_id);
int Compare(const IndexFreeListKey& other);
int64_t ObjectStoreId() const;
int64_t IndexId() const;
std::string DebugString() const;
private:
int64_t object_store_id_;
int64_t index_id_;
};
class ObjectStoreNamesKey {
public:
// TODO(jsbell): We never use this to look up object store ids,
// because a mapping is kept in the Database. Can the
// mapping become unreliable? Can we remove this?
static bool Decode(std::string_view* slice, ObjectStoreNamesKey* result);
CONTENT_EXPORT static std::string Encode(
int64_t database_id,
const std::u16string& object_store_name);
int Compare(const ObjectStoreNamesKey& other);
std::string DebugString() const;
std::u16string object_store_name() const { return object_store_name_; }
private:
// TODO(jsbell): Store the encoded string, or just pointers to it.
std::u16string object_store_name_;
};
class IndexNamesKey {
public:
IndexNamesKey();
// TODO(jsbell): We never use this to look up index ids, because a mapping
// is kept at a higher level.
static bool Decode(std::string_view* slice, IndexNamesKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
const std::u16string& index_name);
int Compare(const IndexNamesKey& other);
std::string DebugString() const;
std::u16string index_name() const { return index_name_; }
private:
int64_t object_store_id_;
std::u16string index_name_;
};
class ObjectStoreDataKey {
public:
static const int64_t kSpecialIndexNumber;
ObjectStoreDataKey();
~ObjectStoreDataKey();
static bool Decode(std::string_view* slice, ObjectStoreDataKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
const std::string& encoded_user_key);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
const blink::IndexedDBKey& user_key);
std::string DebugString() const;
blink::IndexedDBKey DecodeUserKey() const;
private:
std::string encoded_user_key_;
};
class ExistsEntryKey {
public:
ExistsEntryKey();
ExistsEntryKey(const ExistsEntryKey&) = delete;
ExistsEntryKey& operator=(const ExistsEntryKey&) = delete;
~ExistsEntryKey();
static bool Decode(std::string_view* slice, ExistsEntryKey* result);
CONTENT_EXPORT static std::string Encode(int64_t database_id,
int64_t object_store_id,
const std::string& encoded_key);
static std::string Encode(int64_t database_id,
int64_t object_store_id,
const blink::IndexedDBKey& user_key);
std::string DebugString() const;
blink::IndexedDBKey DecodeUserKey() const;
private:
static const int64_t kSpecialIndexNumber;
std::string encoded_user_key_;
};
class CONTENT_EXPORT BlobEntryKey {
public:
BlobEntryKey() : database_id_(0), object_store_id_(0) {}
static bool Decode(std::string_view* slice, BlobEntryKey* result);
static bool FromObjectStoreDataKey(std::string_view* slice,
BlobEntryKey* result);
static std::string ReencodeToObjectStoreDataKey(std::string_view* slice);
static std::string EncodeMinKeyForObjectStore(int64_t database_id,
int64_t object_store_id);
static std::string EncodeStopKeyForObjectStore(int64_t database_id,
int64_t object_store_id);
static std::string Encode(int64_t database_id,
int64_t object_store_id,
const blink::IndexedDBKey& user_key);
std::string Encode() const;
std::string DebugString() const;
int64_t database_id() const { return database_id_; }
int64_t object_store_id() const { return object_store_id_; }
private:
static const int64_t kSpecialIndexNumber;
static std::string Encode(int64_t database_id,
int64_t object_store_id,
const std::string& encoded_user_key);
int64_t database_id_;
int64_t object_store_id_;
// This is the user's ObjectStoreDataKey, not the BlobEntryKey itself.
std::string encoded_user_key_;
};
class IndexDataKey {
public:
CONTENT_EXPORT IndexDataKey();
CONTENT_EXPORT IndexDataKey(IndexDataKey&& other);
IndexDataKey(const IndexDataKey&) = delete;
IndexDataKey& operator=(const IndexDataKey&) = delete;
CONTENT_EXPORT ~IndexDataKey();
CONTENT_EXPORT static bool Decode(std::string_view* slice,
IndexDataKey* result);
CONTENT_EXPORT static std::string Encode(
int64_t database_id,
int64_t object_store_id,
int64_t index_id,
const std::string_view encoded_user_key,
const std::string_view encoded_primary_key,
int64_t sequence_number);
static std::string Encode(int64_t database_id,
int64_t object_store_id,
int64_t index_id,
const blink::IndexedDBKey& user_key);
CONTENT_EXPORT static std::string Encode(
int64_t database_id,
int64_t object_store_id,
int64_t index_id,
const blink::IndexedDBKey& user_key,
const blink::IndexedDBKey& user_primary_key);
CONTENT_EXPORT static std::string EncodeMinKey(int64_t database_id,
int64_t object_store_id,
int64_t index_id);
// An index's keys are guaranteed to fall in [EncodeMinKey(), EncodeMaxKey()]
CONTENT_EXPORT static std::string EncodeMaxKey(int64_t database_id,
int64_t object_store_id,
int64_t index_id);
int64_t DatabaseId() const;
int64_t ObjectStoreId() const;
int64_t IndexId() const;
blink::IndexedDBKey DecodeUserKey() const;
blink::IndexedDBKey DecodePrimaryKey() const;
CONTENT_EXPORT std::string Encode() const;
std::string DebugString() const;
private:
int64_t database_id_;
int64_t object_store_id_;
int64_t index_id_;
std::string encoded_user_key_;
std::string encoded_primary_key_;
int64_t sequence_number_;
};
} // namespace content::indexed_db
#endif // CONTENT_BROWSER_INDEXED_DB_INDEXED_DB_LEVELDB_CODING_H_