UNPKG

@react-native-firebase/app

Version:

A well tested, feature rich Firebase implementation for React Native, supporting iOS & Android. Individual module support for Admob, Analytics, Auth, Crash Reporting, Cloud Firestore, Database, Dynamic Links, Functions, Messaging (FCM), Remote Config, Sto

412 lines (398 loc) 13.8 kB
import FDBCursor from './FDBCursor.js'; import FDBCursorWithValue from './FDBCursorWithValue.js'; import FDBIndex from './FDBIndex.js'; import FDBKeyRange from './FDBKeyRange.js'; import FDBRequest from './FDBRequest.js'; import canInjectKey from './lib/canInjectKey.js'; import enforceRange from './lib/enforceRange.js'; import { ConstraintError, DataError, InvalidAccessError, InvalidStateError, NotFoundError, ReadOnlyError, TransactionInactiveError, } from './lib/errors.js'; import extractKey from './lib/extractKey.js'; import FakeDOMStringList from './lib/FakeDOMStringList.js'; import Index from './lib/Index.js'; import validateKeyPath from './lib/validateKeyPath.js'; import valueToKey from './lib/valueToKey.js'; import valueToKeyRange from './lib/valueToKeyRange.js'; const confirmActiveTransaction = objectStore => { if (objectStore._rawObjectStore.deleted) { throw new InvalidStateError(); } if (objectStore.transaction._state !== 'active') { throw new TransactionInactiveError(); } }; const buildRecordAddPut = (objectStore, value, key) => { confirmActiveTransaction(objectStore); if (objectStore.transaction.mode === 'readonly') { throw new ReadOnlyError(); } if (objectStore.keyPath !== null) { if (key !== undefined) { throw new DataError(); } } const clone = structuredClone(value); if (objectStore.keyPath !== null) { const tempKey = extractKey(objectStore.keyPath, clone); if (tempKey !== undefined) { valueToKey(tempKey); } else { if (!objectStore._rawObjectStore.keyGenerator) { throw new DataError(); } else if (!canInjectKey(objectStore.keyPath, clone)) { throw new DataError(); } } } if ( objectStore.keyPath === null && objectStore._rawObjectStore.keyGenerator === null && key === undefined ) { throw new DataError(); } if (key !== undefined) { key = valueToKey(key); } return { key, value: clone, }; }; // http://www.w3.org/TR/2015/REC-IndexedDB-20150108/#object-store class FDBObjectStore { _indexesCache = new Map(); constructor(transaction, rawObjectStore) { this._rawObjectStore = rawObjectStore; this._name = rawObjectStore.name; this.keyPath = rawObjectStore.keyPath; this.autoIncrement = rawObjectStore.autoIncrement; this.transaction = transaction; this.indexNames = new FakeDOMStringList(...Array.from(rawObjectStore.rawIndexes.keys()).sort()); } get name() { return this._name; } // http://w3c.github.io/IndexedDB/#dom-idbobjectstore-name set name(name) { const transaction = this.transaction; if (!transaction.db._runningVersionchangeTransaction) { throw new InvalidStateError(); } confirmActiveTransaction(this); name = String(name); if (name === this._name) { return; } if (this._rawObjectStore.rawDatabase.rawObjectStores.has(name)) { throw new ConstraintError(); } const oldName = this._name; const oldObjectStoreNames = [...transaction.db.objectStoreNames]; this._name = name; this._rawObjectStore.name = name; this.transaction._objectStoresCache.delete(oldName); this.transaction._objectStoresCache.set(name, this); this._rawObjectStore.rawDatabase.rawObjectStores.delete(oldName); this._rawObjectStore.rawDatabase.rawObjectStores.set(name, this._rawObjectStore); transaction.db.objectStoreNames = new FakeDOMStringList( ...Array.from(this._rawObjectStore.rawDatabase.rawObjectStores.keys()) .filter(objectStoreName => { const objectStore = this._rawObjectStore.rawDatabase.rawObjectStores.get(objectStoreName); return objectStore && !objectStore.deleted; }) .sort(), ); const oldScope = new Set(transaction._scope); const oldTransactionObjectStoreNames = [...transaction.objectStoreNames]; this.transaction._scope.delete(oldName); transaction._scope.add(name); transaction.objectStoreNames = new FakeDOMStringList(...Array.from(transaction._scope).sort()); transaction._rollbackLog.push(() => { this._name = oldName; this._rawObjectStore.name = oldName; this.transaction._objectStoresCache.delete(name); this.transaction._objectStoresCache.set(oldName, this); this._rawObjectStore.rawDatabase.rawObjectStores.delete(name); this._rawObjectStore.rawDatabase.rawObjectStores.set(oldName, this._rawObjectStore); transaction.db.objectStoreNames = new FakeDOMStringList(...oldObjectStoreNames); transaction._scope = oldScope; transaction.objectStoreNames = new FakeDOMStringList(...oldTransactionObjectStoreNames); }); } put(value, key) { if (arguments.length === 0) { throw new TypeError(); } const record = buildRecordAddPut(this, value, key); return this.transaction._execRequestAsync({ operation: this._rawObjectStore.storeRecord.bind( this._rawObjectStore, record, false, this.transaction._rollbackLog, ), source: this, }); } add(value, key) { if (arguments.length === 0) { throw new TypeError(); } const record = buildRecordAddPut(this, value, key); return this.transaction._execRequestAsync({ operation: this._rawObjectStore.storeRecord.bind( this._rawObjectStore, record, true, this.transaction._rollbackLog, ), source: this, }); } delete(key) { if (arguments.length === 0) { throw new TypeError(); } confirmActiveTransaction(this); if (this.transaction.mode === 'readonly') { throw new ReadOnlyError(); } if (!(key instanceof FDBKeyRange)) { key = valueToKey(key); } return this.transaction._execRequestAsync({ operation: this._rawObjectStore.deleteRecord.bind( this._rawObjectStore, key, this.transaction._rollbackLog, ), source: this, }); } get(key) { if (arguments.length === 0) { throw new TypeError(); } confirmActiveTransaction(this); if (!(key instanceof FDBKeyRange)) { key = valueToKey(key); } return this.transaction._execRequestAsync({ operation: this._rawObjectStore.getValue.bind(this._rawObjectStore, key), source: this, }); } // http://w3c.github.io/IndexedDB/#dom-idbobjectstore-getall getAll(query, count) { if (arguments.length > 1 && count !== undefined) { count = enforceRange(count, 'unsigned long'); } confirmActiveTransaction(this); const range = valueToKeyRange(query); return this.transaction._execRequestAsync({ operation: this._rawObjectStore.getAllValues.bind(this._rawObjectStore, range, count), source: this, }); } // http://w3c.github.io/IndexedDB/#dom-idbobjectstore-getkey getKey(key) { if (arguments.length === 0) { throw new TypeError(); } confirmActiveTransaction(this); if (!(key instanceof FDBKeyRange)) { key = valueToKey(key); } return this.transaction._execRequestAsync({ operation: this._rawObjectStore.getKey.bind(this._rawObjectStore, key), source: this, }); } // http://w3c.github.io/IndexedDB/#dom-idbobjectstore-getallkeys getAllKeys(query, count) { if (arguments.length > 1 && count !== undefined) { count = enforceRange(count, 'unsigned long'); } confirmActiveTransaction(this); const range = valueToKeyRange(query); return this.transaction._execRequestAsync({ operation: this._rawObjectStore.getAllKeys.bind(this._rawObjectStore, range, count), source: this, }); } clear() { confirmActiveTransaction(this); if (this.transaction.mode === 'readonly') { throw new ReadOnlyError(); } return this.transaction._execRequestAsync({ operation: this._rawObjectStore.clear.bind( this._rawObjectStore, this.transaction._rollbackLog, ), source: this, }); } openCursor(range, direction) { confirmActiveTransaction(this); if (range === null) { range = undefined; } if (range !== undefined && !(range instanceof FDBKeyRange)) { range = FDBKeyRange.only(valueToKey(range)); } const request = new FDBRequest(); request.source = this; request.transaction = this.transaction; const cursor = new FDBCursorWithValue(this, range, direction, request); return this.transaction._execRequestAsync({ operation: cursor._iterate.bind(cursor), request, source: this, }); } openKeyCursor(range, direction) { confirmActiveTransaction(this); if (range === null) { range = undefined; } if (range !== undefined && !(range instanceof FDBKeyRange)) { range = FDBKeyRange.only(valueToKey(range)); } const request = new FDBRequest(); request.source = this; request.transaction = this.transaction; const cursor = new FDBCursor(this, range, direction, request, true); return this.transaction._execRequestAsync({ operation: cursor._iterate.bind(cursor), request, source: this, }); } // tslint:-next-line max-line-length // http://www.w3.org/TR/2015/REC-IndexedDB-20150108/#widl-IDBObjectStore-createIndex-IDBIndex-DOMString-name-DOMString-sequence-DOMString--keyPath-IDBIndexParameters-optionalParameters createIndex(name, keyPath, optionalParameters = {}) { if (arguments.length < 2) { throw new TypeError(); } const multiEntry = optionalParameters.multiEntry !== undefined ? optionalParameters.multiEntry : false; const unique = optionalParameters.unique !== undefined ? optionalParameters.unique : false; if (this.transaction.mode !== 'versionchange') { throw new InvalidStateError(); } confirmActiveTransaction(this); if (this.indexNames.contains(name)) { throw new ConstraintError(); } validateKeyPath(keyPath); if (Array.isArray(keyPath) && multiEntry) { throw new InvalidAccessError(); } // The index that is requested to be created can contain constraints on the data allowed in the index's // referenced object store, such as requiring uniqueness of the values referenced by the index's keyPath. If the // referenced object store already contains data which violates these constraints, this MUST NOT cause the // implementation of createIndex to throw an exception or affect what it returns. The implementation MUST still // create and return an IDBIndex object. Instead the implementation must queue up an operation to abort the // "versionchange" transaction which was used for the createIndex call. const indexNames = [...this.indexNames]; this.transaction._rollbackLog.push(() => { const index2 = this._rawObjectStore.rawIndexes.get(name); if (index2) { index2.deleted = true; } this.indexNames = new FakeDOMStringList(...indexNames); this._rawObjectStore.rawIndexes.delete(name); }); const index = new Index(this._rawObjectStore, name, keyPath, multiEntry, unique); this.indexNames._push(name); this.indexNames._sort(); this._rawObjectStore.rawIndexes.set(name, index); index.initialize(this.transaction); // This is async by design return new FDBIndex(this, index); } // https://w3c.github.io/IndexedDB/#dom-idbobjectstore-index index(name) { if (arguments.length === 0) { throw new TypeError(); } if (this._rawObjectStore.deleted || this.transaction._state === 'finished') { throw new InvalidStateError(); } const index = this._indexesCache.get(name); if (index !== undefined) { return index; } const rawIndex = this._rawObjectStore.rawIndexes.get(name); if (!this.indexNames.contains(name) || rawIndex === undefined) { throw new NotFoundError(); } const index2 = new FDBIndex(this, rawIndex); this._indexesCache.set(name, index2); return index2; } deleteIndex(name) { if (arguments.length === 0) { throw new TypeError(); } if (this.transaction.mode !== 'versionchange') { throw new InvalidStateError(); } confirmActiveTransaction(this); const rawIndex = this._rawObjectStore.rawIndexes.get(name); if (rawIndex === undefined) { throw new NotFoundError(); } this.transaction._rollbackLog.push(() => { rawIndex.deleted = false; this._rawObjectStore.rawIndexes.set(name, rawIndex); this.indexNames._push(name); this.indexNames._sort(); }); this.indexNames = new FakeDOMStringList( ...Array.from(this.indexNames).filter(indexName => { return indexName !== name; }), ); rawIndex.deleted = true; // Not sure if this is supposed to happen synchronously this.transaction._execRequestAsync({ operation: () => { const rawIndex2 = this._rawObjectStore.rawIndexes.get(name); // Hack in case another index is given this name before this async request is processed. It'd be better // to have a real unique ID for each index. if (rawIndex === rawIndex2) { this._rawObjectStore.rawIndexes.delete(name); } }, source: this, }); } // http://www.w3.org/TR/2015/REC-IndexedDB-20150108/#widl-IDBObjectStore-count-IDBRequest-any-key count(key) { confirmActiveTransaction(this); if (key === null) { key = undefined; } if (key !== undefined && !(key instanceof FDBKeyRange)) { key = FDBKeyRange.only(valueToKey(key)); } return this.transaction._execRequestAsync({ operation: () => { return this._rawObjectStore.count(key); }, source: this, }); } toString() { return '[object IDBObjectStore]'; } } export default FDBObjectStore;