345 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * Copyright (c) 2015-present, Facebook, Inc.
 | |
|  *
 | |
|  * This source code is licensed under the MIT license found in the
 | |
|  * LICENSE file in the root directory of this source tree.
 | |
|  *
 | |
|  * @providesModule AsyncStorage
 | |
|  * @noflow
 | |
|  * @flow-weak
 | |
|  * @jsdoc
 | |
|  */
 | |
| 'use strict';
 | |
| 
 | |
| const NativeModules = require('NativeModules');
 | |
| 
 | |
| // Use RocksDB if available, then SQLite, then file storage.
 | |
| const RCTAsyncStorage = NativeModules.AsyncRocksDBStorage ||
 | |
|   NativeModules.AsyncSQLiteDBStorage ||
 | |
|   NativeModules.AsyncLocalStorage;
 | |
| 
 | |
| /**
 | |
|  * `AsyncStorage` is a simple, unencrypted, asynchronous, persistent, key-value
 | |
|  * storage system that is global to the app.  It should be used instead of
 | |
|  * LocalStorage.
 | |
|  *
 | |
|  * See http://facebook.github.io/react-native/docs/asyncstorage.html
 | |
|  */
 | |
| var AsyncStorage = {
 | |
|   _getRequests: ([]: Array<any>),
 | |
|   _getKeys: ([]: Array<string>),
 | |
|   _immediate: (null: ?number),
 | |
| 
 | |
|   /**
 | |
|    * Fetches an item for a `key` and invokes a callback upon completion.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#getitem
 | |
|    */
 | |
|   getItem: function(
 | |
|     key: string,
 | |
|     callback?: ?(error: ?Error, result: ?string) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiGet([key], function(errors, result) {
 | |
|         // Unpack result to get value from [[key,value]]
 | |
|         var value = (result && result[0] && result[0][1]) ? result[0][1] : null;
 | |
|         var errs = convertErrors(errors);
 | |
|         callback && callback(errs && errs[0], value);
 | |
|         if (errs) {
 | |
|           reject(errs[0]);
 | |
|         } else {
 | |
|           resolve(value);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Sets the value for a `key` and invokes a callback upon completion.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#setitem
 | |
|    */
 | |
|   setItem: function(
 | |
|     key: string,
 | |
|     value: string,
 | |
|     callback?: ?(error: ?Error) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiSet([[key,value]], function(errors) {
 | |
|         var errs = convertErrors(errors);
 | |
|         callback && callback(errs && errs[0]);
 | |
|         if (errs) {
 | |
|           reject(errs[0]);
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Removes an item for a `key` and invokes a callback upon completion.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#removeitem
 | |
|    */
 | |
|   removeItem: function(
 | |
|     key: string,
 | |
|     callback?: ?(error: ?Error) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiRemove([key], function(errors) {
 | |
|         var errs = convertErrors(errors);
 | |
|         callback && callback(errs && errs[0]);
 | |
|         if (errs) {
 | |
|           reject(errs[0]);
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Merges an existing `key` value with an input value, assuming both values
 | |
|    * are stringified JSON.
 | |
|    *
 | |
|    * **NOTE:** This is not supported by all native implementations.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#mergeitem
 | |
|    */
 | |
|   mergeItem: function(
 | |
|     key: string,
 | |
|     value: string,
 | |
|     callback?: ?(error: ?Error) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiMerge([[key,value]], function(errors) {
 | |
|         var errs = convertErrors(errors);
 | |
|         callback && callback(errs && errs[0]);
 | |
|         if (errs) {
 | |
|           reject(errs[0]);
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Erases *all* `AsyncStorage` for all clients, libraries, etc. You probably
 | |
|    * don't want to call this; use `removeItem` or `multiRemove` to clear only
 | |
|    * your app's keys.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#clear
 | |
|    */
 | |
|   clear: function(callback?: ?(error: ?Error) => void): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.clear(function(error) {
 | |
|         callback && callback(convertError(error));
 | |
|         if (error && convertError(error)){
 | |
|           reject(convertError(error));
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Gets *all* keys known to your app; for all callers, libraries, etc.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#getallkeys
 | |
|    */
 | |
|   getAllKeys: function(callback?: ?(error: ?Error, keys: ?Array<string>) => void): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.getAllKeys(function(error, keys) {
 | |
|         callback && callback(convertError(error), keys);
 | |
|         if (error) {
 | |
|           reject(convertError(error));
 | |
|         } else {
 | |
|           resolve(keys);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * The following batched functions are useful for executing a lot of
 | |
|    * operations at once, allowing for native optimizations and provide the
 | |
|    * convenience of a single callback after all operations are complete.
 | |
|    *
 | |
|    * These functions return arrays of errors, potentially one for every key.
 | |
|    * For key-specific errors, the Error object will have a key property to
 | |
|    * indicate which key caused the error.
 | |
|    */
 | |
| 
 | |
|   /**
 | |
|    * Flushes any pending requests using a single batch call to get the data.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#flushgetrequests
 | |
|    * */
 | |
|   flushGetRequests: function(): void {
 | |
|     const getRequests = this._getRequests;
 | |
|     const getKeys = this._getKeys;
 | |
| 
 | |
|     this._getRequests = [];
 | |
|     this._getKeys = [];
 | |
| 
 | |
|     RCTAsyncStorage.multiGet(getKeys, function(errors, result) {
 | |
|       // Even though the runtime complexity of this is theoretically worse vs if we used a map,
 | |
|       // it's much, much faster in practice for the data sets we deal with (we avoid
 | |
|       // allocating result pair arrays). This was heavily benchmarked.
 | |
|       //
 | |
|       // Is there a way to avoid using the map but fix the bug in this breaking test?
 | |
|       // https://github.com/facebook/react-native/commit/8dd8ad76579d7feef34c014d387bf02065692264
 | |
|       const map = {};
 | |
|       result && result.forEach(([key, value]) => { map[key] = value; return value; });
 | |
|       const reqLength = getRequests.length;
 | |
|       for (let i = 0; i < reqLength; i++) {
 | |
|         const request = getRequests[i];
 | |
|         const requestKeys = request.keys;
 | |
|         const requestResult = requestKeys.map(key => [key, map[key]]);
 | |
|         request.callback && request.callback(null, requestResult);
 | |
|         request.resolve && request.resolve(requestResult);
 | |
|       }
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * This allows you to batch the fetching of items given an array of `key`
 | |
|    * inputs. Your callback will be invoked with an array of corresponding
 | |
|    * key-value pairs found.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#multiget
 | |
|    */
 | |
|   multiGet: function(
 | |
|     keys: Array<string>,
 | |
|     callback?: ?(errors: ?Array<Error>, result: ?Array<Array<string>>) => void
 | |
|   ): Promise {
 | |
|     if (!this._immediate) {
 | |
|       this._immediate = setImmediate(() => {
 | |
|         this._immediate = null;
 | |
|         this.flushGetRequests();
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     var getRequest = {
 | |
|       keys: keys,
 | |
|       callback: callback,
 | |
|       // do we need this?
 | |
|       keyIndex: this._getKeys.length,
 | |
|       resolve: null,
 | |
|       reject: null,
 | |
|     };
 | |
| 
 | |
|     var promiseResult = new Promise((resolve, reject) => {
 | |
|       getRequest.resolve = resolve;
 | |
|       getRequest.reject = reject;
 | |
|     });
 | |
| 
 | |
|     this._getRequests.push(getRequest);
 | |
|     // avoid fetching duplicates
 | |
|     keys.forEach(key => {
 | |
|       if (this._getKeys.indexOf(key) === -1) {
 | |
|         this._getKeys.push(key);
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     return promiseResult;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Use this as a batch operation for storing multiple key-value pairs. When
 | |
|    * the operation completes you'll get a single callback with any errors.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#multiset
 | |
|    */
 | |
|   multiSet: function(
 | |
|     keyValuePairs: Array<Array<string>>,
 | |
|     callback?: ?(errors: ?Array<Error>) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiSet(keyValuePairs, function(errors) {
 | |
|         var error = convertErrors(errors);
 | |
|         callback && callback(error);
 | |
|         if (error) {
 | |
|           reject(error);
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Call this to batch the deletion of all keys in the `keys` array.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#multiremove
 | |
|    */
 | |
|   multiRemove: function(
 | |
|     keys: Array<string>,
 | |
|     callback?: ?(errors: ?Array<Error>) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiRemove(keys, function(errors) {
 | |
|         var error = convertErrors(errors);
 | |
|         callback && callback(error);
 | |
|         if (error) {
 | |
|           reject(error);
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Batch operation to merge in existing and new values for a given set of
 | |
|    * keys. This assumes that the values are stringified JSON.
 | |
|    *
 | |
|    * **NOTE**: This is not supported by all native implementations.
 | |
|    *
 | |
|    * See http://facebook.github.io/react-native/docs/asyncstorage.html#multimerge
 | |
|    */
 | |
|   multiMerge: function(
 | |
|     keyValuePairs: Array<Array<string>>,
 | |
|     callback?: ?(errors: ?Array<Error>) => void
 | |
|   ): Promise {
 | |
|     return new Promise((resolve, reject) => {
 | |
|       RCTAsyncStorage.multiMerge(keyValuePairs, function(errors) {
 | |
|         var error = convertErrors(errors);
 | |
|         callback && callback(error);
 | |
|         if (error) {
 | |
|           reject(error);
 | |
|         } else {
 | |
|           resolve(null);
 | |
|         }
 | |
|       });
 | |
|     });
 | |
|   },
 | |
| };
 | |
| 
 | |
| // Not all native implementations support merge.
 | |
| if (!RCTAsyncStorage.multiMerge) {
 | |
|   delete AsyncStorage.mergeItem;
 | |
|   delete AsyncStorage.multiMerge;
 | |
| }
 | |
| 
 | |
| function convertErrors(errs) {
 | |
|   if (!errs) {
 | |
|     return null;
 | |
|   }
 | |
|   return (Array.isArray(errs) ? errs : [errs]).map((e) => convertError(e));
 | |
| }
 | |
| 
 | |
| function convertError(error) {
 | |
|   if (!error) {
 | |
|     return null;
 | |
|   }
 | |
|   var out = new Error(error.message);
 | |
|   out.key = error.key; // flow doesn't like this :(
 | |
|   return out;
 | |
| }
 | |
| 
 | |
| module.exports = AsyncStorage;
 | 
