diff --git a/.babelrc b/.babelrc index 6c1e0ce..a9ce136 100644 --- a/.babelrc +++ b/.babelrc @@ -1,3 +1,3 @@ { "presets": ["react-native"] -} \ No newline at end of file +} diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 0000000..0f09989 --- /dev/null +++ b/.editorconfig @@ -0,0 +1,10 @@ +# editorconfig.org +root = true + +[*] +indent_style = space +indent_size = 2 +end_of_line = lf +charset = utf-8 +trim_trailing_whitespace = true +insert_final_newline = true diff --git a/.eslintrc b/.eslintrc new file mode 100644 index 0000000..c9d71b9 --- /dev/null +++ b/.eslintrc @@ -0,0 +1,40 @@ +{ + "extends": "airbnb", + "parser": "babel-eslint", + "ecmaFeatures": { + "jsx": true + }, + "plugins": [ + "flowtype" + ], + "env": { + "es6": true, + "jasmine": true + }, + "parserOptions": { + "ecmaFeatures": { + "experimentalObjectRestSpread": true + } + }, + "rules": { + "class-methods-use-this": 0, + "no-underscore-dangle": 0, + "no-use-before-define": 0, + "arrow-body-style": 0, + "import/prefer-default-export": 0, + "radix": 0, + "new-cap": 0, + "max-len": 0, + "no-continue": 0, + "no-console": 0, + "global-require": 0, + "import/extensions": 0, + "import/no-unresolved": 0, + "import/no-extraneous-dependencies": 0, + "react/jsx-filename-extension": 0 + }, + "globals": { + "__DEV__": true, + "window": true + } +} diff --git a/.flowconfig b/.flowconfig new file mode 100644 index 0000000..f43b490 --- /dev/null +++ b/.flowconfig @@ -0,0 +1,105 @@ +[ignore] + + +# Some modules have their own node_modules with overlap +.*/node_modules/node-haste/.* + + +# React Native problems +.*/node_modules/react-native/Libraries/Animated/src/AnimatedInterpolation.js +.*/node_modules/react-native/Libraries/Animated/src/Interpolation.js +.*/node_modules/react-native/Libraries/BugReporting/dumpReactTree.js +.*/node_modules/react-native/Libraries/CustomComponents/NavigationExperimental/NavigationHeader.js +.*/node_modules/react-native/Libraries/CustomComponents/NavigationExperimental/NavigationPagerStyleInterpolater.js +.*/node_modules/react-native/Libraries/Experimental/WindowedListView.js +.*/node_modules/react-native/Libraries/Image/Image.io.js +.*/node_modules/react-native/Libraries/NavigationExperimental/NavigationExperimental.js +.*/node_modules/react-native/Libraries/NavigationExperimental/NavigationHeaderStyleInterpolator.js +.*/node_modules/react-native/Libraries/Network/FormData.js +.*/node_modules/react-native/Libraries/ReactIOS/YellowBox.js + + + +# Ignore react and fbjs where there are overlaps, but don't ignore +# anything that react-native relies on +.*/node_modules/fbjs/lib/Map.js +.*/node_modules/fbjs/lib/ErrorUtils.js + +# Flow has a built-in definition for the 'react' module which we prefer to use +# over the currently-untyped source +.*/node_modules/react/react.js +.*/node_modules/react/lib/React.js +.*/node_modules/react/lib/ReactDOM.js + +.*/__mocks__/.* +.*/__tests__/.* + +.*/commoner/test/source/widget/share.js + +# Ignore commoner tests +.*/node_modules/commoner/test/.* + +# See https://github.com/facebook/flow/issues/442 +.*/react-tools/node_modules/commoner/lib/reader.js + +# Ignore jest +.*/node_modules/jest-cli/.* + +# Ignore Website +.*/website/.* + +# Ignore generators +.*/local-cli/generator.* + +# Ignore BUCK generated folders +.*\.buckd/ + +.*/node_modules/is-my-json-valid/test/.*\.json +.*/node_modules/iconv-lite/encodings/tables/.*\.json +.*/node_modules/y18n/test/.*\.json +.*/node_modules/spdx-license-ids/spdx-license-ids.json +.*/node_modules/spdx-exceptions/index.json +.*/node_modules/resolve/test/subdirs/node_modules/a/b/c/x.json +.*/node_modules/resolve/lib/core.json +.*/node_modules/jsonparse/samplejson/.*\.json +.*/node_modules/json5/test/.*\.json +.*/node_modules/ua-parser-js/test/.*\.json +.*/node_modules/builtin-modules/builtin-modules.json +.*/node_modules/binary-extensions/binary-extensions.json +.*/node_modules/url-regex/tlds.json +.*/node_modules/joi/.*\.json +.*/node_modules/isemail/.*\.json +.*/node_modules/tr46/.*\.json +.*/node_modules/protobufjs/src/bower.json +.*/node_modules/grpc/node_modules/protobufjs/src/bower.json + +[include] +node_modules/fbjs/lib + +[libs] +lib/flow.js +node_modules/react-native/Libraries/react-native/react-native-interface.js +node_modules/react-native/flow +node_modules/fbjs/flow/lib + +[options] +module.system=haste + +experimental.strict_type_args=true +unsafe.enable_getters_and_setters=true + +esproposal.class_static_fields=enable +esproposal.class_instance_fields=enable + +munge_underscores=true + +module.name_mapper='^image![a-zA-Z0-9$_-]+$' -> 'GlobalImageStub' +module.name_mapper='^[./a-zA-Z0-9$_-]+\.\(bmp\|gif\|jpg\|jpeg\|png\|psd\|svg\|webp\|m4v\|mov\|mp4\|mpeg\|mpg\|webm\|aac\|aiff\|caf\|m4a\|mp3\|wav\|html\|pdf\)$' -> 'RelativeImageStub' + +suppress_type=$FlowIssue +suppress_type=$FlowFixMe +suppress_type=$FixMe + +suppress_comment=\\(.\\|\n\\)*\\$FlowFixMe\\($\\|[^(]\\|(\\(>=0\\.\\(2[0-4]\\|1[0-9]\\|[0-9]\\).[0-9]\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\) +suppress_comment=\\(.\\|\n\\)*\\$FlowIssue\\((\\(>=0\\.\\(2[0-4]\\|1[0-9]\\|[0-9]\\).[0-9]\\)? *\\(site=[a-z,_]*react_native[a-z,_]*\\)?)\\)?:? #[0-9]+ +suppress_comment=\\(.\\|\n\\)*\\$FlowFixedInNextDeploy diff --git a/.gitignore b/.gitignore index 1347dc0..b5f5b47 100644 --- a/.gitignore +++ b/.gitignore @@ -29,6 +29,7 @@ android/.gradle/ android/.signing/ # User-specific configurations +android/.idea/gradle.xml android/.idea/libraries/ android/.idea/workspace.xml android/.idea/tasks.xml @@ -51,8 +52,10 @@ android/*.iml ehthumbs.db Thumbs.dbandroid/gradle android/gradlew +android/build android/gradlew.bat android/gradle/ .idea .idea coverage +yarn.lock diff --git a/.watchmanconfig b/.watchmanconfig new file mode 100644 index 0000000..b347186 --- /dev/null +++ b/.watchmanconfig @@ -0,0 +1,11 @@ +{ +"ignore_dirs": [ + ".git", + "node_modules", + "android/build", + "android/.idea", + "android/.gradle", + "android/gradle", + ".idea" + ] +} diff --git a/Firestack.podspec b/Firestack.podspec index ac45018..b569fe1 100644 --- a/Firestack.podspec +++ b/Firestack.podspec @@ -1,61 +1,28 @@ require 'json' -package = JSON.parse(File.read('package.json')) -version = package["version"] -repo = package['repository'] -author = package['author'] -all_pods = [ - 'FirebaseAnalytics', 'FirebaseAuth', 'FirebaseRemoteConfig', - 'FirebaseDatabase', 'FirebaseStorage', 'FirebaseInstanceID', - 'GoogleInterchangeUtilities', 'GoogleIPhoneUtilities', - 'GoogleNetworkingUtilities', 'GoogleParsingUtilities', - 'GoogleSymbolUtilities' -] +package = JSON.parse(File.read('package.json')) Pod::Spec.new do |s| - - s.name = "Firestack" - s.version = version - s.summary = "Firestack makes working with Firebase v3 easy" - - s.description = <<-DESC - Wanna integrate firebase into your app using React Native? - DESC - - s.homepage = "http://fullstackreact.com" - - s.license = { :type => "MIT", :file => "LICENSE" } - s.author = { "Ari Lerner" => author } - s.social_media_url = 'http://twitter.com/fullstackio' - - # When using multiple platforms - s.ios.deployment_target = "8.0" - # s.osx.deployment_target = "10.7" - # s.watchos.deployment_target = "2.0" - # s.tvos.deployment_target = "9.0" - - s.source = { :git => repo['url'], :tag => "v#{version}" } - s.public_header_files = "ios/Firestack/*.h" - - s.source_files = 'ios/Firestack/*.{h,m}' - s.preserve_paths = 'README.md', 'package.json', '*.js' - - s.ios.frameworks = [ - 'CFNetwork', 'Security', 'SystemConfiguration' - ] - s.ios.libraries = ['icucore', 'c++', 'sqlite3', 'z'] - - s.xcconfig = { - 'HEADER_SEARCH_PATHS' => [ - "$(inherited)", - "${SRCROOT}/../../React/**", - "${SRCROOT}/../../node_modules/react-native/**" - ].join(' '), - 'FRAMEWORK_SEARCH_PATHS' => [ - "$(inherited)", - "${PODS_ROOT}/Firebase/**", - "${PODS_ROOT}/FirebaseStorage/**", - ].join(' '), - 'OTHER_LDFLAGS' => '$(inherited) -ObjC' - } -end \ No newline at end of file + s.name = "Firestack" + s.version = package["version"] + s.summary = package["description"] + s.description = <<-DESC + Wanna integrate firebase into your app using React Native? + DESC + s.homepage = "http://fullstackreact.com" + s.license = package['license'] + s.author = "Ari Lerner" + s.source = { :git => "https://github.com/fullstackreact/react-native-firestack.git", :tag => "v#{s.version}" } + s.social_media_url = 'http://twitter.com/fullstackio' + s.platform = :ios, "8.0" + s.header_dir = 'ios/Firestack' + s.preserve_paths = 'README.md', 'package.json', '*.js' + s.source_files = 'ios/Firestack/*.{h,m}' + s.dependency 'React' + s.dependency 'Firebase/Auth' + s.dependency 'Firebase/Core' + s.dependency 'Firebase/Database' + s.dependency 'Firebase/Messaging' + s.dependency 'Firebase/RemoteConfig' + s.dependency 'Firebase/Storage' +end diff --git a/README.md b/README.md index 84c3333..6ed926f 100644 --- a/README.md +++ b/README.md @@ -1,903 +1,67 @@ -## Firestack +# Firestack -Firestack makes using the latest [Firebase](http://firebase.com) straight-forward. +Firestack makes using the latest [Firebase](http://firebase.com) with React Native straight-forward. -[](https://gitter.im/fullstackreact/react-native-firestack?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge) - -## What - -Firestack is a _light-weight_ layer sitting atop the native Firebase libraries for iOS and Android and mirrors the React Native JS api as closely as possible. - -For a detailed discussion of how Firestack works as well as how to contribute, check out our [contribution guide](https://github.com/fullstackreact/react-native-firestack/blob/master/Contributing.md). - -## Features - -* Nearly automatic, rapid setup on Firebase -* Covers lots of awesome features of Firebase: - * authentication - * username and password - * social auth (implemented, but need to add providers) - * storage handling - * upload files - * download urls - * download files - * real-time database - * presence out-of-the-box - * analytics - * Remote configuration -* Redux support built-in (but not required) -* Android and iOS support -* Community supported and professionally backed -* Intended on being as drop-dead simple as possible -* And so much more - -## Example app - -We have a working application example available in at [fullstackreact/FirestackApp](https://github.com/fullstackreact/FirestackApp). Check it out for more details about how to use Firestack. - -## Why? - -Firebase is awesome and it's combination with the Google Cloud Platform makes it super awesome. Sadly, the latest version of Firebase requires the `window` object. That's where Firestack comes in! Firestack provides a really thin layer that sits on top of the native Firebase SDKs and attempts to use the JavaScript library as much as possible rather than reinventing the wheel. - -## Installing - -Getting `react-native-firestack` up and running in your app should be a 2 step process + 1 for each platform. - -1. Install the `npm` package -2. Link the project with `react-native link react-native-firestack` -3. To ensure Android is setup, check your `MainApplication.java` for the `FirestackPackage()` line. - -Those steps in more detail: - -Install the `npm` package with: - -```bash -npm install react-native-firestack --save -``` - -To use Firestack, we'll need to have a development environment that includes the same prerequisites of Firebase. - -### iOS (with cocoapods) - -Unfortunately, due to AppStore restrictions, we currently do _not_ package Firebase libraries in with Firestack. However, the good news is we've automated the process (with many thanks to the Auth0 team for inspiration) of setting up with cocoapods. This will happen automatically upon linking the package with `react-native-cli`. - -**Remember to use the `ios/[YOUR APP NAME].xcworkspace` instead of the `ios/[YOUR APP NAME].xcproj` file from now on**. - -We need to link the package with our development packaging. We have two options to handle linking: - -#### Automatically with react-native-cli - -React native ships with a `link` command that can be used to link the projects together, which can help automate the process of linking our package environments. - -```bash -react-native link react-native-firestack -``` - -Update the newly installed pods once the linking is done: - -```bash -cd ios && pod update --verbose -``` - -#### Manually - -If you prefer not to use `rnpm`, we can manually link the package together with the following steps, after `npm install`: - -1. In XCode, right click on `Libraries` and find the `Add Files to [project name]`. - - - -2. Add the `node_modules/react-native-firestack/ios/Firestack.xcodeproj` - - - -3. Ensure that the `Build Settings` of the `Firestack.xcodeproj` project is ticked to _All_ and it's `Header Search Paths` include both of the following paths _and_ are set to _recursive_: - - 1. `$(SRCROOT)/../../react-native/React` - 2. `$(SRCROOT)/../node_modules/react-native/React` - 3. `${PROJECT_DIR}/../../../ios/Pods` - - - -4. Setting up cocoapods - -Since we're dependent upon cocoapods (or at least the Firebase libraries being available at the root project -- i.e. your application), we have to make them available for Firestack to find them. - -Using cocoapods is the easiest way to get started with this linking. Add or update a `Podfile` at `ios/Podfile` in your app with the following: - -```ruby -source 'https://github.com/CocoaPods/Specs.git' -[ - 'Firebase/Core', - 'Firebase/Auth', - 'Firebase/Storage', - 'Firebase/Database', - 'Firebase/RemoteConfig', - 'Firebase/Messaging' -].each do |lib| - pod lib -end -``` - -Then you can run `(cd ios && pod install)` to get the pods opened. If you do use this route, remember to use the `.xcworkspace` file. - -If you don't want to use cocoapods, you don't need to use it! Just make sure you link the Firebase libraries in your project manually. For more information, check out the relevant Firebase docs at [https://firebase.google.com/docs/ios/setup#frameworks](https://firebase.google.com/docs/ios/setup#frameworks). - -### Android - -Full Android support is coming soon, as it currently supports a smaller feature-set than the iOS version. Just as we do with iOS, we'll need to install the library using `npm` and call `link` on the library: - -```bash -react-native link react-native-firestack -``` - -Firestack includes the Firebase libraries and will link those directly into our project automatically. - -#### Manually - -To install `react-native-firestack` manually in our project, we'll need to import the package from `io.fullstack.firestack` in our project's `android/app/src/main/java/com/[app name]/MainApplication.java` and list it as a package for ReactNative in the `getPackages()` function: - -```java -package com.appName; -// ... -import io.fullstack.firestack.FirestackPackage; -// ... -public class MainApplication extends Application implements ReactApplication { - // ... - - @Override - protected List<ReactPackage> getPackages() { - return Arrays.<ReactPackage>asList( - new MainReactPackage(), - new FirestackPackage() - ); - } - }; - // ... -} -``` - -We'll also need to list it in our `android/app/build.gradle` file as a dependency that we want React Native to compile. In the `dependencies` listing, add the `compile` line: - -```java -dependencies { - compile project(':react-native-firestack') -} -``` - -Add to `AndroidManifest.xml` file -```diff - <activity android:name="com.facebook.react.devsupport.DevSettingsActivity" /> -+ <service android:name="io.fullstack.firestack.FirestackMessagingService"> -+ <intent-filter> -+ <action android:name="com.google.firebase.MESSAGING_EVENT"/> -+ </intent-filter> -+ </service> - -+ <service android:name="io.fullstack.firestack.FirestackInstanceIdService" android:exported="false"> -+ <intent-filter> -+ <action android:name="com.google.firebase.INSTANCE_ID_EVENT"/> -+ </intent-filter> -+ </service> -``` - -## Firebase setup - -The Firestack library is intended on making it easy to work with [Firebase](https://firebase.google.com/) and provides a small native shim to the Firebase native code. - -To add Firebase to your project, make sure to create a project in the [Firebase console](https://firebase.google.com/console) - - - -Each platform uses a different setup method after creating the project. - -### iOS - -After creating a Firebase project, click on the [Add Firebase to your iOS app](http://d.pr/i/3sEL.png) and follow the steps from there to add the configuration file. You do _not_ need to set up a cocoapods project (this is already done through firestack). Make sure not to forget the `Copy Files` phase in iOS. - -[Download the Firebase config file](https://support.google.com/firebase/answer/7015592) and place it in your app directory next to your app source code: - - - -Once you download the configuration file, make sure you place it in the root of your Xcode project. Every different Bundle ID (aka, even different project variants needs their own configuration file). - -Lastly, due to some dependencies requirements, Firestack supports iOS versions 8.0 and up. Make sure to update the minimum version of your iOS app to `8.0`. - -### Android - -There are several ways to setup Firebase on Android. The _easiest_ way is to pass the configuration settings in JavaScript. In that way, there is no setup for the native platform. - -#### google-services.json setup -If you prefer to include the default settings in the source of your app, download the `google-services.json` file provided by Firebase in the _Add Firebase to Android_ platform menu in your Firebase configuration console. - -Next you'll have to add the google-services gradle plugin in order to parse it. - -Add the google-services gradle plugin as a dependency in the *project* level build.gradle -`android/build.gradle` -```java -buildscript { - // ... - dependencies { - // ... - classpath 'com.google.gms:google-services:3.0.0' - } -} -``` - -In your app build.gradle file, add the gradle plugin at the VERY BOTTOM of the file (below all dependencies) -`android/app/build.gradle` -```java -apply plugin: 'com.google.gms.google-services' -``` - -## Usage - -After creating a Firebase project and installing the library, we can use it in our project by importing the library in our JavaScript: - -```javascript -import Firestack from 'react-native-firestack' -``` - -We need to tell the Firebase library we want to _configure_ the project. Firestack provides a way to configure both the native and the JavaScript side of the project at the same time with a single command: - -```javascript -const firestack = new Firestack(); -``` - -We can pass _custom_ options by passing an object with configuration options. The configuration object will be generated first by the native configuration object, if set and then will be overridden if passed in JS. That is, all of the following key/value pairs are optional if the native configuration is set. - -| option | type | Default Value | Description | -|----------------|----------|-------------------------|----------------------------------------| -| debug | bool | false | When set to true, Firestack will log messages to the console and fire `debug` events we can listen to in `js` | -| bundleID | string | Default from app `[NSBundle mainBundle]` | The bundle ID for the app to be bundled with | -| googleAppID | string | "" | The Google App ID that is used to uniquely identify an instance of an app. | -| databaseURL | string | "" | The database root (i.e. https://my-app.firebaseio.com) | -| deepLinkURLScheme | string | "" | URL scheme to set up durable deep link service | -| storageBucket | string | "" | The Google Cloud storage bucket name | -| androidClientID | string | "" | The Android client ID used in Google AppInvite when an iOS app has it's android version | -| GCMSenderID | string | "" | The Project number from the Google Developer's console used to configure Google Cloud Messaging | -| trackingID | string | "" | The tracking ID for Google Analytics | -| clientID | string | "" | The OAuth2 client ID for iOS application used to authenticate Google Users for signing in with Google | -| APIKey | string | "" | The secret iOS API key used for authenticating requests from our app | - -For instance: - -```javascript -const configurationOptions = { - debug: true -}; -const firestack = new Firestack(configurationOptions); -firestack.on('debug', msg => console.log('Received debug message', msg)) -``` - -## API documentation - -Firestack is broken up into multiple parts, based upon the different API features that Firebase provides. - -All methods return a promise. - -### Authentication - -Firestack handles authentication for us out of the box, both with email/password-based authentication and through oauth providers (with a separate library to handle oauth providers). - -> Android requires the Google Play services to installed for authentication to function. - -#### listenForAuth() - -Firebase gives us a reactive method for listening for authentication. That is we can set up a listener to call a method when the user logs in and out. To set up the listener, call the `listenForAuth()` method: - -```javascript -firestack.auth.listenForAuth(function(evt) { - // evt is the authentication event - // it contains an `error` key for carrying the - // error message in case of an error - // and a `user` key upon successful authentication - if (!evt.authenticated) { - // There was an error or there is no user - console.error(evt.error) - } else { - // evt.user contains the user details - console.log('User details', evt.user); - } -}) -.then(() => console.log('Listening for authentication changes')) -``` - -#### unlistenForAuth() - -We can remove this listener by calling the `unlistenForAuth()` method. This is important to release resources from our app when we don't need to hold on to the listener any longer. - -```javascript -firestack.auth.unlistenForAuth() -``` - -#### createUserWithEmail() - -We can create a user by calling the `createUserWithEmail()` function. The `createUserWithEmail()` accepts two parameters, an email and a password. - -```javascript -firestack.auth.createUserWithEmail('ari@fullstack.io', '123456') - .then((user) => { - console.log('user created', user) - }) - .catch((err) => { - console.error('An error occurred', err); - }) -``` - -#### signInWithEmail() - -To sign a user in with their email and password, use the `signInWithEmail()` function. It accepts two parameters, the user's email and password: - -```javascript -firestack.auth.signInWithEmail('ari@fullstack.io', '123456') - .then((user) => { - console.log('User successfully logged in', user) - }) - .catch((err) => { - console.error('User signin error', err); - }) -``` - -#### signInWithCustomToken() - -To sign a user using a self-signed custom token, use the `signInWithCustomToken()` function. It accepts one parameter, the custom token: - -```javascript -firestack.auth.signInWithCustomToken(TOKEN) - .then((user) => { - console.log('User successfully logged in', user) - }) - .catch((err) => { - console.error('User signin error', err); - }) -``` - -#### signInWithProvider() - -We can use an external authentication provider, such as twitter/facebook for authentication. In order to use an external provider, we need to include another library to handle authentication. - -> By using a separate library, we can keep our dependencies a little lower and the size of the application down. - -### OAuth setup with library - -[Currently undergoing updates] - -### socialLogin with custom Library -If you don't want to use [react-native-oauth](https://github.com/fullstackreact/react-native-oauth), you can use other library such as [react-native-facebook-login](https://github.com/magus/react-native-facebook-login). - -```javascript -var {FBLogin, FBLoginManager} = require('react-native-facebook-login'); - -var Login = React.createClass({ - render: function() { - return ( - <FBLogin - onLogin={function(data){ - console.log("Logged in!"); - console.log(data); - let token = data.credentials.token - firestack.signInWithProvider('facebook', token, '') // facebook need only access token. - .then((user)=>{ - console.log(user) - }) - }} - /> - ); - } -}); -``` - -If the `signInWithProvider()` method resolves correct and we have already set up our `listenForAuth()` method properly, it will fire and we'll have a logged in user through Firebase. - -### reauthenticateWithCredentialForProvider() - -When the auth token has expired, we can ask firebase to reauthenticate with the provider. This method accepts the _same_ arguments as `signInWithProvider()` accepts. - -#### updateUserEmail() - -We can update the current user's email by using the command: `updateUserEmail()`. It accepts a single argument: the user's new email: - -```javascript -firestack.auth.updateUserEmail('ari+rocks@fullstack.io') - .then((res) => console.log('Updated user email')) - .catch(err => console.error('There was an error updating user email')) -``` - -#### updateUserPassword() - -We can update the current user's password using the `updateUserPassword()` method. It accepts a single parameter: the new password for the current user - -```javascript -firestack.auth.updateUserPassword('somethingReallyS3cr3t733t') - .then(res => console.log('Updated user password')) - .catch(err => console.error('There was an error updating your password')) -``` - -### sendPasswordResetWithEmail() - -To send a password reset for a user based upon their email, we can call the `sendPasswordResetWithEmail()` method. It accepts a single parameter: the email of the user to send a reset email. - -```javascript -firestack.auth.sendPasswordResetWithEmail('ari+rocks@fullstack.io') - .then(res => console.log('Check your inbox for further instructions')) - .catch(err => console.error('There was an error :(')) -``` - -#### updateUserProfile() - -To update the current user's profile, we can call the `updateUserProfile()` method. - -It accepts a single parameter: - -* object which contains updated key/values for the user's profile. Possible keys are listed [here](https://firebase.google.com/docs/auth/ios/manage-users#update_a_users_profile). - -```javascript -firestack.auth.updateUserProfile({ - displayName: 'Ari Lerner' -}) - .then(res => console.log('Your profile has been updated')) - .catch(err => console.error('There was an error :(')) ``` - -#### deleteUser() - -It's possible to delete a user completely from your account on Firebase. Calling the `deleteUser()` method will take care of this for you. - -```javascript -firestack.auth.deleteUser() -.then(res => console.log('Sad to see you go')) -.catch(err => console.error('There was an error - Now you are trapped!')) +npm i react-native-firestack --save ``` -#### getToken() - -If you want user's token, use `getToken()` method. - -```javascript -firestack.auth.getToken() -.then(res => console.log(res.token)) -.catch(err => console.error('error')) -``` - -#### signOut() - -To sign the current user out, use the `signOut()` method. It accepts no parameters - -```javascript -firestack.auth.signOut() -.then(res => console.log('You have been signed out')) -.catch(err => console.error('Uh oh... something weird happened')) -``` - -#### getCurrentUser() - -Although you _can_ get the current user using the `getCurrentUser()` method, it's better to use this from within the callback function provided by `listenForAuth()`. However, if you need to get the current user, call the `getCurrentUser()` method: - -```javascript -firestack.auth.getCurrentUser() -.then(user => console.log('The currently logged in user', user)) -.catch(err => console.error('An error occurred')) -``` - -### Analytics - -Wouldn't it be nice to send analytics about your app usage from your users? Well, you totally can! The Firebase analytics console is incredibly useful and Firestack has a method for interacting with it. You can send any event with contextual information, which automatically includes the currently logged in user using the `logEventWithName()` method. It accepts two parameters: the name of the event and an object containing any contextual information. The values should be serializable (i.e. no complex instance objects). - -#### logEventWithName() - -```javascript -firestack.analytics.logEventWithName("launch", { - 'screen': 'Main screen' -}) -.then(res => console.log('Sent event named launch')) -.catch(err => console.error('You should never end up here')); -``` - -### Storage - -Firebase's integration with the Google platform expanded it's features to include hosting user-generated files, like photos. Firestack provides a thin layer to handle uploading files to Firebase's storage service. - -#### setStorageUrl() - -In order to store anything on Firebase, we need to set the storage url provided by Firebase. This can be set by using the `setStorageUrl()` method. Your storageUrl can be found on the firebase console. - - - -The `setStorageUrl()` method accepts a single parameter: your root storage url (without leading "gs://"). - -```javascript -firestack.storage.setStorageUrl(`${config.firebase.storageBucket}`) -``` - -If the `storageBucket` key is passed as a configuration option, this method is automatically called by default. - -#### uploadFile() - -We can upload a file using the `uploadFile()` method. Using the `uploadFile()` method, we can set the name of the destination file, the path where we want to store it, as well as any metadata along with the file. - -```javascript -firestack.storage.uploadFile(`photos/${auth.user.uid}/${filename}`, path, { - contentType: 'image/jpeg', - contentEncoding: 'base64', -}) -.then((res) => console.log('The file has been uploaded')) -.catch(err => console.error('There was an error uploading the file', err)) -``` - -To upload camera photos, we can combine this method with the `react-native-camera` plugin, for instance: - -```javascript -this.camera.capture() -.then(({path}) => { - firestack.storage.uploadFile(`photos/${auth.user.uid}/${filename}`, path, { - contentType: 'image/jpeg', - contentEncoding: 'base64', - }) -}) -.catch(err => console.error(err)); -``` - -To combine the `react-native-camera` plugin with firestack, we recommend setting the `captureTarget` to the `temp` storage path, like so: - -```javascript -<Camera - ref={(cam) => { - this.camera = cam; - }} - captureTarget={Camera.constants.CaptureTarget.temp} - style={styles.preview} - aspect={Camera.constants.Aspect.fill}> - <Text style={styles.capture} onPress={this.takePicture.bind(this)}>[CAPTURE]</Text> -</Camera> -``` - -Firestack also gives you the ability to listen for database events on upload. The final parameter the `uploadFile()` function accepts is a callback that will be called anytime a storage event is fired. - -The following events are supported: - -* upload_progress -* upload_paused -* upload_resumed - -For example, the `takePicture` function from the example above might look something similar to: - -```javascript -takePicture() { - this.camera.capture() - .then(({path}) => { - const filename = 'photo.jpg' - firestack.storage.uploadFile(`photos/${filename}`, path, { - contentType: 'image/jpeg', - contentEncoding: 'base64', - }, (evt) => { - console.log('Got an event in JS', evt); - }) - .then((res) => { - console.log('result from upload file: ', res); - }) - .catch((err) => { - console.log('error happened with uploadFile', err); - }) - }) - .catch(err => console.error(err)); -} -``` - -#### downloadUrl() - -The `downloadUrl()` method allows us to fetch the URL from the storage obejct in Firebase. It's defined on the `storageRef` object and can be used like so: - -```javascript -const storageRef = data.firestack.storage.ref('photos/photo.jpg'); -storageRef.downloadUrl() -.then(res => { - // res is an object that contains - // the `url` as well as the path to the file in `path` -}) -``` - -#### download() - -It's possible to download remote files as well. The `download()` method will take a remote file and download and save it to the user's device. It is implemented on the `storageRef`: - -```javascript -const storageRef = data.firestack.storage.ref('photos/photo.jpg'); -const localPath = `downloadedFile.jpg`; -storageRef.download(localPath, (msg) => { - // downloading state callback -}) -.then(res => { - // res contains details about the downloaded file -}) -.catch(err => { - // error contains any errors in downloading -}); -``` - -The method accepts a callback that gets called with any download events: - -* download_progress ({eventName: 'download_progress', progress: float }); -* download_paused ({eventName: 'download_paused'}) -* download_resumed ({eventName: 'download_resumed'}) - -As helpful constants, Firestack exports a few storage constants on the `firestack.constants` getter: - -* MAIN_BUNDLE_PATH -* CACHES_DIRECTORY_PATH -* DOCUMENT_DIRECTORY_PATH -* EXTERNAL_DIRECTORY_PATH -* EXTERNAL_STORAGE_DIRECTORY_PATH -* TEMP_DIRECTORY_PATH -* LIBRARY_DIRECTORY_PATH - -And we also export the filetype constants as well: - -* FILETYPE_REGULAR -* FILETYPE_DIRECTORY - -> Note: this idea comes almost directory from [react-native-fs](https://github.com/johanneslumpe/react-native-fs), so we don't claim credit for coming up with this fantastic idea. - -### Realtime Database - -The native Firebase JavaScript library provides a featureful realtime database that works out of the box. Firestack provides an attribute to interact with the database without needing to configure the JS library. - -Ranking strategy - -Add a new record with timestamp using this solution: - -firebaseApp.database.ref('posts').push().then((res) => { - let newPostKey = res.key; - firebaseApp.ServerValue.then(map => { - const postData = { - name: name, - timestamp: map.TIMESTAMP, - text: this.state.postText, - title: this.state.postTitle, - puid: newPostKey - } - let updates = {} - updates['/posts/' + newPostKey] = postData - firebaseApp.database.ref().update(updates).then(() => { - this.setState({ - postStatus: 'Posted! Thank You.', - postText: '', - }); - }).catch(() => { - this.setState({ postStatus: 'Something went wrong!!!' }); - }) - }) -}) - -Then retrieve the feed using this: - -firebaseApp.database.ref('posts').orderByChild('timestamp').limitToLast(30).once('value') -.then((snapshot) => { - this.props.savePosts(snapshot.val()) - const val = snapshot.val(); - console.log(val); -}) - -#### DatabaseRef - -Firestack attempts to provide the same API as the JS Firebase library for both Android and iOS platforms. [Check out the firebase guide](https://firebase.google.com/docs/database/web/read-and-write) for more information on how to use the JS library. - -#### Example - -```javascript - -function handleValueChange(snapshot) { - if (snapshot.val()) { - console.log('The list was updated'); - } -} - -const LIST_KEY = 'path/to/data'; -firestack.database.ref(LIST_KEY).on('value', handleValueChange); - -// Calling `.off` with a reference to the callback function will only remove that specific listener. -// This is useful if multiple components are listening and unlistening to the same ref path. -firestack.database.ref(LIST_KEY).off('value', handleValueChange); - -// Calling `.off` without passing the callback function will remove *all* 'value' listeners for that ref -firestack.database.ref(LIST_KEY).off('value'); - -``` - -// TODO: Finish documenting - -#### Offline data persistence - -For handling offline operations, you can enable persistence by using the `setPersistence()` command. You can turn it on and off by passing the boolean of `true` or `false`. - -```javascript -firestack.database.setPersistence(true); -``` - -The database refs has a `keepSynced()` function to tell the firestack library to keep the data at the `ref` in sync. - -```javascript -const ref = firestack.database - .ref('chat-messages') - .child('roomId'); -ref.keepSynced(true); -``` - -### Presence - -Firestack comes in with a built-in method for handling user connections. We just need to set the presence ref url and tell Firestack to keep track of the user by their child path. - -```javascript -firestack.presence // the presence api - .on('users/connections') // set the users/connections as the - // root for presence handling - .setOnline('auser') // Set the child of auser as online -``` - -While the _device_ is online (the connection), the value of the child object at `users/connections/auser` will be: - -```javascript -{ - online: true, - lastOnline: TIMESTAMP -} -``` - -When the device is offline, the value will be updated with `online: false`: - -```javascript -{ - online: false, - lastOnline: TIMESTAMP -} -``` - -To set up your own handlers on the presence object, you can call `onConnect()` and pass a callback. The method will be called with the `connectedDevice` database reference and you can set up your own handlers: - -```javascript -const presence = firestack.presence - .on('users/connections'); -presence.onConnect((ref) => { - ref.onDisconnect().remove(); // Remove the entry - // or - ref.set({ - location: someLocation - }); - // or whatever you want as it's called with the database - // reference. All methods on the DatabaseRef object are - // available here on the `ref` -}) -``` - -### ServerValue - -Firebase provides some static values based upon the server. We can use the `ServerValue` constant to retrieve these. For instance, to grab the TIMESTAMP on the server, use the `TIMESTAMP` value: - -```javascript -const timestamp = firestack.ServerValue.TIMESTAMP -``` - -### Cloud Messaging - -Access the device registration token - -```javascript - firestack.cloudMessaging.getToken().then(function (token) { - console.log('device token', token); - }); -``` - -Monitor token generation - -```javascript - // add listener - firestack.cloudMessaging.listenForTokenRefresh(function (token) { - console.log('refresh device token', token); - }); - - // remove listener - firestack.cloudMessaging.unlistenForTokenRefresh(); -``` - -Subscribe to topic - -```javascript - firestack.cloudMessaging.subscribeToTopic("topic_name").then(function (topic) { - console.log('Subscribe:'+topic); - }).catch(function(err){ - console.error(err); - }); -``` - -Unsubscribe from topic - -```javascript - firestack.cloudMessaging.unsubscribeFromTopic("topic_name").then(function (topic) { - console.log('unsubscribe:'+topic); - }).catch(function(err){ - console.error(err); - }); -``` - -Receive Messages - -```javascript - firestack.cloudMessaging.listenForReceiveNotification((msg) =>{ - console.log('Receive Messages:'+msg.data); - console.log('Receive Messages:'+msg.notification); - - }); -``` - -### Events - -#### on() - -We can listen to arbitrary events fired by the Firebase library using the `on()` method. The `on()` method accepts a name and a function callback: - -```javascript -firestack.on('listenForAuth', (evt) => console.log('Got an event')); -``` - -#### off() - -To unsubscribe to events fired by Firebase, we can call the `off()` method with the name of the event we want to unsubscribe. - -```javascript -firestack.off('listenForAuth'); -``` - -## FirestackModule - -Firestack provides a built-in way to connect your Redux app using the `FirestackModule` export from Firestack. +[](https://gitter.im/fullstackreact/react-native-firestack?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge) +[](https://www.npmjs.com/package/react-native-firestack) +[](/LICENSE) -## Running with the `master` branch +Firestack is a _light-weight_ layer sitting on-top of the native Firebase libraries for both iOS and Android which mirrors the React Native JS api as closely as possible. -Most of our work is committed to the master branch. If you want to run the bleeding-edge version of Firestack, you'll need to follow these instructions. +Featuring; authentication, storage, real-time database, presence, analytics, cloud messaging, remote configuration, redux support and more! -Since `react-native` doesn't like symlinks, we need to clone the raw repository into our `node_modules/` manually. First, in order to tell `react-native` we are using the package `react-native-firestack`, make sure to install the `npm` version: +## Firestack vs Firebase JS lib -```bash -npm install --save react-native-firestack -``` +Although the [Firebase](https://www.npmjs.com/package/firebase) JavaScript library will work with React Native, it is mainly designed for the web. -After the `npm` version is installed, you can either clone the repo directly into our `node_modules/` directory: +The native SDK's are much better for performance compared to the web SDK. The web SDK will run on the same thread as your apps ([JS thread](https://facebook.github.io/react-native/docs/performance.html#javascript-frame-rate)) therefore limiting your JS framerate, potentially affecting things touch events and transitions/animations. -```bash -git clone https://github.com/fullstackreact/react-native-firestack.git ./node_modules/react-native-firestack -``` +The native SDK's also contains functionality that the web SDK's do not, for example [Analytics](/docs/api/analytics.md) and [Remote Config](/docs/api/remote-config.md). -Alternatively, you can clone the repo somewhere else and `rsync` the directory over to the `node_modules/` directory. +## Example app -> This is the method I use as it allows me to separate the codebases: +We have a working application example available in at [fullstackreact/FirestackApp](https://github.com/fullstackreact/FirestackApp). Check it out for more details about how to use Firestack. -```bash -git clone https://github.com/fullstackreact/react-native-firestack.git \ - ~/Development/react-native/mine/react-native-firestack/ - -## And rsync -rsync -avhW --delete \ - --exclude='node_modules' \ - --exclude='.git' \ - ~/Development/react-native/mine/react-native-firestack/ \ - ./node_modules/react-native-firestack/ -``` +## Documentation + +* Installation + * [iOS](docs/installation.ios.md) + * [Android](docs/installation.android.md) +* [Firebase Setup](docs/firebase-setup.md) +* API + * [Authentication](docs/api/authentication.md) + * [Analytics](docs/api/analytics.md) + * [Storage](docs/api/storage.md) + * [Realtime Database](docs/api/database.md) + * [Presence](docs/api/presence.md) + * [ServerValue](docs/api/server-value.md) + * [Cloud Messaging](docs/api/cloud-messaging.md) + * [Remote Config](docs/api/remote-config.md) + * [Events](docs/api/events.md) +* [Redux](docs/redux.md) + +## Feature overview +Feature | Firestack | Firebase +------------ | ------------- | ------------- +Real time database | -- | YES +Authentication | -- | YES +Cloud messaging | -- | YES +Storage | -- | YES +Crash reporting | -- | YES +Notifications | -- | YES +Remote config | -- | YES +Dynamic links | -- | YES +AdMob | -- | YES +Analytics | -- | YES +App indexing | -- | YES +Hosting | -- | YES +AdWords | -- | YES +Invites | -- | YES ## Contributing -This is _open-source_ software and we can make it rock for everyone through contributions. - -How do you contribute? Check out our contribution guide at [CONTRIBUTING.md](https://github.com/fullstackreact/react-native-firestack/blob/master/Contributing.md) - -## TODO - -The following is left to be done: - -- [x] Complete FirebaseModule functionality -- [ ] Document FirebaseModule -- [X] Add Android support - - auth/analytics/database/storage/presence are feature-complete. remoteconfig/messaging are mostly-there. -- [x] Add Cloud Messaging - - [ ] Add JS api -- [ ] Move to use swift (cleaner syntax) -- [ ] TODO: Finish Facebook integration +For a detailed discussion of how Firestack works as well as how to contribute, check out our [contribution guide](https://github.com/fullstackreact/react-native-firestack/blob/master/Contributing.md). diff --git a/android/build.gradle b/android/build.gradle index a06d876..f0c0545 100644 --- a/android/build.gradle +++ b/android/build.gradle @@ -1,3 +1,16 @@ +// START - required to allow working on this project inside Android Studio +// YES, jcenter is required twice - it somehow tricks studio into compiling deps below +// doesn't break anything anywhere else and projects using this lib work as normal +buildscript { + repositories { + jcenter() + } + dependencies { + classpath 'com.android.tools.build:gradle:2.1.3' + } +} +// END + apply plugin: 'com.android.library' android { @@ -18,15 +31,26 @@ android { } } +// START - required to allow working on this project inside Android Studio +// YES, jcenter is required twice - it somehow tricks studio into compiling deps below +// doesn't break anything anywhere else and projects using this lib work as normal +// you'll now have code completion/validation and all the other AS goodies. +allprojects { + repositories { + jcenter() + } +} +// END + dependencies { compile 'com.facebook.react:react-native:0.20.+' - compile 'com.google.android.gms:play-services-base:9.8.0' - - compile 'com.google.firebase:firebase-core:9.8.0' - compile 'com.google.firebase:firebase-auth:9.8.0' - compile 'com.google.firebase:firebase-analytics:9.8.0' - compile 'com.google.firebase:firebase-database:9.8.0' - compile 'com.google.firebase:firebase-storage:9.8.0' - compile 'com.google.firebase:firebase-messaging:9.8.0' + compile 'com.google.android.gms:play-services-base:+' + compile 'com.google.firebase:firebase-core:10.0.1' + compile 'com.google.firebase:firebase-config:10.0.1' + compile 'com.google.firebase:firebase-auth:10.0.1' + compile 'com.google.firebase:firebase-analytics:10.0.1' + compile 'com.google.firebase:firebase-database:10.0.1' + compile 'com.google.firebase:firebase-storage:10.0.1' + compile 'com.google.firebase:firebase-messaging:10.0.1' } diff --git a/android/src/main/java/io/fullstack/firestack/FirestackAnalytics.java b/android/src/main/java/io/fullstack/firestack/FirestackAnalytics.java deleted file mode 100644 index ec67022..0000000 --- a/android/src/main/java/io/fullstack/firestack/FirestackAnalytics.java +++ /dev/null @@ -1,226 +0,0 @@ -package io.fullstack.firestack; - -import android.content.Context; -import android.util.Log; -import android.os.Bundle; -import java.util.Iterator; -import java.util.Map; -import android.support.annotation.NonNull; -import android.support.annotation.Nullable; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.bridge.ReactContext; - -import com.google.android.gms.tasks.OnCompleteListener; -import com.google.android.gms.tasks.OnFailureListener; -import com.google.android.gms.tasks.Task; -import com.google.firebase.FirebaseApp; -import com.google.firebase.analytics.FirebaseAnalytics; -import com.google.firebase.analytics.FirebaseAnalytics.Event.*; -import com.google.firebase.analytics.FirebaseAnalytics.Param; - -class FirestackAnalyticsModule extends ReactContextBaseJavaModule { - - private static final String TAG = "FirestackAnalytics"; - - private Context context; - private ReactContext mReactContext; - private FirebaseAnalytics mFirebaseAnalytics; - - public FirestackAnalyticsModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - mReactContext = reactContext; - - Log.d(TAG, "New instance"); - mFirebaseAnalytics = FirebaseAnalytics.getInstance(this.context); - } - - @Override - public String getName() { - return TAG; - } - - @ReactMethod - public void logEventWithName(final String name, final ReadableMap props, final Callback callback) { - // TODO - // FirestackUtils.todoNote(TAG, "logEventWithName", callback); - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(props); - final String eventName = getEventName(name); - final Bundle bundle = makeEventBundle(name, m); - Log.d(TAG, "Logging event " + eventName); - mFirebaseAnalytics.logEvent(name, bundle); - } - - private String getEventName(final String name) { - if (name == FirebaseAnalytics.Event.ADD_PAYMENT_INFO) {return FirebaseAnalytics.Event.ADD_PAYMENT_INFO; } - else if (name == FirebaseAnalytics.Event.ADD_TO_CART) {return FirebaseAnalytics.Event.ADD_TO_CART;} - else if (name == FirebaseAnalytics.Event.ADD_TO_WISHLIST) {return FirebaseAnalytics.Event.ADD_TO_WISHLIST;} - else if (name == FirebaseAnalytics.Event.APP_OPEN) {return FirebaseAnalytics.Event.APP_OPEN;} - else if (name == FirebaseAnalytics.Event.BEGIN_CHECKOUT) {return FirebaseAnalytics.Event.BEGIN_CHECKOUT;} - else if (name == FirebaseAnalytics.Event.ECOMMERCE_PURCHASE) {return FirebaseAnalytics.Event.ECOMMERCE_PURCHASE;} - else if (name == FirebaseAnalytics.Event.GENERATE_LEAD) {return FirebaseAnalytics.Event.GENERATE_LEAD;} - else if (name == FirebaseAnalytics.Event.JOIN_GROUP) {return FirebaseAnalytics.Event.JOIN_GROUP;} - else if (name == FirebaseAnalytics.Event.LEVEL_UP) {return FirebaseAnalytics.Event.LEVEL_UP;} - else if (name == FirebaseAnalytics.Event.LOGIN) {return FirebaseAnalytics.Event.LOGIN;} - else if (name == FirebaseAnalytics.Event.POST_SCORE) {return FirebaseAnalytics.Event.POST_SCORE;} - else if (name == FirebaseAnalytics.Event.PRESENT_OFFER) {return FirebaseAnalytics.Event.PRESENT_OFFER;} - else if (name == FirebaseAnalytics.Event.PURCHASE_REFUND) {return FirebaseAnalytics.Event.PURCHASE_REFUND;} - else if (name == FirebaseAnalytics.Event.SEARCH) {return FirebaseAnalytics.Event.SEARCH;} - else if (name == FirebaseAnalytics.Event.SELECT_CONTENT) {return FirebaseAnalytics.Event.SELECT_CONTENT;} - else if (name == FirebaseAnalytics.Event.SHARE) {return FirebaseAnalytics.Event.SHARE;} - else if (name == FirebaseAnalytics.Event.SIGN_UP) {return FirebaseAnalytics.Event.SIGN_UP;} - else if (name == FirebaseAnalytics.Event.SPEND_VIRTUAL_CURRENCY) {return FirebaseAnalytics.Event.SPEND_VIRTUAL_CURRENCY;} - else if (name == FirebaseAnalytics.Event.TUTORIAL_BEGIN) {return FirebaseAnalytics.Event.TUTORIAL_BEGIN;} - else if (name == FirebaseAnalytics.Event.TUTORIAL_COMPLETE) {return FirebaseAnalytics.Event.TUTORIAL_COMPLETE;} - else if (name == FirebaseAnalytics.Event.UNLOCK_ACHIEVEMENT) {return FirebaseAnalytics.Event.UNLOCK_ACHIEVEMENT;} - else if (name == FirebaseAnalytics.Event.VIEW_ITEM) {return FirebaseAnalytics.Event.VIEW_ITEM;} - else if (name == FirebaseAnalytics.Event.VIEW_ITEM_LIST) {return FirebaseAnalytics.Event.VIEW_ITEM_LIST;} - else if (name == FirebaseAnalytics.Event.VIEW_SEARCH_RESULTS) {return FirebaseAnalytics.Event.VIEW_SEARCH_RESULTS;} - else return name; - } - - private Bundle makeEventBundle(final String name, final Map<String, Object> map) { - Bundle bundle = new Bundle(); - // Available from the Analytics event - if (map.containsKey("id")) { - String id = (String) map.get("id"); - bundle.putString(FirebaseAnalytics.Param.ITEM_ID, id); - } - if (map.containsKey("name")) { - String val = (String) map.get("name"); - bundle.putString(FirebaseAnalytics.Param.ITEM_NAME, val); - } - if (map.containsKey("category")) { - String val = (String) map.get("category"); - bundle.putString(FirebaseAnalytics.Param.ITEM_NAME, val); - } - if (map.containsKey("quantity")) { - long val = (long) map.get("quantity"); - bundle.putLong(FirebaseAnalytics.Param.QUANTITY, val); - } - if (map.containsKey("price")) { - long val = (long) map.get("price"); - bundle.putLong(FirebaseAnalytics.Param.PRICE, val); - } - if (map.containsKey("value")) { - long val = (long) map.get("value"); - bundle.putLong(FirebaseAnalytics.Param.VALUE, val); - } - if (map.containsKey("currency")) { - String val = (String) map.get("currency"); - bundle.putString(FirebaseAnalytics.Param.CURRENCY, val); - } - if (map.containsKey("origin")) { - String val = (String) map.get("origin"); - bundle.putString(FirebaseAnalytics.Param.ORIGIN, val); - } - if (map.containsKey("item_location_id")) { - String val = (String) map.get("item_location_id"); - bundle.putString(FirebaseAnalytics.Param.ITEM_LOCATION_ID, val); - } - if (map.containsKey("location")) { - String val = (String) map.get("location"); - bundle.putString(FirebaseAnalytics.Param.LOCATION, val); - } - if (map.containsKey("destination")) { - String val = (String) map.get("destination"); - bundle.putString(FirebaseAnalytics.Param.DESTINATION, val); - } - if (map.containsKey("start_date")) { - String val = (String) map.get("start_date"); - bundle.putString(FirebaseAnalytics.Param.START_DATE, val); - } - if (map.containsKey("end_date")) { - String val = (String) map.get("end_date"); - bundle.putString(FirebaseAnalytics.Param.END_DATE, val); - } - if (map.containsKey("transaction_id")) { - String val = (String) map.get("transaction_id"); - bundle.putString(FirebaseAnalytics.Param.TRANSACTION_ID, val); - } - if (map.containsKey("number_of_nights")) { - long val = (long) map.get("number_of_nights"); - bundle.putLong(FirebaseAnalytics.Param.NUMBER_OF_NIGHTS, val); - } - if (map.containsKey("number_of_rooms")) { - long val = (long) map.get("number_of_rooms"); - bundle.putLong(FirebaseAnalytics.Param.NUMBER_OF_ROOMS, val); - } - if (map.containsKey("number_of_passengers")) { - long val = (long) map.get("number_of_passengers"); - bundle.putLong(FirebaseAnalytics.Param.NUMBER_OF_PASSENGERS, val); - } - if (map.containsKey("travel_class")) { - String val = (String) map.get("travel_class"); - bundle.putString(FirebaseAnalytics.Param.TRAVEL_CLASS, val); - } - if (map.containsKey("coupon")) { - String val = (String) map.get("coupon"); - bundle.putString(FirebaseAnalytics.Param.COUPON, val); - } - if (map.containsKey("tax")) { - long val = (long) map.get("tax"); - bundle.putLong(FirebaseAnalytics.Param.TAX, val); - } - if (map.containsKey("shipping")) { - double val = (double) map.get("shipping"); - bundle.putDouble(FirebaseAnalytics.Param.NUMBER_OF_PASSENGERS, val); - } - if (map.containsKey("group_id")) { - String val = (String) map.get("group_id"); - bundle.putString(FirebaseAnalytics.Param.GROUP_ID, val); - } - if (map.containsKey("level")) { - long val = (long) map.get("level"); - bundle.putLong(FirebaseAnalytics.Param.LEVEL, val); - } - if (map.containsKey("character")) { - String val = (String) map.get("character"); - bundle.putString(FirebaseAnalytics.Param.CHARACTER, val); - } - if (map.containsKey("score")) { - long val = (long) map.get("score"); - bundle.putLong(FirebaseAnalytics.Param.SCORE, val); - } - if (map.containsKey("search_term")) { - String val = (String) map.get("search_term"); - bundle.putString(FirebaseAnalytics.Param.SEARCH_TERM, val); - } - if (map.containsKey("content_type")) { - String val = (String) map.get("content_type"); - bundle.putString(FirebaseAnalytics.Param.CONTENT_TYPE, val); - } - if (map.containsKey("sign_up_method")) { - String val = (String) map.get("sign_up_method"); - bundle.putString(FirebaseAnalytics.Param.SIGN_UP_METHOD, val); - } - if (map.containsKey("virtual_currency_name")) { - String val = (String) map.get("virtual_currency_name"); - bundle.putString(FirebaseAnalytics.Param.VIRTUAL_CURRENCY_NAME, val); - } - if (map.containsKey("achievement_id")) { - String val = (String) map.get("achievement_id"); - bundle.putString(FirebaseAnalytics.Param.ACHIEVEMENT_ID, val); - } - if (map.containsKey("flight_number")) { - String val = (String) map.get("flight_number"); - bundle.putString(FirebaseAnalytics.Param.FLIGHT_NUMBER, val); - } - - Iterator<Map.Entry<String, Object>> entries = map.entrySet().iterator(); - while (entries.hasNext()) { - Map.Entry<String, Object> entry = entries.next(); - if (bundle.getBundle(entry.getKey()) == null) { - bundle.putString(entry.getKey(), entry.getValue().toString()); - } - } - return bundle; - } -} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackAuth.java b/android/src/main/java/io/fullstack/firestack/FirestackAuth.java deleted file mode 100644 index 36dd584..0000000 --- a/android/src/main/java/io/fullstack/firestack/FirestackAuth.java +++ /dev/null @@ -1,644 +0,0 @@ - -package io.fullstack.firestack; - -import android.content.Context; -import android.util.Log; -import java.util.Map; -import android.net.Uri; -import android.support.annotation.NonNull; -import android.support.annotation.Nullable; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.bridge.ReadableNativeMap; -import com.facebook.react.modules.core.DeviceEventManagerModule; -import com.facebook.react.bridge.ReactContext; - -import com.google.android.gms.tasks.OnCompleteListener; -import com.google.android.gms.tasks.OnFailureListener; -import com.google.android.gms.tasks.Task; -import com.google.firebase.FirebaseApp; -import com.google.firebase.FirebaseOptions; - -import com.google.firebase.auth.AuthCredential; -import com.google.firebase.auth.AuthResult; -import com.google.firebase.auth.UserProfileChangeRequest; -import com.google.firebase.auth.FacebookAuthProvider; -import com.google.firebase.auth.FirebaseAuth; -import com.google.firebase.auth.FirebaseUser; -import com.google.firebase.auth.GetTokenResult; -import com.google.firebase.auth.GoogleAuthProvider; -import com.google.firebase.auth.FirebaseAuthException; - -class FirestackAuthModule extends ReactContextBaseJavaModule { - private final int NO_CURRENT_USER = 100; - private final int ERROR_FETCHING_TOKEN = 101; - - private static final String TAG = "FirestackAuth"; - - private Context context; - private ReactContext mReactContext; - private FirebaseAuth mAuth; - private FirebaseApp app; - private FirebaseUser user; - private FirebaseAuth.AuthStateListener mAuthListener; - - public FirestackAuthModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - mReactContext = reactContext; - - Log.d(TAG, "New FirestackAuth instance"); - } - - @Override - public String getName() { - return TAG; - } - - @ReactMethod - public void listenForAuth() { - mAuthListener = new FirebaseAuth.AuthStateListener() { - - @Override - public void onAuthStateChanged(@NonNull FirebaseAuth firebaseAuth) { - WritableMap msgMap = Arguments.createMap(); - msgMap.putString("eventName", "listenForAuth"); - - if (firebaseAuth.getCurrentUser() != null) { - WritableMap userMap = getUserMap(); - - msgMap.putBoolean("authenticated", true); - msgMap.putMap("user", userMap); - - FirestackUtils.sendEvent(mReactContext, "listenForAuth", msgMap); - } else { - msgMap.putBoolean("authenticated", false); - FirestackUtils.sendEvent(mReactContext, "listenForAuth", msgMap); - } - } - }; - - mAuth = FirebaseAuth.getInstance(); - mAuth.addAuthStateListener(mAuthListener); - } - - @ReactMethod - public void unlistenForAuth(final Callback callback) { - if (mAuthListener != null) { - mAuth.removeAuthStateListener(mAuthListener); - - WritableMap resp = Arguments.createMap(); - resp.putString("status", "complete"); - - callback.invoke(null, resp); - } - } - - @ReactMethod - public void createUserWithEmail(final String email, final String password, final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - mAuth.createUserWithEmailAndPassword(email, password) - .addOnCompleteListener(new OnCompleteListener<AuthResult>() { - @Override - public void onComplete(@NonNull Task<AuthResult> task) { - if (task.isSuccessful()) { - FirestackAuthModule.this.user = task.getResult().getUser(); - userCallback(FirestackAuthModule.this.user, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void signInWithEmail(final String email, final String password, final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - mAuth.signInWithEmailAndPassword(email, password) - .addOnCompleteListener(new OnCompleteListener<AuthResult>() { - @Override - public void onComplete(@NonNull Task<AuthResult> task) { - if (task.isSuccessful()) { - FirestackAuthModule.this.user = task.getResult().getUser(); - userCallback(FirestackAuthModule.this.user, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - Log.e(TAG, "An exception occurred: " + ex.getMessage()); - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void signInWithProvider(final String provider, final String authToken, final String authSecret, final Callback callback) { - if (provider.equals("facebook")) { - this.facebookLogin(authToken,callback); - } else if (provider.equals("google")) { - this.googleLogin(authToken,callback); - } else - // TODO - FirestackUtils.todoNote(TAG, "signInWithProvider", callback); - } - - @ReactMethod - public void signInAnonymously(final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - mAuth.signInAnonymously() - .addOnCompleteListener(new OnCompleteListener<AuthResult>() { - @Override - public void onComplete(@NonNull Task<AuthResult> task) { - Log.d(TAG, "signInAnonymously:onComplete:" + task.isSuccessful()); - - if (task.isSuccessful()) { - FirestackAuthModule.this.user = task.getResult().getUser(); - anonymousUserCallback(FirestackAuthModule.this.user, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void signInWithCustomToken(final String customToken, final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - mAuth.signInWithCustomToken(customToken) - .addOnCompleteListener(new OnCompleteListener<AuthResult>() { - @Override - public void onComplete(@NonNull Task<AuthResult> task) { - Log.d(TAG, "signInWithCustomToken:onComplete:" + task.isSuccessful()); - if (task.isSuccessful()) { - FirestackAuthModule.this.user = task.getResult().getUser(); - userCallback(FirestackAuthModule.this.user, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void reauthenticateWithCredentialForProvider(final String provider, final String authToken, final String authSecret, final Callback callback) { - AuthCredential credential; - - if (provider.equals("facebook")) { - credential = FacebookAuthProvider.getCredential(authToken); - } else if (provider.equals("google")) { - credential = GoogleAuthProvider.getCredential(authToken, null); - } else { - // TODO: - FirestackUtils.todoNote(TAG, "reauthenticateWithCredentialForProvider", callback); - // AuthCredential credential; - // Log.d(TAG, "reauthenticateWithCredentialForProvider called with: " + provider); - return; - } - - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - if (user != null) { - user.reauthenticate(credential) - .addOnCompleteListener(new OnCompleteListener<Void>() { - @Override - public void onComplete(@NonNull Task<Void> task) { - if (task.isSuccessful()) { - Log.d(TAG, "User re-authenticated with " + provider); - FirebaseUser u = FirebaseAuth.getInstance().getCurrentUser(); - userCallback(u, callback); - } else { - // userErrorCallback(task, callback); - } - } - }); - } else { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", NO_CURRENT_USER); - err.putString("errorMessage", "No current user"); - callback.invoke(err); - } - } - - @ReactMethod - public void updateUserEmail(final String email, final Callback callback) { - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - - if (user != null) { - user.updateEmail(email) - .addOnCompleteListener(new OnCompleteListener<Void>() { - @Override - public void onComplete(@NonNull Task<Void> task) { - if (task.isSuccessful()) { - Log.d(TAG, "User email address updated"); - FirebaseUser u = FirebaseAuth.getInstance().getCurrentUser(); - userCallback(u, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } else { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", NO_CURRENT_USER); - err.putString("errorMessage", "No current user"); - callback.invoke(err); - } - } - - @ReactMethod - public void updateUserPassword(final String newPassword, final Callback callback) { - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - - if (user != null) { - user.updatePassword(newPassword) - .addOnCompleteListener(new OnCompleteListener<Void>() { - @Override - public void onComplete(@NonNull Task<Void> task) { - if (task.isSuccessful()) { - Log.d(TAG, "User password updated"); - - FirebaseUser u = FirebaseAuth.getInstance().getCurrentUser(); - userCallback(u, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } else { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", NO_CURRENT_USER); - err.putString("errorMessage", "No current user"); - callback.invoke(err); - } - } - - @ReactMethod - public void sendPasswordResetWithEmail(final String email, final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - mAuth.sendPasswordResetEmail(email) - .addOnCompleteListener(new OnCompleteListener<Void>() { - @Override - public void onComplete(@NonNull Task<Void> task) { - if(task.isSuccessful()){ - WritableMap resp = Arguments.createMap(); - resp.putString("status", "complete"); - callback.invoke(null, resp); - } else { - callback.invoke(task.getException().toString()); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void deleteUser(final Callback callback) { - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - - if (user != null) { - user.delete() - .addOnCompleteListener(new OnCompleteListener<Void>() { - @Override - public void onComplete(@NonNull Task<Void> task) { - if (task.isSuccessful()) { - Log.d(TAG, "User account deleted"); - WritableMap resp = Arguments.createMap(); - resp.putString("status", "complete"); - resp.putString("msg", "User account deleted"); - callback.invoke(null, resp); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } else { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", NO_CURRENT_USER); - err.putString("errorMessage", "No current user"); - callback.invoke(err); - } - } - - @ReactMethod - public void getToken(final Callback callback) { - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - - user.getToken(true) - .addOnCompleteListener(new OnCompleteListener<GetTokenResult>() { - @Override - public void onComplete(@NonNull Task<GetTokenResult> task) { - if (task.isSuccessful()) { - String token = task.getResult().getToken(); - WritableMap resp = Arguments.createMap(); - resp.putString("status", "complete"); - resp.putString("token", token); - callback.invoke(null, resp); - } else { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", ERROR_FETCHING_TOKEN); - err.putString("errorMessage", task.getException().getMessage()); - callback.invoke(err); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void updateUserProfile(ReadableMap props, final Callback callback) { - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - - UserProfileChangeRequest.Builder profileBuilder = new UserProfileChangeRequest.Builder(); - - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(props); - - if (m.containsKey("displayName")) { - String displayName = (String) m.get("displayName"); - profileBuilder.setDisplayName(displayName); - } - - if (m.containsKey("photoUri")) { - String photoUriStr = (String) m.get("photoUri"); - Uri uri = Uri.parse(photoUriStr); - profileBuilder.setPhotoUri(uri); - } - - UserProfileChangeRequest profileUpdates = profileBuilder.build(); - - user.updateProfile(profileUpdates) - .addOnCompleteListener(new OnCompleteListener<Void>() { - @Override - public void onComplete(@NonNull Task<Void> task) { - if (task.isSuccessful()) { - Log.d(TAG, "User profile updated"); - FirebaseUser u = FirebaseAuth.getInstance().getCurrentUser(); - userCallback(u, callback); - } else { - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void signOut(final Callback callback) { - FirebaseAuth.getInstance().signOut(); - this.user = null; - - WritableMap resp = Arguments.createMap(); - resp.putString("status", "complete"); - resp.putString("msg", "User signed out"); - callback.invoke(null, resp); - } - - @ReactMethod - public void getCurrentUser(final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - this.user = mAuth.getCurrentUser(); - if(this.user == null){ - noUserCallback(callback); - }else{ - userCallback(this.user, callback); - } - } - - // TODO: Check these things - @ReactMethod - public void googleLogin(String IdToken, final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - AuthCredential credential = GoogleAuthProvider.getCredential(IdToken, null); - mAuth.signInWithCredential(credential) - .addOnCompleteListener(new OnCompleteListener<AuthResult>() { - @Override - public void onComplete(@NonNull Task<AuthResult> task) { - if (task.isSuccessful()) { - FirestackAuthModule.this.user = task.getResult().getUser(); - userCallback(FirestackAuthModule.this.user, callback); - }else{ - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - @ReactMethod - public void facebookLogin(String Token, final Callback callback) { - mAuth = FirebaseAuth.getInstance(); - - AuthCredential credential = FacebookAuthProvider.getCredential(Token); - mAuth.signInWithCredential(credential) - .addOnCompleteListener(new OnCompleteListener<AuthResult>() { - @Override - public void onComplete(@NonNull Task<AuthResult> task) { - if (task.isSuccessful()) { - FirestackAuthModule.this.user = task.getResult().getUser(); - userCallback(FirestackAuthModule.this.user, callback); - }else{ - // userErrorCallback(task, callback); - } - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - // Internal helpers - public void userCallback(FirebaseUser passedUser, final Callback callback) { - - if (passedUser == null) { - mAuth = FirebaseAuth.getInstance(); - this.user = mAuth.getCurrentUser(); - } else { - this.user = passedUser; - } - - this.user.getToken(true).addOnCompleteListener(new OnCompleteListener<GetTokenResult>() { - @Override - public void onComplete(@NonNull Task<GetTokenResult> task) { - WritableMap msgMap = Arguments.createMap(); - WritableMap userMap = getUserMap(); - if (FirestackAuthModule.this.user != null) { - final String token = task.getResult().getToken(); - userMap.putString("token", token); - userMap.putBoolean("anonymous", false); - } - - msgMap.putMap("user", userMap); - msgMap.putBoolean("authenticated", true); - callback.invoke(null, msgMap); - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - // TODO: Reduce to one method - public void anonymousUserCallback(FirebaseUser passedUser, final Callback callback) { - - if (passedUser == null) { - mAuth = FirebaseAuth.getInstance(); - this.user = mAuth.getCurrentUser(); - } else { - this.user = passedUser; - } - - this.user.getToken(true) - .addOnCompleteListener(new OnCompleteListener<GetTokenResult>() { - @Override - public void onComplete(@NonNull Task<GetTokenResult> task) { - WritableMap msgMap = Arguments.createMap(); - WritableMap userMap = getUserMap(); - - if (FirestackAuthModule.this.user != null) { - final String token = task.getResult().getToken(); - - userMap.putString("token", token); - userMap.putBoolean("anonymous", true); - } - - msgMap.putMap("user", userMap); - - callback.invoke(null, msgMap); - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception ex) { - userExceptionCallback(ex, callback); - } - }); - } - - - public void noUserCallback(final Callback callback) { - WritableMap message = Arguments.createMap(); - - message.putString("errorMessage", "no_user"); - message.putString("eventName", "no_user"); - message.putBoolean("authenticated", false); - - callback.invoke(null, message); - } - - public void userErrorCallback(Task task, final Callback onFail) { - userExceptionCallback(task.getException(), onFail); - } - - public void userExceptionCallback(Exception exp, final Callback onFail) { - WritableMap error = Arguments.createMap(); - error.putString("errorMessage", exp.getMessage()); - error.putString("allErrorMessage", exp.toString()); - - try { - throw exp; - } catch (FirebaseAuthException ex) { - error.putString("errorCode", ex.getErrorCode()); - } catch (Exception ex) { - Log.e(TAG, ex.getMessage()); - } - - onFail.invoke(error); - } - - private WritableMap getUserMap() { - WritableMap userMap = Arguments.createMap(); - - FirebaseUser user = FirebaseAuth.getInstance().getCurrentUser(); - - if (user != null) { - final String email = user.getEmail(); - final String uid = user.getUid(); - final String provider = user.getProviderId(); - final String name = user.getDisplayName(); - final Uri photoUrl = user.getPhotoUrl(); - - userMap.putString("email", email); - userMap.putString("uid", uid); - userMap.putString("providerId", provider); - userMap.putBoolean("emailVerified", user.isEmailVerified()); - - if (name != null) { - userMap.putString("displayName", name); - } - - if (photoUrl != null) { - userMap.putString("photoUrl", photoUrl.toString()); - } - } else { - userMap.putString("msg", "no user"); - } - - return userMap; - } -} -n userMap; - } -} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackCloudMessaging.java b/android/src/main/java/io/fullstack/firestack/FirestackCloudMessaging.java deleted file mode 100644 index ef17a88..0000000 --- a/android/src/main/java/io/fullstack/firestack/FirestackCloudMessaging.java +++ /dev/null @@ -1,207 +0,0 @@ -package io.fullstack.firestack; - -import java.util.Map; - -import android.content.Context; -import android.content.IntentFilter; -import android.content.Intent; -import android.content.BroadcastReceiver; -import android.util.Log; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.Promise; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.bridge.ReadableMapKeySetIterator; -import com.facebook.react.bridge.ReadableType; -import com.facebook.react.bridge.WritableMap; - -import com.google.firebase.iid.FirebaseInstanceId; -import com.google.firebase.messaging.FirebaseMessaging; -import com.google.firebase.messaging.RemoteMessage; - -/** - * Created by nori on 2016/09/12. - */ -public class FirestackCloudMessaging extends ReactContextBaseJavaModule { - - private static final String TAG = "FirestackCloudMessaging"; - private static final String EVENT_NAME_TOKEN = "FirestackRefreshToken"; - private static final String EVENT_NAME_NOTIFICATION = "FirestackReceiveNotification"; - private static final String EVENT_NAME_SEND = "FirestackUpstreamSend"; - - public static final String INTENT_NAME_TOKEN = "io.fullstack.firestack.refreshToken"; - public static final String INTENT_NAME_NOTIFICATION = "io.fullstack.firestack.ReceiveNotification"; - public static final String INTENT_NAME_SEND = "io.fullstack.firestack.Upstream"; - - private ReactContext mReactContext; - private IntentFilter mRefreshTokenIntentFilter; - private IntentFilter mReceiveNotificationIntentFilter; - private IntentFilter mReceiveSendIntentFilter; - - public FirestackCloudMessaging(ReactApplicationContext reactContext) { - super(reactContext); - mReactContext = reactContext; - mRefreshTokenIntentFilter = new IntentFilter(INTENT_NAME_TOKEN); - mReceiveNotificationIntentFilter = new IntentFilter(INTENT_NAME_NOTIFICATION); - mReceiveSendIntentFilter = new IntentFilter(INTENT_NAME_SEND); - initRefreshTokenHandler(); - initMessageHandler(); - initSendHandler(); - Log.d(TAG, "New instance"); - } - - @Override - public String getName() { - return TAG; - } - - @ReactMethod - public void getToken(final Callback callback) { - - try { - String token = FirebaseInstanceId.getInstance().getToken(); - Log.d(TAG, "Firebase token: " + token); - callback.invoke(null, token); - } catch (Exception e) { - WritableMap error = Arguments.createMap(); - error.putString("message", e.getMessage()); - callback.invoke(error); - } - } - - /** - * - */ - private void initRefreshTokenHandler() { - getReactApplicationContext().registerReceiver(new BroadcastReceiver() { - @Override - public void onReceive(Context context, Intent intent) { - WritableMap params = Arguments.createMap(); - params.putString("token", intent.getStringExtra("token")); - ReactContext ctx = getReactApplicationContext(); - Log.d(TAG, "initRefreshTokenHandler received event " + EVENT_NAME_TOKEN); - FirestackUtils.sendEvent(ctx, EVENT_NAME_TOKEN, params); - } - - ; - }, mRefreshTokenIntentFilter); - } - - @ReactMethod - public void subscribeToTopic(String topic, final Callback callback) { - try { - FirebaseMessaging.getInstance().subscribeToTopic(topic); - callback.invoke(null,topic); - } catch (Exception e) { - e.printStackTrace(); - Log.d(TAG, "Firebase token: " + e); - WritableMap error = Arguments.createMap(); - error.putString("message", e.getMessage()); - callback.invoke(error); - - } - } - - @ReactMethod - public void unsubscribeFromTopic(String topic, final Callback callback) { - try { - FirebaseMessaging.getInstance().unsubscribeFromTopic(topic); - callback.invoke(null,topic); - } catch (Exception e) { - WritableMap error = Arguments.createMap(); - error.putString("message", e.getMessage()); - callback.invoke(error); - } - } - - private void initMessageHandler() { - Log.d(TAG, "Firestack initMessageHandler called"); - getReactApplicationContext().registerReceiver(new BroadcastReceiver() { - @Override - public void onReceive(Context context, Intent intent) { - RemoteMessage remoteMessage = intent.getParcelableExtra("data"); - Log.d(TAG, "Firebase onReceive: " + remoteMessage); - WritableMap params = Arguments.createMap(); - if (remoteMessage.getData().size() != 0) { - WritableMap dataMap = Arguments.createMap(); - Map<String, String> data = remoteMessage.getData(); - //Set<String> keysIterator = data.keySet(); - for (String key : data.keySet()) { - dataMap.putString(key, data.get(key)); - } - params.putMap("data", dataMap); - } else { - params.putNull("data"); - } - if (remoteMessage.getNotification() != null) { - WritableMap notificationMap = Arguments.createMap(); - RemoteMessage.Notification notification = remoteMessage.getNotification(); - notificationMap.putString("title", notification.getTitle()); - notificationMap.putString("body", notification.getBody()); - notificationMap.putString("icon", notification.getIcon()); - notificationMap.putString("sound", notification.getSound()); - notificationMap.putString("tag", notification.getTag()); - params.putMap("notification", notificationMap); - } else { - params.putNull("notification"); - } - ReactContext ctx = getReactApplicationContext(); - FirestackUtils.sendEvent(ctx, EVENT_NAME_NOTIFICATION, params); - } - }, mReceiveNotificationIntentFilter); - } - - @ReactMethod - public void send(String senderId, String messageId, String messageType, ReadableMap params, final Callback callback) { - FirebaseMessaging fm = FirebaseMessaging.getInstance(); - RemoteMessage.Builder remoteMessage = new RemoteMessage.Builder(senderId); - remoteMessage.setMessageId(messageId); - remoteMessage.setMessageType(messageType); - ReadableMapKeySetIterator iterator = params.keySetIterator(); - while (iterator.hasNextKey()) { - String key = iterator.nextKey(); - ReadableType type = params.getType(key); - if (type == ReadableType.String) { - remoteMessage.addData(key, params.getString(key)); - Log.d(TAG, "Firebase send: " + key); - Log.d(TAG, "Firebase send: " + params.getString(key)); - } - } - try { - fm.send(remoteMessage.build()); - WritableMap res = Arguments.createMap(); - res.putString("status", "success"); - callback.invoke(null, res); - } catch(Exception e) { - Log.e(TAG, "Error sending message", e); - WritableMap error = Arguments.createMap(); - error.putString("code", e.toString()); - error.putString("message", e.toString()); - callback.invoke(error); - } - } - - private void initSendHandler() { - getReactApplicationContext().registerReceiver(new BroadcastReceiver() { - @Override - public void onReceive(Context context, Intent intent) { - WritableMap params = Arguments.createMap(); - if (intent.getBooleanExtra("hasError", false)) { - WritableMap error = Arguments.createMap(); - error.putInt("code", intent.getIntExtra("errCode", 0)); - error.putString("message", intent.getStringExtra("errorMessage")); - params.putMap("err", error); - } else { - params.putNull("err"); - } - ReactContext ctx = getReactApplicationContext(); - FirestackUtils.sendEvent(ctx, EVENT_NAME_SEND, params); - } - }, mReceiveSendIntentFilter); - } -} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackDatabase.java b/android/src/main/java/io/fullstack/firestack/FirestackDatabase.java deleted file mode 100644 index 37aeca3..0000000 --- a/android/src/main/java/io/fullstack/firestack/FirestackDatabase.java +++ /dev/null @@ -1,719 +0,0 @@ -package io.fullstack.firestack; - -import android.content.Context; -import android.util.Log; -import java.util.HashMap; -import java.util.List; -import java.util.ListIterator; -import java.util.Map; -import android.net.Uri; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.bridge.ReadableMapKeySetIterator; -import com.facebook.react.bridge.ReadableArray; -import com.facebook.react.bridge.ReactContext; - -import com.google.firebase.database.FirebaseDatabase; -import com.google.firebase.database.DatabaseReference; -import com.google.firebase.database.ChildEventListener; -import com.google.firebase.database.OnDisconnect; -import com.google.firebase.database.Query; -import com.google.firebase.database.ValueEventListener; -import com.google.firebase.database.DataSnapshot; -import com.google.firebase.database.DatabaseError; - -class FirestackDBReference { - private static final String TAG = "FirestackDBReference"; - - private String mPath; - private ReadableArray mModifiers; - private HashMap<String, Boolean> mListeners = new HashMap<String, Boolean>(); - private FirestackDatabaseModule mDatabase; - private ChildEventListener mEventListener; - private ValueEventListener mValueListener; - private ValueEventListener mOnceValueListener; - private ReactContext mReactContext; - - public FirestackDBReference(final ReactContext context, final String path) { - mReactContext = context; - mPath = path; - } - - public void setModifiers(final ReadableArray modifiers) { - mModifiers = modifiers; - } - - public void addChildEventListener(final String name, final ReadableArray modifiers) { - final FirestackDBReference self = this; - - if (mEventListener == null) { - mEventListener = new ChildEventListener() { - @Override - public void onChildAdded(DataSnapshot dataSnapshot, String previousChildName) { - self.handleDatabaseEvent("child_added", mPath, dataSnapshot); - } - - @Override - public void onChildChanged(DataSnapshot dataSnapshot, String previousChildName) { - self.handleDatabaseEvent("child_changed", mPath, dataSnapshot); - } - - @Override - public void onChildRemoved(DataSnapshot dataSnapshot) { - self.handleDatabaseEvent("child_removed", mPath, dataSnapshot); - } - - @Override - public void onChildMoved(DataSnapshot dataSnapshot, String previousChildName) { - self.handleDatabaseEvent("child_moved", mPath, dataSnapshot); - } - - @Override - public void onCancelled(DatabaseError error) { - self.handleDatabaseError(name, mPath, error); - } - }; - } - - Query ref = this.getDatabaseQueryAtPathAndModifiers(modifiers); - ref.addChildEventListener(mEventListener); - this.setListeningTo(mPath, name); - } - - public void addValueEventListener(final String name, final ReadableArray modifiers) { - final FirestackDBReference self = this; - - mValueListener = new ValueEventListener() { - @Override - public void onDataChange(DataSnapshot dataSnapshot) { - self.handleDatabaseEvent("value", mPath, dataSnapshot); - } - - @Override - public void onCancelled(DatabaseError error) { - self.handleDatabaseError("value", mPath, error); - } - }; - - Query ref = this.getDatabaseQueryAtPathAndModifiers(modifiers); - ref.addValueEventListener(mValueListener); - this.setListeningTo(mPath, "value"); - } - - public void addOnceValueEventListener(final ReadableArray modifiers, - final Callback callback) { - final FirestackDBReference self = this; - - mOnceValueListener = new ValueEventListener() { - @Override - public void onDataChange(DataSnapshot dataSnapshot) { - WritableMap data = FirestackUtils.dataSnapshotToMap("value", mPath, dataSnapshot); - callback.invoke(null, data); - } - - @Override - public void onCancelled(DatabaseError error) { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", error.getCode()); - err.putString("errorDetails", error.getDetails()); - err.putString("description", error.getMessage()); - callback.invoke(err); - } - }; - - Query ref = this.getDatabaseQueryAtPathAndModifiers(modifiers); - ref.addListenerForSingleValueEvent(mOnceValueListener); - } - - public Boolean isListeningTo(final String path, final String evtName) { - String key = this.pathListeningKey(path, evtName); - return mListeners.containsKey(key); - } - - /** - * Note: these path/eventType listeners only get removed when javascript calls .off() and cleanup is run on the entire path - */ - public void setListeningTo(final String path, final String evtName) { - String key = this.pathListeningKey(path, evtName); - mListeners.put(key, true); - } - - public void notListeningTo(final String path, final String evtName) { - String key = this.pathListeningKey(path, evtName); - mListeners.remove(key); - } - - private String pathListeningKey(final String path, final String eventName) { - return "listener/" + path + "/" + eventName; - } - - public void cleanup() { - Log.d(TAG, "cleaning up database reference " + this); - this.removeChildEventListener(); - this.removeValueEventListener(); - } - - public void removeChildEventListener() { - if (mEventListener != null) { - DatabaseReference ref = this.getDatabaseRef(); - ref.removeEventListener(mEventListener); - this.notListeningTo(mPath, "child_added"); - this.notListeningTo(mPath, "child_changed"); - this.notListeningTo(mPath, "child_removed"); - this.notListeningTo(mPath, "child_moved"); - mEventListener = null; - } - } - - public void removeValueEventListener() { - DatabaseReference ref = this.getDatabaseRef(); - if (mValueListener != null) { - ref.removeEventListener(mValueListener); - this.notListeningTo(mPath, "value"); - mValueListener = null; - } - if (mOnceValueListener != null) { - ref.removeEventListener(mOnceValueListener); - mOnceValueListener = null; - } - } - - private void handleDatabaseEvent(final String name, final String path, final DataSnapshot dataSnapshot) { - if (!FirestackDBReference.this.isListeningTo(path, name)) { - return; - } - WritableMap data = FirestackUtils.dataSnapshotToMap(name, path, dataSnapshot); - WritableMap evt = Arguments.createMap(); - evt.putString("eventName", name); - evt.putString("path", path); - evt.putMap("body", data); - - FirestackUtils.sendEvent(mReactContext, "database_event", evt); - } - - private void handleDatabaseError(final String name, final String path, final DatabaseError error) { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", error.getCode()); - err.putString("errorDetails", error.getDetails()); - err.putString("description", error.getMessage()); - - WritableMap evt = Arguments.createMap(); - evt.putString("eventName", name); - evt.putString("path", path); - evt.putMap("body", err); - - FirestackUtils.sendEvent(mReactContext, "database_error", evt); - } - - public DatabaseReference getDatabaseRef() { - return FirebaseDatabase.getInstance().getReference(mPath); - } - - private Query getDatabaseQueryAtPathAndModifiers(final ReadableArray modifiers) { - DatabaseReference ref = this.getDatabaseRef(); - - List<Object> strModifiers = FirestackUtils.recursivelyDeconstructReadableArray(modifiers); - ListIterator<Object> it = strModifiers.listIterator(); - Query query = ref.orderByKey(); - - while(it.hasNext()) { - String str = (String) it.next(); - - String[] strArr = str.split(":"); - String methStr = strArr[0]; - - if (methStr.equalsIgnoreCase("orderByKey")) { - query = ref.orderByKey(); - } else if (methStr.equalsIgnoreCase("orderByValue")) { - query = ref.orderByValue(); - } else if (methStr.equalsIgnoreCase("orderByPriority")) { - query = ref.orderByPriority(); - } else if (methStr.contains("orderByChild")) { - String key = strArr[1]; - Log.d(TAG, "orderByChild: " + key); - query = ref.orderByChild(key); - } else if (methStr.contains("limitToLast")) { - String key = strArr[1]; - int limit = Integer.parseInt(key); - Log.d(TAG, "limitToLast: " + limit); - query = query.limitToLast(limit); - } else if (methStr.contains("limitToFirst")) { - String key = strArr[1]; - int limit = Integer.parseInt(key); - Log.d(TAG, "limitToFirst: " + limit); - query = query.limitToFirst(limit); - } else if (methStr.contains("equalTo")) { - String value = strArr[1]; - String key = strArr.length >= 3 ? strArr[2] : null; - if (key == null) { - query = query.equalTo(value); - } else { - query = query.equalTo(value, key); - } - } else if (methStr.contains("endAt")) { - String value = strArr[1]; - String key = strArr.length >= 3 ? strArr[2] : null; - if (key == null) { - query = query.endAt(value); - } else { - query = query.endAt(value, key); - } - } else if (methStr.contains("startAt")) { - String value = strArr[1]; - String key = strArr.length >= 3 ? strArr[2] : null; - if (key == null) { - query = query.startAt(value); - } else { - query = query.startAt(value, key); - } - } - } - - return query; - } - -} - -class FirestackDatabaseModule extends ReactContextBaseJavaModule { - - private static final String TAG = "FirestackDatabase"; - - private Context context; - private ReactContext mReactContext; - private HashMap<String, FirestackDBReference> mDBListeners = new HashMap<String, FirestackDBReference>(); - - public FirestackDatabaseModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - mReactContext = reactContext; - } - - @Override - public String getName() { - return TAG; - } - - // Persistence - @ReactMethod - public void enablePersistence( - final Boolean enable, - final Callback callback) { - try { - FirebaseDatabase.getInstance() - .setPersistenceEnabled(enable); - } catch (Throwable t) { - Log.e(TAG, "FirebaseDatabase setPersistenceEnabled exception", t); - } - - WritableMap res = Arguments.createMap(); - res.putString("status", "success"); - callback.invoke(null, res); - } - - @ReactMethod - public void keepSynced( - final String path, - final Boolean enable, - final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - ref.keepSynced(enable); - - WritableMap res = Arguments.createMap(); - res.putString("status", "success"); - res.putString("path", path); - callback.invoke(null, res); - } - - // Database - @ReactMethod - public void set( - final String path, - final ReadableMap props, - final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - - final FirestackDatabaseModule self = this; - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(props); - - DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError error, DatabaseReference ref) { - handleCallback("set", callback, error, ref); - } - }; - - ref.setValue(m, listener); - } - - @ReactMethod - public void update(final String path, - final ReadableMap props, - final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - final FirestackDatabaseModule self = this; - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(props); - - DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError error, DatabaseReference ref) { - handleCallback("update", callback, error, ref); - } - }; - - ref.updateChildren(m, listener); - } - - @ReactMethod - public void remove(final String path, - final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - final FirestackDatabaseModule self = this; - DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError error, DatabaseReference ref) { - handleCallback("remove", callback, error, ref); - } - }; - - ref.removeValue(listener); - } - - @ReactMethod - public void push(final String path, - final ReadableMap props, - final Callback callback) { - - Log.d(TAG, "Called push with " + path); - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - DatabaseReference newRef = ref.push(); - - final Uri url = Uri.parse(newRef.toString()); - final String newPath = url.getPath(); - - ReadableMapKeySetIterator iterator = props.keySetIterator(); - if (iterator.hasNextKey()) { - Log.d(TAG, "Passed value to push"); - // lame way to check if the `props` are empty - final FirestackDatabaseModule self = this; - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(props); - - DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError error, DatabaseReference ref) { - if (error != null) { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", error.getCode()); - err.putString("errorDetails", error.getDetails()); - err.putString("description", error.getMessage()); - callback.invoke(err); - } else { - WritableMap res = Arguments.createMap(); - res.putString("status", "success"); - res.putString("ref", newPath); - callback.invoke(null, res); - } - } - }; - - newRef.setValue(m, listener); - } else { - Log.d(TAG, "No value passed to push: " + newPath); - WritableMap res = Arguments.createMap(); - res.putString("result", "success"); - res.putString("ref", newPath); - callback.invoke(null, res); - } - } - - @ReactMethod - public void on(final String path, - final ReadableArray modifiers, - final String name, - final Callback callback) { - FirestackDBReference ref = this.getDBHandle(path); - - WritableMap resp = Arguments.createMap(); - - if (name.equals("value")) { - ref.addValueEventListener(name, modifiers); - } else { - ref.addChildEventListener(name, modifiers); - } - - this.saveDBHandle(path, ref); - resp.putString("result", "success"); - Log.d(TAG, "Added listener " + name + " for " + ref); - - resp.putString("handle", path); - callback.invoke(null, resp); - } - - @ReactMethod - public void onOnce(final String path, - final ReadableArray modifiers, - final String name, - final Callback callback) { - Log.d(TAG, "Setting one-time listener on event: " + name + " for path " + path); - FirestackDBReference ref = this.getDBHandle(path); - ref.addOnceValueEventListener(modifiers, callback); - } - - /** - * At the time of this writing, off() only gets called when there are no more subscribers to a given path. - * `mListeners` might therefore be out of sync (though javascript isnt listening for those eventTypes, so - * it doesn't really matter- just polluting the RN bridge a little more than necessary. - * off() should therefore clean *everything* up - */ - @ReactMethod - public void off(final String path, @Deprecated final String name, final Callback callback) { - this.removeDBHandle(path); - Log.d(TAG, "Removed listener " + path); - WritableMap resp = Arguments.createMap(); - resp.putString("handle", path); - resp.putString("result", "success"); - callback.invoke(null, resp); - } - - // On Disconnect - @ReactMethod - public void onDisconnectSetObject(final String path, final ReadableMap props, final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(props); - - OnDisconnect od = ref.onDisconnect(); - od.setValue(m, new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError databaseError, DatabaseReference databaseReference) { - handleCallback("onDisconnectSetObject", callback, databaseError, databaseReference); - } - }); - } - - @ReactMethod - public void onDisconnectSetString(final String path, final String value, final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - - OnDisconnect od = ref.onDisconnect(); - od.setValue(value, new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError databaseError, DatabaseReference databaseReference) { - handleCallback("onDisconnectSetString", callback, databaseError, databaseReference); - } - }); - } - - @ReactMethod - public void onDisconnectRemove(final String path, final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - - OnDisconnect od = ref.onDisconnect(); - od.removeValue(new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError databaseError, DatabaseReference databaseReference) { - handleCallback("onDisconnectRemove", callback, databaseError, databaseReference); - } - }); - } - @ReactMethod - public void onDisconnectCancel(final String path, final Callback callback) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - - OnDisconnect od = ref.onDisconnect(); - od.cancel(new DatabaseReference.CompletionListener() { - @Override - public void onComplete(DatabaseError databaseError, DatabaseReference databaseReference) { - handleCallback("onDisconnectCancel", callback, databaseError, databaseReference); - } - }); - } - - // Private helpers - // private void handleDatabaseEvent(final String name, final DataSnapshot dataSnapshot) { - // WritableMap data = this.dataSnapshotToMap(name, dataSnapshot); - // WritableMap evt = Arguments.createMap(); - // evt.putString("eventName", name); - // evt.putMap("body", data); - // FirestackUtils.sendEvent(mReactContext, "database_event", evt); - // } - - // private void handleDatabaseError(final String name, final DatabaseError error) { - // WritableMap err = Arguments.createMap(); - // err.putInt("errorCode", error.getCode()); - // err.putString("errorDetails", error.getDetails()); - // err.putString("description", error.getMessage()); - - // WritableMap evt = Arguments.createMap(); - // evt.putString("eventName", name); - // evt.putMap("body", err); - // FirestackUtils.sendEvent(mReactContext, "database_error", evt); - // } - - private void handleCallback( - final String methodName, - final Callback callback, - final DatabaseError databaseError, - final DatabaseReference databaseReference) { - if (databaseError != null) { - WritableMap err = Arguments.createMap(); - err.putInt("errorCode", databaseError.getCode()); - err.putString("errorDetails", databaseError.getDetails()); - err.putString("description", databaseError.getMessage()); - callback.invoke(err); - } else { - WritableMap res = Arguments.createMap(); - res.putString("status", "success"); - res.putString("method", methodName); - callback.invoke(null, res); - } - } - - private FirestackDBReference getDBHandle(final String path) { - if (!mDBListeners.containsKey(path)) { - ReactContext ctx = getReactApplicationContext(); - mDBListeners.put(path, new FirestackDBReference(ctx, path)); - } - - return mDBListeners.get(path); - } - - private void saveDBHandle(final String path, final FirestackDBReference dbRef) { - mDBListeners.put(path, dbRef); - } - - private void removeDBHandle(final String path) { - if (mDBListeners.containsKey(path)) { - FirestackDBReference r = mDBListeners.get(path); - r.cleanup(); - mDBListeners.remove(path); - } - } - - private String keyPath(final String path, final String eventName) { - return path + "-" + eventName; - } - - // TODO: move to FirestackDBReference? - private DatabaseReference getDatabaseReferenceAtPath(final String path) { - DatabaseReference mDatabase = FirebaseDatabase.getInstance().getReference(path); - return mDatabase; - } - - private Query getDatabaseQueryAtPathAndModifiers( - final String path, - final ReadableArray modifiers) { - DatabaseReference ref = this.getDatabaseReferenceAtPath(path); - - List<Object> strModifiers = FirestackUtils.recursivelyDeconstructReadableArray(modifiers); - ListIterator<Object> it = strModifiers.listIterator(); - Query query = ref.orderByKey(); - - while(it.hasNext()) { - String str = (String) it.next(); - String[] strArr = str.split(":"); - String methStr = strArr[0]; - - if (methStr.equalsIgnoreCase("orderByKey")) { - query = ref.orderByKey(); - } else if (methStr.equalsIgnoreCase("orderByValue")) { - query = ref.orderByValue(); - } else if (methStr.equalsIgnoreCase("orderByPriority")) { - query = ref.orderByPriority(); - } else if (methStr.contains("orderByChild")) { - String key = strArr[1]; - Log.d(TAG, "orderByChild: " + key); - query = ref.orderByChild(key); - } else if (methStr.contains("limitToLast")) { - String key = strArr[1]; - int limit = Integer.parseInt(key); - Log.d(TAG, "limitToLast: " + limit); - query = query.limitToLast(limit); - } else if (methStr.contains("limitToFirst")) { - String key = strArr[1]; - int limit = Integer.parseInt(key); - Log.d(TAG, "limitToFirst: " + limit); - query = query.limitToFirst(limit); - } else if (methStr.contains("equalTo")) { - String value = strArr[1]; - String key = strArr.length >= 3 ? strArr[2] : null; - if (key == null) { - query = query.equalTo(value); - } else { - query = query.equalTo(value, key); - } - } else if (methStr.contains("endAt")) { - String value = strArr[1]; - String key = strArr.length >= 3 ? strArr[2] : null; - if (key == null) { - query = query.endAt(value); - } else { - query = query.endAt(value, key); - } - } else if (methStr.contains("startAt")) { - String value = strArr[1]; - String key = strArr.length >= 3 ? strArr[2] : null; - if (key == null) { - query = query.startAt(value); - } else { - query = query.startAt(value, key); - } - } - } - - return query; - } - - private WritableMap dataSnapshotToMap(String name, String path, DataSnapshot dataSnapshot) { - return FirestackUtils.dataSnapshotToMap(name, path, dataSnapshot); - } - - private <Any> Any castSnapshotValue(DataSnapshot snapshot) { - if (snapshot.hasChildren()) { - WritableMap data = Arguments.createMap(); - for (DataSnapshot child : snapshot.getChildren()) { - Any castedChild = castSnapshotValue(child); - switch (castedChild.getClass().getName()) { - case "java.lang.Boolean": - data.putBoolean(child.getKey(), (Boolean) castedChild); - break; - case "java.lang.Long": - data.putDouble(child.getKey(), (Long) castedChild); - break; - case "java.lang.Double": - data.putDouble(child.getKey(), (Double) castedChild); - break; - case "java.lang.String": - data.putString(child.getKey(), (String) castedChild); - break; - case "com.facebook.react.bridge.WritableNativeMap": - data.putMap(child.getKey(), (WritableMap) castedChild); - break; - } - } - return (Any) data; - } else { - if (snapshot.getValue() != null) { - String type = snapshot.getValue().getClass().getName(); - switch (type) { - case "java.lang.Boolean": - return (Any)((Boolean) snapshot.getValue()); - case "java.lang.Long": - return (Any) ((Long) snapshot.getValue()); - case "java.lang.Double": - return (Any)((Double) snapshot.getValue()); - case "java.lang.String": - return (Any)((String) snapshot.getValue()); - default: - return (Any) null; - } - } else { - return (Any) null; - } - } - } -} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackInstanceIdService.java b/android/src/main/java/io/fullstack/firestack/FirestackInstanceIdService.java index 8922ef4..f27f8c9 100644 --- a/android/src/main/java/io/fullstack/firestack/FirestackInstanceIdService.java +++ b/android/src/main/java/io/fullstack/firestack/FirestackInstanceIdService.java @@ -1,15 +1,14 @@ package io.fullstack.firestack; -/** - * Created by nori on 2016/09/12. - */ -import android.content.Intent; -import android.os.Bundle; import android.util.Log; +import android.os.Bundle; +import android.content.Intent; import com.google.firebase.iid.FirebaseInstanceId; import com.google.firebase.iid.FirebaseInstanceIdService; +import io.fullstack.firestack.messaging.FirestackMessaging; + public class FirestackInstanceIdService extends FirebaseInstanceIdService { private static final String TAG = "FSInstanceIdService"; @@ -21,10 +20,7 @@ public class FirestackInstanceIdService extends FirebaseInstanceIdService { public void onTokenRefresh() { String refreshedToken = FirebaseInstanceId.getInstance().getToken(); Log.d(TAG, "Refreshed token: " + refreshedToken); - - - // send Intent - Intent i = new Intent(FirestackCloudMessaging.INTENT_NAME_TOKEN); + Intent i = new Intent(FirestackMessaging.INTENT_NAME_TOKEN); Bundle bundle = new Bundle(); bundle.putString("token", refreshedToken); i.putExtras(bundle); diff --git a/android/src/main/java/io/fullstack/firestack/FirestackMessagingService.java b/android/src/main/java/io/fullstack/firestack/FirestackMessagingService.java index 485e762..9c32564 100644 --- a/android/src/main/java/io/fullstack/firestack/FirestackMessagingService.java +++ b/android/src/main/java/io/fullstack/firestack/FirestackMessagingService.java @@ -7,6 +7,8 @@ import com.google.firebase.messaging.RemoteMessage; import com.google.firebase.messaging.SendException; +import io.fullstack.firestack.messaging.FirestackMessaging; + public class FirestackMessagingService extends FirebaseMessagingService { private static final String TAG = "FSMessagingService"; @@ -16,16 +18,16 @@ public void onMessageReceived(RemoteMessage remoteMessage) { Log.d(TAG, "Remote message received"); // debug Log.d(TAG, "From: " + remoteMessage.getFrom()); + if (remoteMessage.getData().size() > 0) { Log.d(TAG, "Message data payload: " + remoteMessage.getData()); } + if (remoteMessage.getNotification() != null) { Log.d(TAG, "Message Notification Body: " + remoteMessage.getNotification().getBody()); } - if (remoteMessage.getNotification() != null) { - } - Intent i = new Intent(FirestackCloudMessaging.INTENT_NAME_NOTIFICATION); + Intent i = new Intent(FirestackMessaging.INTENT_NAME_NOTIFICATION); i.putExtra("data", remoteMessage); sendOrderedBroadcast(i, null); @@ -35,7 +37,7 @@ public void onMessageReceived(RemoteMessage remoteMessage) { public void onMessageSent(String msgId) { // Called when an upstream message has been successfully sent to the GCM connection server. Log.d(TAG, "upstream message has been successfully sent"); - Intent i = new Intent(FirestackCloudMessaging.INTENT_NAME_SEND); + Intent i = new Intent(FirestackMessaging.INTENT_NAME_SEND); i.putExtra("msgId", msgId); sendOrderedBroadcast(i, null); } @@ -44,7 +46,7 @@ public void onMessageSent(String msgId) { public void onSendError(String msgId, Exception exception) { // Called when there was an error sending an upstream message. Log.d(TAG, "error sending an upstream message"); - Intent i = new Intent(FirestackCloudMessaging.INTENT_NAME_SEND); + Intent i = new Intent(FirestackMessaging.INTENT_NAME_SEND); i.putExtra("msgId", msgId); i.putExtra("hasError", true); SendException sendException = (SendException) exception; diff --git a/android/src/main/java/io/fullstack/firestack/FirestackModule.java b/android/src/main/java/io/fullstack/firestack/FirestackModule.java index ac2418f..d537411 100644 --- a/android/src/main/java/io/fullstack/firestack/FirestackModule.java +++ b/android/src/main/java/io/fullstack/firestack/FirestackModule.java @@ -1,25 +1,24 @@ package io.fullstack.firestack; -import android.content.Context; -import android.util.Log; import java.util.Map; -import android.support.annotation.NonNull; +import java.util.HashMap; + +import android.util.Log; +import android.content.Context; import android.support.annotation.Nullable; +import com.facebook.react.bridge.Callback; import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.ReadableMap; +import com.facebook.react.bridge.WritableMap; +import com.facebook.react.bridge.ReactMethod; import com.facebook.react.bridge.LifecycleEventListener; import com.facebook.react.bridge.ReactApplicationContext; import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.modules.core.DeviceEventManagerModule; -import com.facebook.react.bridge.ReactContext; -import com.google.android.gms.tasks.OnCompleteListener; -import com.google.android.gms.tasks.OnFailureListener; -import com.google.android.gms.tasks.Task; +import com.google.android.gms.common.ConnectionResult; +import com.google.android.gms.common.GoogleApiAvailability; + import com.google.firebase.FirebaseApp; import com.google.firebase.FirebaseOptions; import com.google.firebase.database.ServerValue; @@ -28,18 +27,13 @@ interface KeySetterFn { String setKeyOrDefault(String a, String b); } -class FirestackModule extends ReactContextBaseJavaModule implements LifecycleEventListener { +@SuppressWarnings("WeakerAccess") +public class FirestackModule extends ReactContextBaseJavaModule implements LifecycleEventListener { private static final String TAG = "Firestack"; - private Context context; - private ReactContext mReactContext; private FirebaseApp app; - public FirestackModule(ReactApplicationContext reactContext, Context context) { + public FirestackModule(ReactApplicationContext reactContext) { super(reactContext); - this.context = context; - mReactContext = reactContext; - - Log.d(TAG, "New instance"); } @Override @@ -47,12 +41,27 @@ public String getName() { return TAG; } + private WritableMap getPlayServicesStatus() { + GoogleApiAvailability gapi = GoogleApiAvailability.getInstance(); + final int status = gapi.isGooglePlayServicesAvailable(getReactApplicationContext()); + WritableMap result = Arguments.createMap(); + result.putInt("status", status); + if (status == ConnectionResult.SUCCESS) { + result.putBoolean("isAvailable", true); + } else { + result.putBoolean("isAvailable", false); + result.putBoolean("isUserResolvableError", gapi.isUserResolvableError(status)); + result.putString("error", gapi.getErrorString(status)); + } + return result; + } + @ReactMethod public void configureWithOptions(final ReadableMap params, @Nullable final Callback onComplete) { Log.i(TAG, "configureWithOptions"); FirebaseOptions.Builder builder = new FirebaseOptions.Builder(); - FirebaseOptions defaultOptions = FirebaseOptions.fromResource(this.context); + FirebaseOptions defaultOptions = FirebaseOptions.fromResource(getReactApplicationContext().getBaseContext()); if (defaultOptions == null) { defaultOptions = new FirebaseOptions.Builder().build(); @@ -60,8 +69,8 @@ public void configureWithOptions(final ReadableMap params, @Nullable final Callb KeySetterFn fn = new KeySetterFn() { public String setKeyOrDefault( - final String key, - final String defaultValue) { + final String key, + final String defaultValue) { if (params.hasKey(key)) { // User-set key final String val = params.getString(key); @@ -76,47 +85,26 @@ public String setKeyOrDefault( } }; - String val = fn.setKeyOrDefault("applicationId", - defaultOptions.getApplicationId()); - if (val != null) { - builder.setApplicationId(val); - } + String val = fn.setKeyOrDefault("applicationId", defaultOptions.getApplicationId()); + if (val != null) builder.setApplicationId(val); - val = fn.setKeyOrDefault("apiKey", - defaultOptions.getApiKey()); - if (val != null) { - builder.setApiKey(val); - } + val = fn.setKeyOrDefault("apiKey", defaultOptions.getApiKey()); + if (val != null) builder.setApiKey(val); - val = fn.setKeyOrDefault("gcmSenderID", - defaultOptions.getGcmSenderId()); - if (val != null) { - builder.setGcmSenderId(val); - } + val = fn.setKeyOrDefault("gcmSenderID", defaultOptions.getGcmSenderId()); + if (val != null) builder.setGcmSenderId(val); - val = fn.setKeyOrDefault("storageBucket", - defaultOptions.getStorageBucket()); - if (val != null) { - builder.setStorageBucket(val); - } + val = fn.setKeyOrDefault("storageBucket", defaultOptions.getStorageBucket()); + if (val != null) builder.setStorageBucket(val); - val = fn.setKeyOrDefault("databaseURL", - defaultOptions.getDatabaseUrl()); - if (val != null) { - builder.setDatabaseUrl(val); - } + val = fn.setKeyOrDefault("databaseURL", defaultOptions.getDatabaseUrl()); + if (val != null) builder.setDatabaseUrl(val); - val = fn.setKeyOrDefault("databaseUrl", - defaultOptions.getDatabaseUrl()); - if (val != null) { - builder.setDatabaseUrl(val); - } + val = fn.setKeyOrDefault("databaseUrl", defaultOptions.getDatabaseUrl()); + if (val != null) builder.setDatabaseUrl(val); - val = fn.setKeyOrDefault("clientId", - defaultOptions.getApplicationId()); - if (val != null) { - builder.setApplicationId(val); - } + val = fn.setKeyOrDefault("clientId", defaultOptions.getApplicationId()); + if (val != null) builder.setApplicationId(val); // if (params.hasKey("applicationId")) { @@ -156,24 +144,23 @@ public String setKeyOrDefault( // } try { - Log.i(TAG, "Configuring app"); - if (app == null) { - app = FirebaseApp.initializeApp(this.context, builder.build()); - } - Log.i(TAG, "Configured"); + Log.i(TAG, "Configuring app"); + if (app == null) { + app = FirebaseApp.initializeApp(getReactApplicationContext().getBaseContext(), builder.build()); + } + Log.i(TAG, "Configured"); - WritableMap resp = Arguments.createMap(); - resp.putString("msg", "success"); - onComplete.invoke(null, resp); - } - catch (Exception ex){ - Log.e(TAG, "ERROR configureWithOptions"); - Log.e(TAG, ex.getMessage()); + WritableMap resp = Arguments.createMap(); + resp.putString("msg", "success"); + onComplete.invoke(null, resp); + } catch (Exception ex) { + Log.e(TAG, "ERROR configureWithOptions"); + Log.e(TAG, ex.getMessage()); - WritableMap resp = Arguments.createMap(); - resp.putString("msg", ex.getMessage()); + WritableMap resp = Arguments.createMap(); + resp.putString("msg", ex.getMessage()); - onComplete.invoke(resp); + onComplete.invoke(resp); } } @@ -186,26 +173,36 @@ public void serverValue(@Nullable final Callback onComplete) { WritableMap map = Arguments.createMap(); map.putMap("TIMESTAMP", timestampMap); - onComplete.invoke(null, map); + if (onComplete != null) onComplete.invoke(null, map); } - // Internal helpers - @Override - public void onHostResume() { - WritableMap params = Arguments.createMap(); - params.putBoolean("isForground", true); - FirestackUtils.sendEvent(mReactContext, "FirestackAppState", params); - } + // Internal helpers + @Override + public void onHostResume() { + WritableMap params = Arguments.createMap(); + params.putBoolean("isForground", true); + Utils.sendEvent(getReactApplicationContext(), "FirestackAppState", params); + } - @Override - public void onHostPause() { - WritableMap params = Arguments.createMap(); - params.putBoolean("isForground", false); - FirestackUtils.sendEvent(mReactContext, "FirestackAppState", params); - } + @Override + public void onHostPause() { + WritableMap params = Arguments.createMap(); + params.putBoolean("isForground", false); + Utils.sendEvent(getReactApplicationContext(), "FirestackAppState", params); + } - @Override - public void onHostDestroy() { + @Override + public void onHostDestroy() { - } -} \ No newline at end of file + } + + @Override + public Map<String, Object> getConstants() { + final Map<String, Object> constants = new HashMap<>(); + constants.put("googleApiAvailability", getPlayServicesStatus()); + + // TODO remove once this has been moved on ios + constants.put("serverValueTimestamp", ServerValue.TIMESTAMP); + return constants; + } +} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackPackage.java b/android/src/main/java/io/fullstack/firestack/FirestackPackage.java index ea5927d..9a22981 100644 --- a/android/src/main/java/io/fullstack/firestack/FirestackPackage.java +++ b/android/src/main/java/io/fullstack/firestack/FirestackPackage.java @@ -6,12 +6,20 @@ import com.facebook.react.bridge.JavaScriptModule; import com.facebook.react.bridge.NativeModule; import com.facebook.react.bridge.ReactApplicationContext; +import com.facebook.react.uimanager.UIManagerModule; import com.facebook.react.uimanager.ViewManager; +import java.util.List; import java.util.ArrayList; import java.util.Collections; -import java.util.List; +import io.fullstack.firestack.auth.FirestackAuth; +import io.fullstack.firestack.storage.FirestackStorage; +import io.fullstack.firestack.database.FirestackDatabase; +import io.fullstack.firestack.analytics.FirestackAnalytics; +import io.fullstack.firestack.messaging.FirestackMessaging; + +@SuppressWarnings("unused") public class FirestackPackage implements ReactPackage { private Context mContext; @@ -24,13 +32,12 @@ public FirestackPackage() { @Override public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) { List<NativeModule> modules = new ArrayList<>(); - - modules.add(new FirestackModule(reactContext, reactContext.getBaseContext())); - modules.add(new FirestackAuthModule(reactContext)); - modules.add(new FirestackDatabaseModule(reactContext)); - modules.add(new FirestackAnalyticsModule(reactContext)); - modules.add(new FirestackStorageModule(reactContext)); - modules.add(new FirestackCloudMessaging(reactContext)); + modules.add(new FirestackModule(reactContext)); + modules.add(new FirestackAuth(reactContext)); + modules.add(new FirestackDatabase(reactContext)); + modules.add(new FirestackAnalytics(reactContext)); + modules.add(new FirestackStorage(reactContext)); + modules.add(new FirestackMessaging(reactContext)); return modules; } @@ -54,4 +61,4 @@ public List<Class<? extends JavaScriptModule>> createJSModules() { public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) { return Collections.emptyList(); } -} \ No newline at end of file +} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackStorage.java b/android/src/main/java/io/fullstack/firestack/FirestackStorage.java deleted file mode 100644 index 25ca218..0000000 --- a/android/src/main/java/io/fullstack/firestack/FirestackStorage.java +++ /dev/null @@ -1,298 +0,0 @@ -package io.fullstack.firestack; - -import android.os.Environment; -import android.os.StatFs; -import android.content.Context; -import android.util.Log; -import java.util.Map; -import java.util.HashMap; -import java.io.File; -import java.io.FileInputStream; -import java.io.InputStream; -import java.io.FileNotFoundException; - -import android.net.Uri; -import android.provider.MediaStore; -import android.database.Cursor; -import android.support.annotation.NonNull; -import android.support.annotation.Nullable; - -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.ReactApplicationContext; -import com.facebook.react.bridge.ReactContextBaseJavaModule; -import com.facebook.react.bridge.ReactMethod; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.bridge.ReadableMap; -import com.facebook.react.modules.core.DeviceEventManagerModule; -import com.facebook.react.bridge.ReactContext; - -import com.google.android.gms.tasks.OnCompleteListener; -import com.google.android.gms.tasks.OnFailureListener; -import com.google.android.gms.tasks.OnSuccessListener; -import com.google.android.gms.tasks.Task; - -import com.google.firebase.storage.OnProgressListener; -import com.google.firebase.storage.OnPausedListener; - -import com.google.firebase.FirebaseApp; - -import com.google.firebase.storage.FirebaseStorage; -import com.google.firebase.storage.UploadTask; - -import com.google.firebase.storage.StorageMetadata; -import com.google.firebase.storage.StorageReference; - -class FirestackStorageModule extends ReactContextBaseJavaModule { - - private static final String TAG = "FirestackStorage"; - private static final String DocumentDirectoryPath = "DOCUMENT_DIRECTORY_PATH"; - private static final String ExternalDirectoryPath = "EXTERNAL_DIRECTORY_PATH"; - private static final String ExternalStorageDirectoryPath = "EXTERNAL_STORAGE_DIRECTORY_PATH"; - private static final String PicturesDirectoryPath = "PICTURES_DIRECTORY_PATH"; - private static final String TemporaryDirectoryPath = "TEMPORARY_DIRECTORY_PATH"; - private static final String CachesDirectoryPath = "CACHES_DIRECTORY_PATH"; - private static final String DocumentDirectory = "DOCUMENT_DIRECTORY_PATH"; - - private static final String FileTypeRegular = "FILETYPE_REGULAR"; - private static final String FileTypeDirectory = "FILETYPE_DIRECTORY"; - - - private Context context; - private ReactContext mReactContext; - private FirebaseApp app; - - public FirestackStorageModule(ReactApplicationContext reactContext) { - super(reactContext); - this.context = reactContext; - mReactContext = reactContext; - - Log.d(TAG, "New instance"); - } - - @Override - public String getName() { - return TAG; - } - - @ReactMethod - public void downloadUrl(final String javascriptStorageBucket, - final String path, - final Callback callback) { - FirebaseStorage storage = FirebaseStorage.getInstance(); - String storageBucket = storage.getApp().getOptions().getStorageBucket(); - String storageUrl = "gs://"+storageBucket; - Log.d(TAG, "Storage url " + storageUrl + path); - final StorageReference storageRef = storage.getReferenceFromUrl(storageUrl); - final StorageReference fileRef = storageRef.child(path); - - Task<Uri> downloadTask = fileRef.getDownloadUrl(); - downloadTask.addOnSuccessListener(new OnSuccessListener<Uri>() { - @Override - public void onSuccess(Uri uri) { - final WritableMap res = Arguments.createMap(); - - res.putString("status", "success"); - res.putString("bucket", storageRef.getBucket()); - res.putString("fullPath", uri.toString()); - res.putString("path", uri.getPath()); - - storageRef.getMetadata() - .addOnSuccessListener(new OnSuccessListener<StorageMetadata>() { - @Override - public void onSuccess(final StorageMetadata storageMetadata) { - Log.d(TAG, "getMetadata success " + storageMetadata); - res.putString("name", storageMetadata.getName()); - - WritableMap metadata = Arguments.createMap(); - metadata.putString("getBucket", storageMetadata.getBucket()); - metadata.putString("getName", storageMetadata.getName()); - metadata.putDouble("sizeBytes", storageMetadata.getSizeBytes()); - metadata.putDouble("created_at", storageMetadata.getCreationTimeMillis()); - metadata.putDouble("updated_at", storageMetadata.getUpdatedTimeMillis()); - metadata.putString("md5hash", storageMetadata.getMd5Hash()); - metadata.putString("encoding", storageMetadata.getContentEncoding()); - res.putString("url", storageMetadata.getDownloadUrl().toString()); - - res.putMap("metadata", metadata); - callback.invoke(null, res); - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception exception) { - Log.e(TAG, "Failure in download " + exception); - callback.invoke(makeErrorPayload(1, exception)); - } - }); - - } - }).addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception exception) { - Log.e(TAG, "Failed to download file " + exception.getMessage()); - - WritableMap err = Arguments.createMap(); - err.putString("status", "error"); - err.putString("description", exception.getLocalizedMessage()); - - callback.invoke(err); - } - }); - } - - // STORAGE - @ReactMethod - public void uploadFile(final String urlStr, final String name, final String filepath, final ReadableMap metadata, final Callback callback) { - FirebaseStorage storage = FirebaseStorage.getInstance(); - - StorageReference storageRef = storage.getReferenceFromUrl(urlStr); - StorageReference fileRef = storageRef.child(name); - -Log.i(TAG, "From file: " + filepath + " to " + urlStr + " with name " + name); - try { - // InputStream stream = new FileInputStream(new File(filepath)); - Uri file = Uri.fromFile(new File(filepath)); - - StorageMetadata.Builder metadataBuilder = new StorageMetadata.Builder(); - Map<String, Object> m = FirestackUtils.recursivelyDeconstructReadableMap(metadata); - - StorageMetadata md = metadataBuilder.build(); - UploadTask uploadTask = fileRef.putFile(file, md); - // UploadTask uploadTask = fileRef.putStream(stream, md); - - // Register observers to listen for when the download is done or if it fails - uploadTask.addOnFailureListener(new OnFailureListener() { - @Override - public void onFailure(@NonNull Exception exception) { - // Handle unsuccessful uploads - Log.e(TAG, "Failed to upload file " + exception.getMessage()); - - WritableMap err = Arguments.createMap(); - err.putString("description", exception.getLocalizedMessage()); - - callback.invoke(err); - } - }).addOnSuccessListener(new OnSuccessListener<UploadTask.TaskSnapshot>() { - @Override - public void onSuccess(UploadTask.TaskSnapshot taskSnapshot) { - Log.d(TAG, "Successfully uploaded file " + taskSnapshot); - // taskSnapshot.getMetadata() contains file metadata such as size, content-type, and download URL. - WritableMap resp = getDownloadData(taskSnapshot); - // NSDictionary *props = @{ - // @"fullPath": ref.fullPath, - // @"bucket": ref.bucket, - // @"name": ref.name, - // @"metadata": [snapshot.metadata dictionaryRepresentation] - // }; - - callback.invoke(null, resp); - } - }) - .addOnProgressListener(new OnProgressListener<UploadTask.TaskSnapshot>() { - @Override - public void onProgress(UploadTask.TaskSnapshot taskSnapshot) { - double totalBytes = taskSnapshot.getTotalByteCount(); - double bytesTransferred = taskSnapshot.getBytesTransferred(); - double progress = (100.0 * bytesTransferred) / totalBytes; - - System.out.println("Transferred " + bytesTransferred + "/" + totalBytes + "("+progress + "% complete)"); - - if (progress >= 0) { - WritableMap data = Arguments.createMap(); - data.putString("eventName", "upload_progress"); - data.putDouble("progress", progress); - FirestackUtils.sendEvent(mReactContext, "upload_progress", data); - } - } - }).addOnPausedListener(new OnPausedListener<UploadTask.TaskSnapshot>() { - @Override - public void onPaused(UploadTask.TaskSnapshot taskSnapshot) { - System.out.println("Upload is paused"); - StorageMetadata d = taskSnapshot.getMetadata(); - String bucket = d.getBucket(); - WritableMap data = Arguments.createMap(); - data.putString("eventName", "upload_paused"); - data.putString("ref", bucket); - FirestackUtils.sendEvent(mReactContext, "upload_paused", data); - } - }); - } - catch (Exception ex) { - callback.invoke(makeErrorPayload(2, ex)); - } - } - - @ReactMethod - public void getRealPathFromURI(final String uri, final Callback callback) { - try { - Context context = getReactApplicationContext(); - String [] proj = {MediaStore.Images.Media.DATA}; - Cursor cursor = context.getContentResolver().query(Uri.parse(uri), proj, null, null, null); - int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA); - cursor.moveToFirst(); - String path = cursor.getString(column_index); - cursor.close(); - - callback.invoke(null, path); - } catch (Exception ex) { - ex.printStackTrace(); - callback.invoke(makeErrorPayload(1, ex)); - } - } - - private WritableMap getDownloadData(final UploadTask.TaskSnapshot taskSnapshot) { - Uri downloadUrl = taskSnapshot.getDownloadUrl(); - StorageMetadata d = taskSnapshot.getMetadata(); - - WritableMap resp = Arguments.createMap(); - resp.putString("downloadUrl", downloadUrl.toString()); - resp.putString("fullPath", d.getPath()); - resp.putString("bucket", d.getBucket()); - resp.putString("name", d.getName()); - - WritableMap metadataObj = Arguments.createMap(); - metadataObj.putString("cacheControl", d.getCacheControl()); - metadataObj.putString("contentDisposition", d.getContentDisposition()); - metadataObj.putString("contentType", d.getContentType()); - resp.putMap("metadata", metadataObj); - - return resp; - } - - private WritableMap makeErrorPayload(double code, Exception ex) { - WritableMap error = Arguments.createMap(); - error.putDouble("code", code); - error.putString("message", ex.getMessage()); - return error; - } - - // Comes almost directory from react-native-fs - @Override - public Map<String, Object> getConstants() { - final Map<String, Object> constants = new HashMap<>(); - - constants.put(DocumentDirectory, 0); - constants.put(DocumentDirectoryPath, this.getReactApplicationContext().getFilesDir().getAbsolutePath()); - constants.put(TemporaryDirectoryPath, null); - constants.put(PicturesDirectoryPath, Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getAbsolutePath()); - constants.put(CachesDirectoryPath, this.getReactApplicationContext().getCacheDir().getAbsolutePath()); - constants.put(FileTypeRegular, 0); - constants.put(FileTypeDirectory, 1); - - File externalStorageDirectory = Environment.getExternalStorageDirectory(); - if (externalStorageDirectory != null) { - constants.put(ExternalStorageDirectoryPath, externalStorageDirectory.getAbsolutePath()); - } else { - constants.put(ExternalStorageDirectoryPath, null); - } - - File externalDirectory = this.getReactApplicationContext().getExternalFilesDir(null); - if (externalDirectory != null) { - constants.put(ExternalDirectoryPath, externalDirectory.getAbsolutePath()); - } else { - constants.put(ExternalDirectoryPath, null); - } - - return constants; - } -} diff --git a/android/src/main/java/io/fullstack/firestack/FirestackUtils.java b/android/src/main/java/io/fullstack/firestack/FirestackUtils.java deleted file mode 100644 index 32c871a..0000000 --- a/android/src/main/java/io/fullstack/firestack/FirestackUtils.java +++ /dev/null @@ -1,228 +0,0 @@ -package io.fullstack.firestack; - -import android.util.Log; - -import java.util.ArrayList; -import java.util.HashMap; -import java.util.List; -import java.util.Map; - -import com.facebook.react.bridge.ReactContext; -import com.facebook.react.modules.core.DeviceEventManagerModule; -import com.facebook.react.bridge.Arguments; -import com.facebook.react.bridge.Callback; -import com.facebook.react.bridge.WritableMap; -import com.facebook.react.bridge.WritableArray; -import com.facebook.react.bridge.ReadableMap; - -import com.facebook.react.bridge.ReadableArray; -import com.facebook.react.bridge.ReadableMapKeySetIterator; -import com.facebook.react.bridge.ReadableType; -import com.google.firebase.database.DataSnapshot; - -public class FirestackUtils { - private static final String TAG = "FirestackUtils"; - - // TODO NOTE - public static void todoNote(final String tag, final String name, final Callback callback) { - Log.e(tag, "The method " + name + " has not yet been implemented."); - Log.e(tag, "Feel free to contribute to finish the method in the source."); - - WritableMap errorMap = Arguments.createMap(); - errorMap.putString("error", "unimplemented"); - callback.invoke(null, errorMap); - } - - /** - * send a JS event - **/ - public static void sendEvent(final ReactContext context, - final String eventName, - final WritableMap params) { - if (context.hasActiveCatalystInstance()) { - context - .getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) - .emit(eventName, params); - } else { - Log.d(TAG, "Waiting for CatalystInstance before sending event"); - } - } - - // snapshot - public static WritableMap dataSnapshotToMap(String name, - String path, - DataSnapshot dataSnapshot) { - WritableMap data = Arguments.createMap(); - - data.putString("key", dataSnapshot.getKey()); - data.putBoolean("exists", dataSnapshot.exists()); - data.putBoolean("hasChildren", dataSnapshot.hasChildren()); - - data.putDouble("childrenCount", dataSnapshot.getChildrenCount()); - if (!dataSnapshot.hasChildren()) { - Object value = dataSnapshot.getValue(); - String type = value!=null ? value.getClass().getName() : ""; - switch (type) { - case "java.lang.Boolean": - data.putBoolean("value", (Boolean)value); - break; - case "java.lang.Long": - Long longVal = (Long) value; - data.putDouble("value", (double)longVal); - break; - case "java.lang.Double": - data.putDouble("value", (Double) value); - break; - case "java.lang.String": - data.putString("value",(String) value); - break; - default: - data.putString("value", null); - } - } else{ - WritableMap valueMap = FirestackUtils.castSnapshotValue(dataSnapshot); - data.putMap("value", valueMap); - } - - // Child keys - WritableArray childKeys = FirestackUtils.getChildKeys(dataSnapshot); - data.putArray("childKeys", childKeys); - - Object priority = dataSnapshot.getPriority(); - if (priority == null) { - data.putString("priority", null); - } else { - data.putString("priority", priority.toString()); - } - - WritableMap eventMap = Arguments.createMap(); - eventMap.putString("eventName", name); - eventMap.putMap("snapshot", data); - eventMap.putString("path", path); - return eventMap; - } - - public static <Any> Any castSnapshotValue(DataSnapshot snapshot) { - if (snapshot.hasChildren()) { - WritableMap data = Arguments.createMap(); - for (DataSnapshot child : snapshot.getChildren()) { - Any castedChild = castSnapshotValue(child); - switch (castedChild.getClass().getName()) { - case "java.lang.Boolean": - data.putBoolean(child.getKey(), (Boolean) castedChild); - break; - case "java.lang.Long": - Long longVal = (Long) castedChild; - data.putDouble(child.getKey(), (double)longVal); - break; - case "java.lang.Double": - data.putDouble(child.getKey(), (Double) castedChild); - break; - case "java.lang.String": - data.putString(child.getKey(), (String) castedChild); - break; - case "com.facebook.react.bridge.WritableNativeMap": - data.putMap(child.getKey(), (WritableMap) castedChild); - break; - default: - Log.w(TAG, "Invalid type: " + castedChild.getClass().getName()); - break; - } - } - return (Any) data; - } else { - if (snapshot.getValue() != null) { - String type = snapshot.getValue().getClass().getName(); - switch (type) { - case "java.lang.Boolean": - return (Any)(snapshot.getValue()); - case "java.lang.Long": - return (Any)(snapshot.getValue()); - case "java.lang.Double": - return (Any)(snapshot.getValue()); - case "java.lang.String": - return (Any)(snapshot.getValue()); - default: - Log.w(TAG, "Invalid type: "+type); - return (Any) null; - } - } - return (Any) null; - } - } - - public static WritableArray getChildKeys(DataSnapshot snapshot) { - WritableArray childKeys = Arguments.createArray(); - - if (snapshot.hasChildren()) { - for (DataSnapshot child : snapshot.getChildren()) { - childKeys.pushString(child.getKey()); - } - } - - return childKeys; - } - - public static Map<String, Object> recursivelyDeconstructReadableMap(ReadableMap readableMap) { - ReadableMapKeySetIterator iterator = readableMap.keySetIterator(); - Map<String, Object> deconstructedMap = new HashMap<>(); - while (iterator.hasNextKey()) { - String key = iterator.nextKey(); - ReadableType type = readableMap.getType(key); - switch (type) { - case Null: - deconstructedMap.put(key, null); - break; - case Boolean: - deconstructedMap.put(key, readableMap.getBoolean(key)); - break; - case Number: - deconstructedMap.put(key, readableMap.getDouble(key)); - break; - case String: - deconstructedMap.put(key, readableMap.getString(key)); - break; - case Map: - deconstructedMap.put(key, FirestackUtils.recursivelyDeconstructReadableMap(readableMap.getMap(key))); - break; - case Array: - deconstructedMap.put(key, FirestackUtils.recursivelyDeconstructReadableArray(readableMap.getArray(key))); - break; - default: - throw new IllegalArgumentException("Could not convert object with key: " + key + "."); - } - - } - return deconstructedMap; - } - - public static List<Object> recursivelyDeconstructReadableArray(ReadableArray readableArray) { - List<Object> deconstructedList = new ArrayList<>(readableArray.size()); - for (int i = 0; i < readableArray.size(); i++) { - ReadableType indexType = readableArray.getType(i); - switch(indexType) { - case Null: - deconstructedList.add(i, null); - break; - case Boolean: - deconstructedList.add(i, readableArray.getBoolean(i)); - break; - case Number: - deconstructedList.add(i, readableArray.getDouble(i)); - break; - case String: - deconstructedList.add(i, readableArray.getString(i)); - break; - case Map: - deconstructedList.add(i, FirestackUtils.recursivelyDeconstructReadableMap(readableArray.getMap(i))); - break; - case Array: - deconstructedList.add(i, FirestackUtils.recursivelyDeconstructReadableArray(readableArray.getArray(i))); - break; - default: - throw new IllegalArgumentException("Could not convert object at index " + i + "."); - } - } - return deconstructedList; - } -} diff --git a/android/src/main/java/io/fullstack/firestack/Utils.java b/android/src/main/java/io/fullstack/firestack/Utils.java new file mode 100644 index 0000000..a763204 --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/Utils.java @@ -0,0 +1,301 @@ +package io.fullstack.firestack; + +import android.util.Log; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.Callback; +import com.facebook.react.bridge.WritableMap; +import com.facebook.react.bridge.ReadableMap; +import com.facebook.react.bridge.ReactContext; +import com.facebook.react.bridge.WritableArray; +import com.facebook.react.bridge.WritableNativeArray; +import com.facebook.react.modules.core.DeviceEventManagerModule; + +import com.facebook.react.bridge.ReadableType; +import com.facebook.react.bridge.ReadableArray; +import com.google.firebase.database.DataSnapshot; +import com.facebook.react.bridge.ReadableMapKeySetIterator; + +@SuppressWarnings("WeakerAccess") +public class Utils { + private static final String TAG = "Utils"; + + // TODO NOTE + public static void todoNote(final String tag, final String name, final Callback callback) { + Log.e(tag, "The method " + name + " has not yet been implemented."); + Log.e(tag, "Feel free to contribute to finish the method in the source."); + + WritableMap errorMap = Arguments.createMap(); + errorMap.putString("error", "unimplemented"); + callback.invoke(null, errorMap); + } + + /** + * send a JS event + **/ + public static void sendEvent(final ReactContext context, final String eventName, final WritableMap params) { + if (context != null) { + context + .getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) + .emit(eventName, params); + } else { + Log.d(TAG, "Missing context - cannot send event!"); + } + } + + // snapshot + public static WritableMap dataSnapshotToMap( + String name, + String path, + String modifiersString, + DataSnapshot dataSnapshot + ) { + WritableMap data = Arguments.createMap(); + + data.putString("key", dataSnapshot.getKey()); + data.putBoolean("exists", dataSnapshot.exists()); + data.putBoolean("hasChildren", dataSnapshot.hasChildren()); + + data.putDouble("childrenCount", dataSnapshot.getChildrenCount()); + if (!dataSnapshot.hasChildren()) { + Object value = dataSnapshot.getValue(); + String type = value != null ? value.getClass().getName() : ""; + switch (type) { + case "java.lang.Boolean": + data.putBoolean("value", (Boolean) value); + break; + case "java.lang.Long": + Long longVal = (Long) value; + data.putDouble("value", (double) longVal); + break; + case "java.lang.Double": + data.putDouble("value", (Double) value); + break; + case "java.lang.String": + data.putString("value", (String) value); + break; + default: + data.putString("value", null); + } + } else { + Object value = Utils.castSnapshotValue(dataSnapshot); + if (value instanceof WritableNativeArray) { + data.putArray("value", (WritableArray) value); + } else { + data.putMap("value", (WritableMap) value); + } + } + + // Child keys + WritableArray childKeys = Utils.getChildKeys(dataSnapshot); + data.putArray("childKeys", childKeys); + + Object priority = dataSnapshot.getPriority(); + if (priority == null) { + data.putString("priority", null); + } else { + data.putString("priority", priority.toString()); + } + + WritableMap eventMap = Arguments.createMap(); + eventMap.putString("eventName", name); + eventMap.putMap("snapshot", data); + eventMap.putString("path", path); + eventMap.putString("modifiersString", modifiersString); + return eventMap; + } + + public static <Any> Any castSnapshotValue(DataSnapshot snapshot) { + if (snapshot.hasChildren()) { + if (isArray(snapshot)) { + return (Any) buildArray(snapshot); + } else { + return (Any) buildMap(snapshot); + } + } else { + if (snapshot.getValue() != null) { + String type = snapshot.getValue().getClass().getName(); + switch (type) { + case "java.lang.Boolean": + return (Any) (snapshot.getValue()); + case "java.lang.Long": + return (Any) (snapshot.getValue()); + case "java.lang.Double": + return (Any) (snapshot.getValue()); + case "java.lang.String": + return (Any) (snapshot.getValue()); + default: + Log.w(TAG, "Invalid type: " + type); + return (Any) null; + } + } + return (Any) null; + } + } + + private static boolean isArray(DataSnapshot snapshot) { + long expectedKey = 0; + for (DataSnapshot child : snapshot.getChildren()) { + try { + long key = Long.parseLong(child.getKey()); + if (key == expectedKey) { + expectedKey++; + } else { + return false; + } + } catch (NumberFormatException ex) { + return false; + } + } + return true; + } + + private static <Any> WritableArray buildArray(DataSnapshot snapshot) { + WritableArray array = Arguments.createArray(); + for (DataSnapshot child : snapshot.getChildren()) { + Any castedChild = castSnapshotValue(child); + switch (castedChild.getClass().getName()) { + case "java.lang.Boolean": + array.pushBoolean((Boolean) castedChild); + break; + case "java.lang.Long": + Long longVal = (Long) castedChild; + array.pushDouble((double) longVal); + break; + case "java.lang.Double": + array.pushDouble((Double) castedChild); + break; + case "java.lang.String": + array.pushString((String) castedChild); + break; + case "com.facebook.react.bridge.WritableNativeMap": + array.pushMap((WritableMap) castedChild); + break; + case "com.facebook.react.bridge.WritableNativeArray": + array.pushArray((WritableArray) castedChild); + break; + default: + Log.w(TAG, "Invalid type: " + castedChild.getClass().getName()); + break; + } + } + return array; + } + + private static <Any> WritableMap buildMap(DataSnapshot snapshot) { + WritableMap map = Arguments.createMap(); + for (DataSnapshot child : snapshot.getChildren()) { + Any castedChild = castSnapshotValue(child); + + switch (castedChild.getClass().getName()) { + case "java.lang.Boolean": + map.putBoolean(child.getKey(), (Boolean) castedChild); + break; + case "java.lang.Long": + Long longVal = (Long) castedChild; + map.putDouble(child.getKey(), (double) longVal); + break; + case "java.lang.Double": + map.putDouble(child.getKey(), (Double) castedChild); + break; + case "java.lang.String": + map.putString(child.getKey(), (String) castedChild); + break; + case "com.facebook.react.bridge.WritableNativeMap": + map.putMap(child.getKey(), (WritableMap) castedChild); + break; + case "com.facebook.react.bridge.WritableNativeArray": + map.putArray(child.getKey(), (WritableArray) castedChild); + break; + default: + Log.w(TAG, "Invalid type: " + castedChild.getClass().getName()); + break; + } + } + return map; + } + + public static WritableArray getChildKeys(DataSnapshot snapshot) { + WritableArray childKeys = Arguments.createArray(); + + if (snapshot.hasChildren()) { + for (DataSnapshot child : snapshot.getChildren()) { + childKeys.pushString(child.getKey()); + } + } + + return childKeys; + } + + public static Map<String, Object> recursivelyDeconstructReadableMap(ReadableMap readableMap) { + Map<String, Object> deconstructedMap = new HashMap<>(); + if (readableMap == null) { + return deconstructedMap; + } + + ReadableMapKeySetIterator iterator = readableMap.keySetIterator(); + while (iterator.hasNextKey()) { + String key = iterator.nextKey(); + ReadableType type = readableMap.getType(key); + switch (type) { + case Null: + deconstructedMap.put(key, null); + break; + case Boolean: + deconstructedMap.put(key, readableMap.getBoolean(key)); + break; + case Number: + deconstructedMap.put(key, readableMap.getDouble(key)); + break; + case String: + deconstructedMap.put(key, readableMap.getString(key)); + break; + case Map: + deconstructedMap.put(key, Utils.recursivelyDeconstructReadableMap(readableMap.getMap(key))); + break; + case Array: + deconstructedMap.put(key, Utils.recursivelyDeconstructReadableArray(readableMap.getArray(key))); + break; + default: + throw new IllegalArgumentException("Could not convert object with key: " + key + "."); + } + + } + return deconstructedMap; + } + + public static List<Object> recursivelyDeconstructReadableArray(ReadableArray readableArray) { + List<Object> deconstructedList = new ArrayList<>(readableArray.size()); + for (int i = 0; i < readableArray.size(); i++) { + ReadableType indexType = readableArray.getType(i); + switch (indexType) { + case Null: + deconstructedList.add(i, null); + break; + case Boolean: + deconstructedList.add(i, readableArray.getBoolean(i)); + break; + case Number: + deconstructedList.add(i, readableArray.getDouble(i)); + break; + case String: + deconstructedList.add(i, readableArray.getString(i)); + break; + case Map: + deconstructedList.add(i, Utils.recursivelyDeconstructReadableMap(readableArray.getMap(i))); + break; + case Array: + deconstructedList.add(i, Utils.recursivelyDeconstructReadableArray(readableArray.getArray(i))); + break; + default: + throw new IllegalArgumentException("Could not convert object at index " + i + "."); + } + } + return deconstructedList; + } +} diff --git a/android/src/main/java/io/fullstack/firestack/analytics/FirestackAnalytics.java b/android/src/main/java/io/fullstack/firestack/analytics/FirestackAnalytics.java new file mode 100644 index 0000000..14e6ac2 --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/analytics/FirestackAnalytics.java @@ -0,0 +1,103 @@ +package io.fullstack.firestack.analytics; + +import android.util.Log; +import android.app.Activity; +import android.support.annotation.Nullable; + +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.ReactMethod; +import com.facebook.react.bridge.ReadableMap; +import com.google.firebase.analytics.FirebaseAnalytics; +import com.facebook.react.bridge.ReactApplicationContext; +import com.facebook.react.bridge.ReactContextBaseJavaModule; + + +public class FirestackAnalytics extends ReactContextBaseJavaModule { + + private static final String TAG = "FirestackAnalytics"; + + public FirestackAnalytics(ReactApplicationContext reactContext) { + super(reactContext); + Log.d(TAG, "New instance"); + } + + /** + * + * @return + */ + @Override + public String getName() { + return TAG; + } + + @ReactMethod + public void logEvent(final String name, @Nullable final ReadableMap params) { + FirebaseAnalytics.getInstance(getReactApplicationContext()).logEvent(name, Arguments.toBundle(params)); + } + + /** + * + * @param enabled + */ + @ReactMethod + public void setAnalyticsCollectionEnabled(final Boolean enabled) { + FirebaseAnalytics.getInstance(getReactApplicationContext()).setAnalyticsCollectionEnabled(enabled); + } + + /** + * + * @param screenName + * @param screenClassOverride + */ + @ReactMethod + public void setCurrentScreen(final String screenName, final String screenClassOverride) { + final Activity activity = getCurrentActivity(); + if (activity != null) { + // needs to be run on main thread + Log.d(TAG, "setCurrentScreen " + screenName + " - " + screenClassOverride); + activity.runOnUiThread(new Runnable() { + @Override + public void run() { + FirebaseAnalytics.getInstance(getReactApplicationContext()).setCurrentScreen(activity, screenName, screenClassOverride); + } + }); + } + } + + /** + * + * @param milliseconds + */ + @ReactMethod + public void setMinimumSessionDuration(final double milliseconds) { + FirebaseAnalytics.getInstance(getReactApplicationContext()).setMinimumSessionDuration((long) milliseconds); + } + + /** + * + * @param milliseconds + */ + @ReactMethod + public void setSessionTimeoutDuration(final double milliseconds) { + FirebaseAnalytics.getInstance(getReactApplicationContext()).setSessionTimeoutDuration((long) milliseconds); + } + + /** + * + * @param id + */ + @ReactMethod + public void setUserId(final String id) { + FirebaseAnalytics.getInstance(getReactApplicationContext()).setUserId(id); + } + + /** + * + * @param name + * @param value + */ + @ReactMethod + public void setUserProperty(final String name, final String value) { + FirebaseAnalytics.getInstance(getReactApplicationContext()).setUserProperty(name, value); + } +} diff --git a/android/src/main/java/io/fullstack/firestack/auth/FirestackAuth.java b/android/src/main/java/io/fullstack/firestack/auth/FirestackAuth.java new file mode 100644 index 0000000..3f194ba --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/auth/FirestackAuth.java @@ -0,0 +1,619 @@ + +package io.fullstack.firestack.auth; + +import android.util.Log; + +import java.util.Map; + +import android.net.Uri; +import android.support.annotation.NonNull; + +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.ReactApplicationContext; +import com.facebook.react.bridge.ReactContextBaseJavaModule; +import com.facebook.react.bridge.ReactMethod; +import com.facebook.react.bridge.Callback; +import com.facebook.react.bridge.WritableMap; +import com.facebook.react.bridge.ReadableMap; +import com.facebook.react.bridge.ReactContext; + +import com.google.android.gms.tasks.OnCompleteListener; +import com.google.android.gms.tasks.Task; + +import com.google.firebase.auth.AuthCredential; +import com.google.firebase.auth.AuthResult; +import com.google.firebase.auth.UserProfileChangeRequest; +import com.google.firebase.auth.FacebookAuthProvider; +import com.google.firebase.auth.FirebaseAuth; +import com.google.firebase.auth.FirebaseAuthException; +import com.google.firebase.auth.FirebaseUser; +import com.google.firebase.auth.GetTokenResult; +import com.google.firebase.auth.GoogleAuthProvider; +import com.google.firebase.auth.EmailAuthProvider; + + +import io.fullstack.firestack.Utils; + + +@SuppressWarnings("ThrowableResultOfMethodCallIgnored") +public class FirestackAuth extends ReactContextBaseJavaModule { + private final int NO_CURRENT_USER = 100; + private final int ERROR_FETCHING_TOKEN = 101; + private final int ERROR_SENDING_VERIFICATION_EMAIL = 102; + + private static final String TAG = "FirestackAuth"; + + // private Context context; + private ReactContext mReactContext; + private FirebaseAuth mAuth; + private FirebaseAuth.AuthStateListener mAuthListener; + + public FirestackAuth(ReactApplicationContext reactContext) { + super(reactContext); + mReactContext = reactContext; + mAuth = FirebaseAuth.getInstance(); + + Log.d(TAG, "New FirestackAuth instance"); + } + + @Override + public String getName() { + return TAG; + } + + /** + * Returns a no user error. + * + * @param callback JS callback + */ + private void callbackNoUser(Callback callback, Boolean isError) { + WritableMap err = Arguments.createMap(); + err.putInt("errorCode", NO_CURRENT_USER); + err.putString("errorMessage", "No current user"); + + if (isError) { + callback.invoke(err); + } else { + callback.invoke(null, null); + } + } + + @ReactMethod + public void listenForAuth() { + if (mAuthListener == null) { + mAuthListener = new FirebaseAuth.AuthStateListener() { + @Override + public void onAuthStateChanged(@NonNull FirebaseAuth firebaseAuth) { + FirebaseUser user = firebaseAuth.getCurrentUser(); + WritableMap msgMap = Arguments.createMap(); + msgMap.putString("eventName", "listenForAuth"); + + if (user != null) { + // TODO move to helper + WritableMap userMap = getUserMap(user); + msgMap.putBoolean("authenticated", true); + msgMap.putMap("user", userMap); + + Utils.sendEvent(mReactContext, "listenForAuth", msgMap); + } else { + msgMap.putBoolean("authenticated", false); + Utils.sendEvent(mReactContext, "listenForAuth", msgMap); + } + } + }; + mAuth.addAuthStateListener(mAuthListener); + } + } + + @ReactMethod + public void unlistenForAuth(final Callback callback) { + if (mAuthListener != null) { + mAuth.removeAuthStateListener(mAuthListener); + + // TODO move to helper + WritableMap resp = Arguments.createMap(); + resp.putString("status", "complete"); + + callback.invoke(null, resp); + } + } + + @ReactMethod + public void createUserWithEmail(final String email, final String password, final Callback callback) { + mAuth.createUserWithEmailAndPassword(email, password) + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + try { + if (task.isSuccessful()) { + userCallback(task.getResult().getUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + @ReactMethod + public void signInWithEmail(final String email, final String password, final Callback callback) { + + mAuth.signInWithEmailAndPassword(email, password) + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + try { + if (task.isSuccessful()) { + userCallback(task.getResult().getUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + @ReactMethod + public void signInWithProvider(final String provider, final String authToken, final String authSecret, final Callback callback) { + if (provider.equals("facebook")) { + this.facebookLogin(authToken, callback); + } else if (provider.equals("google")) { + this.googleLogin(authToken, callback); + } else + // TODO + Utils.todoNote(TAG, "signInWithProvider", callback); + } + + @ReactMethod + public void linkPassword(final String email, final String password, final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user != null) { + AuthCredential credential = EmailAuthProvider.getCredential(email, password); + user + .linkWithCredential(credential) + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + try { + if (task.isSuccessful()) { + Log.d(TAG, "user linked with password credential"); + userCallback(mAuth.getCurrentUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + @ReactMethod + public void link(final String provider, final String authToken, final String authSecret, final Callback callback) { + if (provider.equals("password")) { + linkPassword(authToken, authSecret, callback); + } else + // TODO other providers + Utils.todoNote(TAG, "linkWithProvider", callback); + } + + @ReactMethod + public void signInAnonymously(final Callback callback) { + Log.d(TAG, "signInAnonymously:called:"); + mAuth.signInAnonymously() + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + Log.d(TAG, "signInAnonymously:onComplete:" + task.isSuccessful()); + + try { + if (task.isSuccessful()) { + userCallback(task.getResult().getUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + @ReactMethod + public void signInWithCustomToken(final String customToken, final Callback callback) { + mAuth.signInWithCustomToken(customToken) + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + Log.d(TAG, "signInWithCustomToken:onComplete:" + task.isSuccessful()); + try { + if (task.isSuccessful()) { + userCallback(task.getResult().getUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + @ReactMethod + public void reauthenticateWithCredentialForProvider(final String provider, final String authToken, final String authSecret, final Callback callback) { + // TODO: + Utils.todoNote(TAG, "reauthenticateWithCredentialForProvider", callback); + // AuthCredential credential; + // Log.d(TAG, "reauthenticateWithCredentialForProvider called with: " + provider); + } + + @ReactMethod + public void updateUserEmail(final String email, final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user != null) { + user + .updateEmail(email) + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + try { + if (task.isSuccessful()) { + Log.d(TAG, "User email address updated"); + userCallback(mAuth.getCurrentUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + @ReactMethod + public void updateUserPassword(final String newPassword, final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user != null) { + user.updatePassword(newPassword) + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + try { + if (task.isSuccessful()) { + Log.d(TAG, "User password updated"); + userCallback(mAuth.getCurrentUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + @ReactMethod + public void sendPasswordResetWithEmail(final String email, final Callback callback) { + mAuth.sendPasswordResetEmail(email) + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + try { + if (task.isSuccessful()) { + WritableMap resp = Arguments.createMap(); + resp.putString("status", "complete"); + callback.invoke(null, resp); + } else { + callback.invoke(task.getException().toString()); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + @ReactMethod + public void deleteUser(final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + if (user != null) { + user.delete() + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + try { + if (task.isSuccessful()) { + Log.d(TAG, "User account deleted"); + WritableMap resp = Arguments.createMap(); + resp.putString("status", "complete"); + resp.putString("msg", "User account deleted"); + callback.invoke(null, resp); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + + @ReactMethod + public void sendEmailVerification(final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user != null) { + user.sendEmailVerification() + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + try { + if (task.isSuccessful()) { + WritableMap resp = Arguments.createMap(); + resp.putString("status", "complete"); + resp.putString("msg", "User verification email sent"); + callback.invoke(null, resp); + } else { + WritableMap err = Arguments.createMap(); + err.putInt("errorCode", ERROR_SENDING_VERIFICATION_EMAIL); + err.putString("errorMessage", task.getException().getMessage()); + callback.invoke(err); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + + @ReactMethod + public void getToken(final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user != null) { + user.getToken(true) + .addOnCompleteListener(new OnCompleteListener<GetTokenResult>() { + @Override + public void onComplete(@NonNull Task<GetTokenResult> task) { + try { + if (task.isSuccessful()) { + String token = task.getResult().getToken(); + WritableMap resp = Arguments.createMap(); + resp.putString("status", "complete"); + resp.putString("token", token); + callback.invoke(null, resp); + } else { + WritableMap err = Arguments.createMap(); + err.putInt("errorCode", ERROR_FETCHING_TOKEN); + err.putString("errorMessage", task.getException().getMessage()); + callback.invoke(err); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + @ReactMethod + public void updateUserProfile(ReadableMap props, final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user != null) { + UserProfileChangeRequest.Builder profileBuilder = new UserProfileChangeRequest.Builder(); + + Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(props); + + if (m.containsKey("displayName")) { + String displayName = (String) m.get("displayName"); + profileBuilder.setDisplayName(displayName); + } + + if (m.containsKey("photoUri")) { + String photoUriStr = (String) m.get("photoUri"); + Uri uri = Uri.parse(photoUriStr); + profileBuilder.setPhotoUri(uri); + } + + UserProfileChangeRequest profileUpdates = profileBuilder.build(); + + user.updateProfile(profileUpdates) + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + try { + if (task.isSuccessful()) { + Log.d(TAG, "User profile updated"); + userCallback(mAuth.getCurrentUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + @ReactMethod + public void signOut(final Callback callback) { + mAuth.signOut(); + + WritableMap resp = Arguments.createMap(); + resp.putString("status", "complete"); + resp.putString("msg", "User signed out"); + callback.invoke(null, resp); + } + + @ReactMethod + public void reloadUser(final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user == null) { + callbackNoUser(callback, false); + } else { + user.reload() + .addOnCompleteListener(new OnCompleteListener<Void>() { + @Override + public void onComplete(@NonNull Task<Void> task) { + if (task.isSuccessful()) { + Log.d(TAG, "user reloaded"); + userCallback(mAuth.getCurrentUser(), callback); + } else { + userErrorCallback(task, callback); + } + } + }); + } + } + + @ReactMethod + public void getCurrentUser(final Callback callback) { + FirebaseUser user = mAuth.getCurrentUser(); + + if (user == null) { + callbackNoUser(callback, false); + } else { + Log.d("USRC", user.getUid()); + userCallback(user, callback); + } + } + + // TODO: Check these things + @ReactMethod + public void googleLogin(String IdToken, final Callback callback) { + AuthCredential credential = GoogleAuthProvider.getCredential(IdToken, null); + mAuth.signInWithCredential(credential) + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + try { + if (task.isSuccessful()) { + userCallback(task.getResult().getUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + @ReactMethod + public void facebookLogin(String Token, final Callback callback) { + AuthCredential credential = FacebookAuthProvider.getCredential(Token); + mAuth.signInWithCredential(credential) + .addOnCompleteListener(new OnCompleteListener<AuthResult>() { + @Override + public void onComplete(@NonNull Task<AuthResult> task) { + try { + if (task.isSuccessful()) { + userCallback(task.getResult().getUser(), callback); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } + + // Internal helpers + private void userCallback(final FirebaseUser user, final Callback callback) { + if (user != null) { + user.getToken(true).addOnCompleteListener(new OnCompleteListener<GetTokenResult>() { + @Override + public void onComplete(@NonNull Task<GetTokenResult> task) { + try { + if (task.isSuccessful()) { + WritableMap userMap = getUserMap(user); + userMap.putString("token", task.getResult().getToken()); + callback.invoke(null, userMap); + } else { + userErrorCallback(task, callback); + } + } catch (Exception ex) { + userExceptionCallback(ex, callback); + } + } + }); + } else { + callbackNoUser(callback, true); + } + } + + private void userErrorCallback(Task task, final Callback onFail) { + WritableMap error = Arguments.createMap(); + error.putString("code", ((FirebaseAuthException) task.getException()).getErrorCode()); + error.putString("message", task.getException().getMessage()); + onFail.invoke(error); + } + + private void userExceptionCallback(Exception ex, final Callback onFail) { + WritableMap error = Arguments.createMap(); + error.putInt("code", ex.hashCode()); + error.putString("message", ex.getMessage()); + onFail.invoke(error); + } + + private WritableMap getUserMap(FirebaseUser user) { + WritableMap userMap = Arguments.createMap(); + if (user != null) { + final String email = user.getEmail(); + final String uid = user.getUid(); + final String provider = user.getProviderId(); + final String name = user.getDisplayName(); + final Boolean verified = user.isEmailVerified(); + final Uri photoUrl = user.getPhotoUrl(); + + userMap.putString("email", email); + userMap.putString("uid", uid); + userMap.putString("providerId", provider); + userMap.putBoolean("emailVerified", verified); + + if (name != null) { + userMap.putString("name", name); + } + + if (photoUrl != null) { + userMap.putString("photoURL", photoUrl.toString()); + } + } else { + userMap.putString("msg", "no user"); + } + + return userMap; + } +} diff --git a/android/src/main/java/io/fullstack/firestack/database/FirestackDatabase.java b/android/src/main/java/io/fullstack/firestack/database/FirestackDatabase.java new file mode 100644 index 0000000..971d2da --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/database/FirestackDatabase.java @@ -0,0 +1,343 @@ +package io.fullstack.firestack.database; + +import java.util.Map; +import android.net.Uri; +import android.util.Log; +import java.util.HashMap; + +import com.facebook.react.bridge.Callback; +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.WritableMap; +import com.facebook.react.bridge.ReadableMap; +import com.facebook.react.bridge.ReactMethod; +import com.facebook.react.bridge.ReactContext; +import com.facebook.react.bridge.ReadableArray; +import com.facebook.react.bridge.ReactApplicationContext; +import com.facebook.react.bridge.ReactContextBaseJavaModule; +import com.facebook.react.bridge.ReadableMapKeySetIterator; + +import com.google.firebase.database.OnDisconnect; +import com.google.firebase.database.DatabaseError; +import com.google.firebase.database.DatabaseReference; +import com.google.firebase.database.FirebaseDatabase; +import com.google.firebase.database.ServerValue; + + +import io.fullstack.firestack.Utils; + +public class FirestackDatabase extends ReactContextBaseJavaModule { + private static final String TAG = "FirestackDatabase"; + private HashMap<String, FirestackDatabaseReference> mDBListeners = new HashMap<String, FirestackDatabaseReference>(); + private FirebaseDatabase mFirebaseDatabase; + + public FirestackDatabase(ReactApplicationContext reactContext) { + super(reactContext); + mFirebaseDatabase = FirebaseDatabase.getInstance(); + } + + @Override + public String getName() { + return TAG; + } + + // Persistence + @ReactMethod + public void enablePersistence( + final Boolean enable, + final Callback callback) { + try { + mFirebaseDatabase.setPersistenceEnabled(enable); + } catch (Throwable t) { + Log.e(TAG, "FirebaseDatabase setPersistenceEnabled exception", t); + } + + WritableMap res = Arguments.createMap(); + res.putString("status", "success"); + callback.invoke(null, res); + } + + @ReactMethod + public void keepSynced( + final String path, + final Boolean enable, + final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + ref.keepSynced(enable); + + WritableMap res = Arguments.createMap(); + res.putString("status", "success"); + res.putString("path", path); + callback.invoke(null, res); + } + + // FirestackDatabase + @ReactMethod + public void set( + final String path, + final ReadableMap props, + final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(props); + + + DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("set", callback, error); + } + }; + + ref.setValue(m.get("value"), listener); + } + + @ReactMethod + public void update(final String path, + final ReadableMap props, + final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(props); + + DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("update", callback, error); + } + }; + + ref.updateChildren(m, listener); + } + + @ReactMethod + public void remove(final String path, + final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("remove", callback, error); + } + }; + + ref.removeValue(listener); + } + + @ReactMethod + public void push(final String path, + final ReadableMap props, + final Callback callback) { + + Log.d(TAG, "Called push with " + path); + DatabaseReference ref = mFirebaseDatabase.getReference(path); + DatabaseReference newRef = ref.push(); + + final Uri url = Uri.parse(newRef.toString()); + final String newPath = url.getPath(); + + ReadableMapKeySetIterator iterator = props.keySetIterator(); + if (iterator.hasNextKey()) { + Log.d(TAG, "Passed value to push"); + // lame way to check if the `props` are empty + Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(props); + + DatabaseReference.CompletionListener listener = new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + if (error != null) { + WritableMap err = Arguments.createMap(); + err.putInt("errorCode", error.getCode()); + err.putString("errorDetails", error.getDetails()); + err.putString("description", error.getMessage()); + callback.invoke(err); + } else { + WritableMap res = Arguments.createMap(); + res.putString("status", "success"); + res.putString("ref", newPath); + callback.invoke(null, res); + } + } + }; + + newRef.setValue(m.get("value"), listener); + } else { + Log.d(TAG, "No value passed to push: " + newPath); + WritableMap res = Arguments.createMap(); + res.putString("status", "success"); + res.putString("ref", newPath); + callback.invoke(null, res); + } + } + + @ReactMethod + public void on(final String path, + final String modifiersString, + final ReadableArray modifiersArray, + final String eventName, + final Callback callback) { + FirestackDatabaseReference ref = this.getDBHandle(path, modifiersArray, modifiersString); + + if (eventName.equals("value")) { + ref.addValueEventListener(); + } else { + ref.addChildEventListener(eventName); + } + + WritableMap resp = Arguments.createMap(); + resp.putString("status", "success"); + resp.putString("handle", path); + callback.invoke(null, resp); + } + + @ReactMethod + public void onOnce(final String path, + final String modifiersString, + final ReadableArray modifiersArray, + final String eventName, + final Callback callback) { + FirestackDatabaseReference ref = this.getDBHandle(path, modifiersArray, modifiersString); + ref.addOnceValueEventListener(callback); + } + + /** + * At the time of this writing, off() only gets called when there are no more subscribers to a given path. + * `mListeners` might therefore be out of sync (though javascript isnt listening for those eventTypes, so + * it doesn't really matter- just polluting the RN bridge a little more than necessary. + * off() should therefore clean *everything* up + */ + @ReactMethod + public void off( + final String path, + final String modifiersString, + final String eventName, + final Callback callback) { + + String key = this.getDBListenerKey(path, modifiersString); + FirestackDatabaseReference r = mDBListeners.get(key); + + if (r != null) { + if (eventName == null || "".equals(eventName)) { + r.cleanup(); + mDBListeners.remove(key); + } else { + r.removeEventListener(eventName); + if (!r.hasListeners()) { + mDBListeners.remove(key); + } ; + } + } + + Log.d(TAG, "Removed listener " + path + "/" + modifiersString); + WritableMap resp = Arguments.createMap(); + resp.putString("handle", path); + resp.putString("status", "success"); + resp.putString("modifiersString", modifiersString); + //TODO: Remaining listeners + callback.invoke(null, resp); + } + + // On Disconnect + @ReactMethod + public void onDisconnectSetObject(final String path, final ReadableMap props, final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(props); + + OnDisconnect od = ref.onDisconnect(); + od.setValue(m, new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("onDisconnectSetObject", callback, error); + } + }); + } + + @ReactMethod + public void onDisconnectSetString(final String path, final String value, final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + + OnDisconnect od = ref.onDisconnect(); + od.setValue(value, new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("onDisconnectSetString", callback, error); + } + }); + } + + @ReactMethod + public void onDisconnectRemove(final String path, final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + + OnDisconnect od = ref.onDisconnect(); + od.removeValue(new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("onDisconnectRemove", callback, error); + } + }); + } + @ReactMethod + public void onDisconnectCancel(final String path, final Callback callback) { + DatabaseReference ref = mFirebaseDatabase.getReference(path); + + OnDisconnect od = ref.onDisconnect(); + od.cancel(new DatabaseReference.CompletionListener() { + @Override + public void onComplete(DatabaseError error, DatabaseReference ref) { + handleCallback("onDisconnectCancel", callback, error); + } + }); + } + + @ReactMethod + public void goOnline() { + mFirebaseDatabase.goOnline(); + } + + @ReactMethod + public void goOffline() { + mFirebaseDatabase.goOffline(); + } + + private void handleCallback( + final String methodName, + final Callback callback, + final DatabaseError databaseError) { + if (databaseError != null) { + WritableMap err = Arguments.createMap(); + err.putInt("errorCode", databaseError.getCode()); + err.putString("errorDetails", databaseError.getDetails()); + err.putString("description", databaseError.getMessage()); + callback.invoke(err); + } else { + WritableMap res = Arguments.createMap(); + res.putString("status", "success"); + res.putString("method", methodName); + callback.invoke(null, res); + } + } + + private FirestackDatabaseReference getDBHandle(final String path, + final ReadableArray modifiersArray, + final String modifiersString) { + String key = this.getDBListenerKey(path, modifiersString); + FirestackDatabaseReference r = mDBListeners.get(key); + + if (r == null) { + ReactContext ctx = getReactApplicationContext(); + r = new FirestackDatabaseReference(ctx, mFirebaseDatabase, path, modifiersArray, modifiersString); + mDBListeners.put(key, r); + } + + return r; + } + + private String getDBListenerKey(String path, String modifiersString) { + return path + " | " + modifiersString; + } + + @Override + public Map<String, Object> getConstants() { + final Map<String, Object> constants = new HashMap<>(); + constants.put("serverValueTimestamp", ServerValue.TIMESTAMP); + return constants; + } +} diff --git a/android/src/main/java/io/fullstack/firestack/database/FirestackDatabaseReference.java b/android/src/main/java/io/fullstack/firestack/database/FirestackDatabaseReference.java new file mode 100644 index 0000000..47e6f11 --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/database/FirestackDatabaseReference.java @@ -0,0 +1,296 @@ +package io.fullstack.firestack.database; + +import java.util.HashSet; +import java.util.List; +import android.util.Log; +import java.util.ListIterator; +import java.util.Set; + +import com.facebook.react.bridge.Callback; +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.WritableMap; +import com.facebook.react.bridge.ReactContext; +import com.facebook.react.bridge.ReadableArray; + +import com.google.firebase.database.Query; +import com.google.firebase.database.DataSnapshot; +import com.google.firebase.database.DatabaseError; +import com.google.firebase.database.FirebaseDatabase; +import com.google.firebase.database.ChildEventListener; +import com.google.firebase.database.ValueEventListener; + +import io.fullstack.firestack.Utils; + +public class FirestackDatabaseReference { + private static final String TAG = "FirestackDBReference"; + + private Query mQuery; + private String mPath; + private String mModifiersString; + private ChildEventListener mEventListener; + private ValueEventListener mValueListener; + private ReactContext mReactContext; + private Set<String> childEventListeners = new HashSet<>(); + + public FirestackDatabaseReference(final ReactContext context, + final FirebaseDatabase firebaseDatabase, + final String path, + final ReadableArray modifiersArray, + final String modifiersString) { + mReactContext = context; + mPath = path; + mModifiersString = modifiersString; + mQuery = this.buildDatabaseQueryAtPathAndModifiers(firebaseDatabase, path, modifiersArray); + } + + public void addChildEventListener(final String eventName) { + if (mEventListener == null) { + mEventListener = new ChildEventListener() { + @Override + public void onChildAdded(DataSnapshot dataSnapshot, String previousChildName) { + handleDatabaseEvent("child_added", dataSnapshot); + } + + @Override + public void onChildChanged(DataSnapshot dataSnapshot, String previousChildName) { + handleDatabaseEvent("child_changed", dataSnapshot); + } + + @Override + public void onChildRemoved(DataSnapshot dataSnapshot) { + handleDatabaseEvent("child_removed", dataSnapshot); + } + + @Override + public void onChildMoved(DataSnapshot dataSnapshot, String previousChildName) { + handleDatabaseEvent("child_moved", dataSnapshot); + } + + @Override + public void onCancelled(DatabaseError error) { + handleDatabaseError(error); + } + }; + mQuery.addChildEventListener(mEventListener); + Log.d(TAG, "Added ChildEventListener for path: " + mPath + " with modifiers: "+ mModifiersString); + } else { + Log.w(TAG, "Trying to add duplicate ChildEventListener for path: " + mPath + " with modifiers: "+ mModifiersString); + } + //Keep track of the events that the JS is interested in knowing about + childEventListeners.add(eventName); + } + + public void addValueEventListener() { + if (mValueListener == null) { + mValueListener = new ValueEventListener() { + @Override + public void onDataChange(DataSnapshot dataSnapshot) { + handleDatabaseEvent("value", dataSnapshot); + } + + @Override + public void onCancelled(DatabaseError error) { + handleDatabaseError(error); + } + }; + mQuery.addValueEventListener(mValueListener); + Log.d(TAG, "Added ValueEventListener for path: " + mPath + " with modifiers: "+ mModifiersString); + //this.setListeningTo(mPath, modifiersString, "value"); + } else { + Log.w(TAG, "Trying to add duplicate ValueEventListener for path: " + mPath + " with modifiers: "+ mModifiersString); + } + } + + public void addOnceValueEventListener(final Callback callback) { + final ValueEventListener onceValueEventListener = new ValueEventListener() { + @Override + public void onDataChange(DataSnapshot dataSnapshot) { + WritableMap data = Utils.dataSnapshotToMap("value", mPath, mModifiersString, dataSnapshot); + callback.invoke(null, data); + } + + @Override + public void onCancelled(DatabaseError error) { + WritableMap err = Arguments.createMap(); + err.putInt("errorCode", error.getCode()); + err.putString("errorDetails", error.getDetails()); + err.putString("description", error.getMessage()); + callback.invoke(err); + } + }; + mQuery.addListenerForSingleValueEvent(onceValueEventListener); + Log.d(TAG, "Added OnceValueEventListener for path: " + mPath + " with modifiers " + mModifiersString); + } + + public void removeEventListener(String eventName) { + if ("value".equals(eventName)) { + this.removeValueEventListener(); + } else { + childEventListeners.remove(eventName); + if (childEventListeners.isEmpty()) { + this.removeChildEventListener(); + } + } + } + + public boolean hasListeners() { + return mEventListener != null || mValueListener != null; + } + + public void cleanup() { + Log.d(TAG, "cleaning up database reference " + this); + childEventListeners.clear(); + this.removeChildEventListener(); + this.removeValueEventListener(); + } + + private void removeChildEventListener() { + if (mEventListener != null) { + mQuery.removeEventListener(mEventListener); + mEventListener = null; + } + } + + private void removeValueEventListener() { + if (mValueListener != null) { + mQuery.removeEventListener(mValueListener); + mValueListener = null; + } + } + + private void handleDatabaseEvent(final String name, final DataSnapshot dataSnapshot) { + WritableMap data = Utils.dataSnapshotToMap(name, mPath, mModifiersString, dataSnapshot); + WritableMap evt = Arguments.createMap(); + evt.putString("eventName", name); + evt.putMap("body", data); + + Utils.sendEvent(mReactContext, "database_event", evt); + } + + private void handleDatabaseError(final DatabaseError error) { + WritableMap err = Arguments.createMap(); + err.putString("eventName", "database_error"); + err.putString("path", mPath); + err.putString("modifiersString", mModifiersString); + err.putInt("errorCode", error.getCode()); + err.putString("errorDetails", error.getDetails()); + err.putString("msg", error.getMessage()); + + WritableMap evt = Arguments.createMap(); + evt.putString("eventName", "database_error"); + evt.putMap("body", err); + + Utils.sendEvent(mReactContext, "database_error", evt); + } + + private Query buildDatabaseQueryAtPathAndModifiers(final FirebaseDatabase firebaseDatabase, + final String path, + final ReadableArray modifiers) { + Query query = firebaseDatabase.getReference(path); + List<Object> strModifiers = Utils.recursivelyDeconstructReadableArray(modifiers); + ListIterator<Object> it = strModifiers.listIterator(); + + while(it.hasNext()) { + String str = (String) it.next(); + + String[] strArr = str.split(":"); + String methStr = strArr[0]; + + if (methStr.equalsIgnoreCase("orderByKey")) { + query = query.orderByKey(); + } else if (methStr.equalsIgnoreCase("orderByValue")) { + query = query.orderByValue(); + } else if (methStr.equalsIgnoreCase("orderByPriority")) { + query = query.orderByPriority(); + } else if (methStr.contains("orderByChild")) { + String key = strArr[1]; + Log.d(TAG, "orderByChild: " + key); + query = query.orderByChild(key); + } else if (methStr.contains("limitToLast")) { + String key = strArr[1]; + int limit = Integer.parseInt(key); + Log.d(TAG, "limitToLast: " + limit); + query = query.limitToLast(limit); + } else if (methStr.contains("limitToFirst")) { + String key = strArr[1]; + int limit = Integer.parseInt(key); + Log.d(TAG, "limitToFirst: " + limit); + query = query.limitToFirst(limit); + } else if (methStr.contains("equalTo")) { + String value = strArr[1]; + String type = strArr[2]; + if ("number".equals(type)) { + double doubleValue = Double.parseDouble(value); + if (strArr.length > 3) { + query = query.equalTo(doubleValue, strArr[3]); + } else { + query = query.equalTo(doubleValue); + } + } else if ("boolean".equals(type)) { + boolean booleanValue = Boolean.parseBoolean(value); + if (strArr.length > 3) { + query = query.equalTo(booleanValue, strArr[3] ); + } else { + query = query.equalTo(booleanValue); + } + } else { + if (strArr.length > 3) { + query = query.equalTo(value, strArr[3]); + } else { + query = query.equalTo(value); + } + } + } else if (methStr.contains("endAt")) { + String value = strArr[1]; + String type = strArr[2]; + if ("number".equals(type)) { + double doubleValue = Double.parseDouble(value); + if (strArr.length > 3) { + query = query.endAt(doubleValue, strArr[3]); + } else { + query = query.endAt(doubleValue); + } + } else if ("boolean".equals(type)) { + boolean booleanValue = Boolean.parseBoolean(value); + if (strArr.length > 3) { + query = query.endAt(booleanValue, strArr[3] ); + } else { + query = query.endAt(booleanValue); + } + } else { + if (strArr.length > 3) { + query = query.endAt(value, strArr[3]); + } else { + query = query.endAt(value); + } + } + } else if (methStr.contains("startAt")) { + String value = strArr[1]; + String type = strArr[2]; + if ("number".equals(type)) { + double doubleValue = Double.parseDouble(value); + if (strArr.length > 3) { + query = query.startAt(doubleValue, strArr[3]); + } else { + query = query.startAt(doubleValue); + } + } else if ("boolean".equals(type)) { + boolean booleanValue = Boolean.parseBoolean(value); + if (strArr.length > 3) { + query = query.startAt(booleanValue, strArr[3] ); + } else { + query = query.startAt(booleanValue); + } + } else { + if (strArr.length > 3) { + query = query.startAt(value, strArr[3]); + } else { + query = query.startAt(value); + } + } + } + } + + return query; + } +} diff --git a/android/src/main/java/io/fullstack/firestack/messaging/FirestackMessaging.java b/android/src/main/java/io/fullstack/firestack/messaging/FirestackMessaging.java new file mode 100644 index 0000000..4114e76 --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/messaging/FirestackMessaging.java @@ -0,0 +1,229 @@ +package io.fullstack.firestack.messaging; + +import java.util.Map; + +import android.content.Context; +import android.content.IntentFilter; +import android.content.Intent; +import android.content.BroadcastReceiver; +import android.util.Log; + +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.Callback; +import com.facebook.react.bridge.ReactApplicationContext; +import com.facebook.react.bridge.ReactContext; +import com.facebook.react.bridge.ReactContextBaseJavaModule; +import com.facebook.react.bridge.ReactMethod; +import com.facebook.react.bridge.ReadableMap; +import com.facebook.react.bridge.ReadableMapKeySetIterator; +import com.facebook.react.bridge.ReadableType; +import com.facebook.react.bridge.WritableMap; + +import com.google.firebase.iid.FirebaseInstanceId; +import com.google.firebase.messaging.FirebaseMessaging; +import com.google.firebase.messaging.RemoteMessage; + +import io.fullstack.firestack.Utils; + +public class FirestackMessaging extends ReactContextBaseJavaModule { + + private static final String TAG = "FirestackMessaging"; + private static final String EVENT_NAME_TOKEN = "FirestackRefreshToken"; + private static final String EVENT_NAME_NOTIFICATION = "FirestackReceiveNotification"; + private static final String EVENT_NAME_SEND = "FirestackUpstreamSend"; + + public static final String INTENT_NAME_TOKEN = "io.fullstack.firestack.refreshToken"; + public static final String INTENT_NAME_NOTIFICATION = "io.fullstack.firestack.ReceiveNotification"; + public static final String INTENT_NAME_SEND = "io.fullstack.firestack.Upstream"; + + private IntentFilter mRefreshTokenIntentFilter; + private IntentFilter mReceiveNotificationIntentFilter; + private IntentFilter mReceiveSendIntentFilter; + private BroadcastReceiver mBroadcastReceiver; + + public FirestackMessaging(ReactApplicationContext reactContext) { + super(reactContext); + mRefreshTokenIntentFilter = new IntentFilter(INTENT_NAME_TOKEN); + mReceiveNotificationIntentFilter = new IntentFilter(INTENT_NAME_NOTIFICATION); + mReceiveSendIntentFilter = new IntentFilter(INTENT_NAME_SEND); + initRefreshTokenHandler(); + initMessageHandler(); + initSendHandler(); + Log.d(TAG, "New instance"); + } + + @Override + public String getName() { + return TAG; + } + + private void initMessageHandler() { + Log.d(TAG, "Firestack initMessageHandler called"); + + if (mBroadcastReceiver == null) { + mBroadcastReceiver = new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + RemoteMessage remoteMessage = intent.getParcelableExtra("data"); + Log.d(TAG, "Firebase onReceive: " + remoteMessage); + WritableMap params = Arguments.createMap(); + + params.putNull("data"); + params.putNull("notification"); + params.putString("id", remoteMessage.getMessageId()); + params.putString("messageId", remoteMessage.getMessageId()); + + + if (remoteMessage.getData().size() != 0) { + WritableMap dataMap = Arguments.createMap(); + Map<String, String> data = remoteMessage.getData(); + + for (String key : data.keySet()) { + dataMap.putString(key, data.get(key)); + } + + params.putMap("data", dataMap); + } + + + if (remoteMessage.getNotification() != null) { + WritableMap notificationMap = Arguments.createMap(); + RemoteMessage.Notification notification = remoteMessage.getNotification(); + notificationMap.putString("title", notification.getTitle()); + notificationMap.putString("body", notification.getBody()); + notificationMap.putString("icon", notification.getIcon()); + notificationMap.putString("sound", notification.getSound()); + notificationMap.putString("tag", notification.getTag()); + params.putMap("notification", notificationMap); + } + + ReactContext ctx = getReactApplicationContext(); + Utils.sendEvent(ctx, EVENT_NAME_NOTIFICATION, params); + } + }; + + } + getReactApplicationContext().registerReceiver(mBroadcastReceiver, mReceiveNotificationIntentFilter); + } + + /** + * + */ + private void initRefreshTokenHandler() { + getReactApplicationContext().registerReceiver(new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + WritableMap params = Arguments.createMap(); + params.putString("token", intent.getStringExtra("token")); + ReactContext ctx = getReactApplicationContext(); + Log.d(TAG, "initRefreshTokenHandler received event " + EVENT_NAME_TOKEN); + Utils.sendEvent(ctx, EVENT_NAME_TOKEN, params); + } + + ; + }, mRefreshTokenIntentFilter); + } + + @ReactMethod + public void subscribeToTopic(String topic, final Callback callback) { + try { + FirebaseMessaging.getInstance().subscribeToTopic(topic); + callback.invoke(null, topic); + } catch (Exception e) { + e.printStackTrace(); + Log.d(TAG, "Firebase token: " + e); + WritableMap error = Arguments.createMap(); + error.putString("message", e.getMessage()); + callback.invoke(error); + + } + } + + @ReactMethod + public void getToken(final Callback callback) { + + try { + String token = FirebaseInstanceId.getInstance().getToken(); + Log.d(TAG, "Firebase token: " + token); + callback.invoke(null, token); + } catch (Exception e) { + WritableMap error = Arguments.createMap(); + error.putString("message", e.getMessage()); + callback.invoke(error); + } + } + + @ReactMethod + public void unsubscribeFromTopic(String topic, final Callback callback) { + try { + FirebaseMessaging.getInstance().unsubscribeFromTopic(topic); + callback.invoke(null, topic); + } catch (Exception e) { + WritableMap error = Arguments.createMap(); + error.putString("message", e.getMessage()); + callback.invoke(error); + } + } + + // String senderId, String messageId, String messageType, + @ReactMethod + public void send(ReadableMap params, final Callback callback) { + ReadableMap data = params.getMap("data"); + FirebaseMessaging fm = FirebaseMessaging.getInstance(); + RemoteMessage.Builder remoteMessage = new RemoteMessage.Builder(params.getString("sender")); + + remoteMessage.setMessageId(params.getString("id")); + remoteMessage.setMessageType(params.getString("type")); + + if (params.hasKey("ttl")) { + remoteMessage.setTtl(params.getInt("ttl")); + } + + if (params.hasKey("collapseKey")) { + remoteMessage.setCollapseKey(params.getString("collapseKey")); + } + + ReadableMapKeySetIterator iterator = data.keySetIterator(); + + while (iterator.hasNextKey()) { + String key = iterator.nextKey(); + ReadableType type = data.getType(key); + if (type == ReadableType.String) { + remoteMessage.addData(key, data.getString(key)); + } + } + + try { + fm.send(remoteMessage.build()); + WritableMap res = Arguments.createMap(); + res.putString("status", "success"); + Log.d(TAG, "send: Message sent"); + callback.invoke(null, res); + } catch (Exception e) { + Log.e(TAG, "send: error sending message", e); + WritableMap error = Arguments.createMap(); + error.putString("code", e.toString()); + error.putString("message", e.toString()); + callback.invoke(error); + } + } + + private void initSendHandler() { + getReactApplicationContext().registerReceiver(new BroadcastReceiver() { + @Override + public void onReceive(Context context, Intent intent) { + WritableMap params = Arguments.createMap(); + if (intent.getBooleanExtra("hasError", false)) { + WritableMap error = Arguments.createMap(); + error.putInt("code", intent.getIntExtra("errCode", 0)); + error.putString("message", intent.getStringExtra("errorMessage")); + params.putMap("err", error); + } else { + params.putNull("err"); + } + ReactContext ctx = getReactApplicationContext(); + Utils.sendEvent(ctx, EVENT_NAME_SEND, params); + } + }, mReceiveSendIntentFilter); + } +} diff --git a/android/src/main/java/io/fullstack/firestack/storage/FirestackStorage.java b/android/src/main/java/io/fullstack/firestack/storage/FirestackStorage.java new file mode 100644 index 0000000..cae695f --- /dev/null +++ b/android/src/main/java/io/fullstack/firestack/storage/FirestackStorage.java @@ -0,0 +1,484 @@ +package io.fullstack.firestack.storage; + +import android.util.Log; +import android.os.Environment; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.util.Map; +import java.util.HashMap; + +import android.net.Uri; +import android.database.Cursor; +import android.provider.MediaStore; +import android.support.annotation.NonNull; + +import com.facebook.react.bridge.Callback; +import com.facebook.react.bridge.Arguments; +import com.facebook.react.bridge.WritableArray; +import com.facebook.react.bridge.WritableMap; +import com.facebook.react.bridge.ReactMethod; +import com.facebook.react.bridge.ReadableMap; +import com.facebook.react.bridge.ReactContext; +import com.facebook.react.bridge.ReactApplicationContext; +import com.facebook.react.bridge.ReactContextBaseJavaModule; + +import com.google.android.gms.tasks.Task; +import com.google.android.gms.tasks.OnFailureListener; +import com.google.android.gms.tasks.OnSuccessListener; + +import com.google.firebase.database.DataSnapshot; +import com.google.firebase.database.DatabaseError; +import com.google.firebase.storage.StorageException; +import com.google.firebase.storage.StorageTask; +import com.google.firebase.storage.StreamDownloadTask; +import com.google.firebase.storage.UploadTask; +import com.google.firebase.storage.FirebaseStorage; +import com.google.firebase.storage.StorageMetadata; +import com.google.firebase.storage.StorageReference; +import com.google.firebase.storage.OnPausedListener; +import com.google.firebase.storage.OnProgressListener; + +import io.fullstack.firestack.Utils; + + +@SuppressWarnings("WeakerAccess") +public class FirestackStorage extends ReactContextBaseJavaModule { + + private static final String TAG = "FirestackStorage"; + private static final String DocumentDirectoryPath = "DOCUMENT_DIRECTORY_PATH"; + private static final String ExternalDirectoryPath = "EXTERNAL_DIRECTORY_PATH"; + private static final String ExternalStorageDirectoryPath = "EXTERNAL_STORAGE_DIRECTORY_PATH"; + private static final String PicturesDirectoryPath = "PICTURES_DIRECTORY_PATH"; + private static final String TemporaryDirectoryPath = "TEMPORARY_DIRECTORY_PATH"; + private static final String CachesDirectoryPath = "CACHES_DIRECTORY_PATH"; + private static final String DocumentDirectory = "DOCUMENT_DIRECTORY_PATH"; + + private static final String FileTypeRegular = "FILETYPE_REGULAR"; + private static final String FileTypeDirectory = "FILETYPE_DIRECTORY"; + + private static final String STORAGE_EVENT = "storage_event"; + private static final String STORAGE_ERROR = "storage_error"; + private static final String STORAGE_STATE_CHANGED = "state_changed"; + private static final String STORAGE_UPLOAD_SUCCESS = "upload_success"; + private static final String STORAGE_UPLOAD_FAILURE = "upload_failure"; + private static final String STORAGE_DOWNLOAD_SUCCESS = "download_success"; + private static final String STORAGE_DOWNLOAD_FAILURE = "download_failure"; + + private ReactContext mReactContext; + + public FirestackStorage(ReactApplicationContext reactContext) { + super(reactContext); + + Log.d(TAG, "New instance"); + } + + @Override + public String getName() { + return TAG; + } + + + public boolean isExternalStorageWritable() { + String state = Environment.getExternalStorageState(); + return Environment.MEDIA_MOUNTED.equals(state); + } + + @ReactMethod + public void delete(final String path, + final Callback callback) { + StorageReference reference = this.getReference(path); + reference.delete().addOnSuccessListener(new OnSuccessListener<Void>() { + @Override + public void onSuccess(Void aVoid) { + WritableMap data = Arguments.createMap(); + data.putString("success", "success"); + data.putString("path", path); + callback.invoke(null, data); + } + }).addOnFailureListener(new OnFailureListener() { + @Override + public void onFailure(Exception exception) { + callback.invoke(makeErrorPayload(1, exception)); + } + }); + } + + @ReactMethod + public void getDownloadURL(final String path, + final Callback callback) { + Log.d(TAG, "Download url for remote path: " + path); + final StorageReference reference = this.getReference(path); + + Task<Uri> downloadTask = reference.getDownloadUrl(); + downloadTask + .addOnSuccessListener(new OnSuccessListener<Uri>() { + @Override + public void onSuccess(Uri uri) { + callback.invoke(null, uri.toString()); + } + }) + .addOnFailureListener(new OnFailureListener() { + @Override + public void onFailure(@NonNull Exception exception) { + callback.invoke(makeErrorPayload(1, exception)); + } + }); + } + + @ReactMethod + public void getMetadata(final String path, + final Callback callback) { + StorageReference reference = this.getReference(path); + reference.getMetadata().addOnSuccessListener(new OnSuccessListener<StorageMetadata>() { + @Override + public void onSuccess(StorageMetadata storageMetadata) { + WritableMap data = getMetadataAsMap(storageMetadata); + callback.invoke(null, data); + } + }).addOnFailureListener(new OnFailureListener() { + @Override + public void onFailure(Exception exception) { + callback.invoke(makeErrorPayload(1, exception)); + } + }); + } + + @ReactMethod + public void updateMetadata(final String path, + final ReadableMap metadata, + final Callback callback) { + StorageReference reference = this.getReference(path); + StorageMetadata md = buildMetadataFromMap(metadata); + reference.updateMetadata(md).addOnSuccessListener(new OnSuccessListener<StorageMetadata>() { + @Override + public void onSuccess(StorageMetadata storageMetadata) { + WritableMap data = getMetadataAsMap(storageMetadata); + callback.invoke(null, data); + } + }).addOnFailureListener(new OnFailureListener() { + @Override + public void onFailure(Exception exception) { + callback.invoke(makeErrorPayload(1, exception)); + } + }); + } + + @ReactMethod + public void downloadFile(final String path, + final String localPath, + final Callback callback) { + if (!isExternalStorageWritable()) { + Log.w(TAG, "downloadFile failed: external storage not writable"); + WritableMap error = Arguments.createMap(); + final int errorCode = 1; + error.putDouble("code", errorCode); + error.putString("description", "downloadFile failed: external storage not writable"); + callback.invoke(error); + return; + } + Log.d(TAG, "downloadFile from remote path: " + path); + + StorageReference reference = this.getReference(path); + + reference.getStream(new StreamDownloadTask.StreamProcessor() { + @Override + public void doInBackground(StreamDownloadTask.TaskSnapshot taskSnapshot, InputStream inputStream) throws IOException { + int indexOfLastSlash = localPath.lastIndexOf("/"); + String pathMinusFileName = indexOfLastSlash>0 ? localPath.substring(0, indexOfLastSlash) + "/" : "/"; + String filename = indexOfLastSlash>0 ? localPath.substring(indexOfLastSlash+1) : localPath; + File fileWithJustPath = new File(pathMinusFileName); + fileWithJustPath.mkdirs(); + File fileWithFullPath = new File(pathMinusFileName, filename); + FileOutputStream output = new FileOutputStream(fileWithFullPath); + int bufferSize = 1024; + byte[] buffer = new byte[bufferSize]; + int len = 0; + while ((len = inputStream.read(buffer)) != -1) { + output.write(buffer, 0, len); + } + output.close(); + } + }).addOnProgressListener(new OnProgressListener<StreamDownloadTask.TaskSnapshot>() { + @Override + public void onProgress(StreamDownloadTask.TaskSnapshot taskSnapshot) { + Log.d(TAG, "Got download progress " + taskSnapshot); + WritableMap event = getDownloadTaskAsMap(taskSnapshot); + handleStorageEvent(STORAGE_STATE_CHANGED, path, event); + } + }).addOnPausedListener(new OnPausedListener<StreamDownloadTask.TaskSnapshot>() { + @Override + public void onPaused(StreamDownloadTask.TaskSnapshot taskSnapshot) { + Log.d(TAG, "Download is paused " + taskSnapshot); + WritableMap event = getDownloadTaskAsMap(taskSnapshot); + handleStorageEvent(STORAGE_STATE_CHANGED, path, event); + } + }).addOnSuccessListener(new OnSuccessListener<StreamDownloadTask.TaskSnapshot>() { + @Override + public void onSuccess(StreamDownloadTask.TaskSnapshot taskSnapshot) { + Log.d(TAG, "Successfully downloaded file " + taskSnapshot); + WritableMap resp = getDownloadTaskAsMap(taskSnapshot); + handleStorageEvent(STORAGE_DOWNLOAD_SUCCESS, path, resp); + //TODO: A little hacky, but otherwise throws a not consumed exception + resp = getDownloadTaskAsMap(taskSnapshot); + callback.invoke(null, resp); + } + }).addOnFailureListener(new OnFailureListener() { + @Override + public void onFailure(@NonNull Exception exception) { + Log.e(TAG, "Failed to download file " + exception.getMessage()); + //TODO: JS Error event + callback.invoke(makeErrorPayload(1, exception)); + } + }); + } + + @ReactMethod + public void putFile(final String path, final String localPath, final ReadableMap metadata, final Callback callback) { + StorageReference reference = this.getReference(path); + + Log.i(TAG, "Upload file: " + localPath + " to " + path); + + try { + Uri file; + if (localPath.startsWith("content://")) { + String realPath = getRealPathFromURI(localPath); + file = Uri.fromFile(new File(realPath)); + } else { + file = Uri.fromFile(new File(localPath)); + } + + StorageMetadata md = buildMetadataFromMap(metadata); + UploadTask uploadTask = reference.putFile(file, md); + + // register observers to listen for when the download is done or if it fails + uploadTask + .addOnFailureListener(new OnFailureListener() { + @Override + public void onFailure(@NonNull Exception exception) { + // handle unsuccessful uploads + Log.e(TAG, "Failed to upload file " + exception.getMessage()); + //TODO: JS Error event + callback.invoke(makeErrorPayload(1, exception)); + } + }) + .addOnSuccessListener(new OnSuccessListener<UploadTask.TaskSnapshot>() { + @Override + public void onSuccess(UploadTask.TaskSnapshot taskSnapshot) { + Log.d(TAG, "Successfully uploaded file " + taskSnapshot); + WritableMap resp = getUploadTaskAsMap(taskSnapshot); + handleStorageEvent(STORAGE_UPLOAD_SUCCESS, path, resp); + //TODO: A little hacky, but otherwise throws a not consumed exception + resp = getUploadTaskAsMap(taskSnapshot); + callback.invoke(null, resp); + } + }) + .addOnProgressListener(new OnProgressListener<UploadTask.TaskSnapshot>() { + @Override + public void onProgress(UploadTask.TaskSnapshot taskSnapshot) { + Log.d(TAG, "Got upload progress " + taskSnapshot); + WritableMap event = getUploadTaskAsMap(taskSnapshot); + handleStorageEvent(STORAGE_STATE_CHANGED, path, event); + } + }) + .addOnPausedListener(new OnPausedListener<UploadTask.TaskSnapshot>() { + @Override + public void onPaused(UploadTask.TaskSnapshot taskSnapshot) { + Log.d(TAG, "Upload is paused " + taskSnapshot); + WritableMap event = getUploadTaskAsMap(taskSnapshot); + handleStorageEvent(STORAGE_STATE_CHANGED, path, event); + } + }); + } catch (Exception ex) { + final int errorCode = 2; + callback.invoke(makeErrorPayload(errorCode, ex)); + } + } + + //Firebase.Storage methods + @ReactMethod + public void setMaxDownloadRetryTime(final double milliseconds) { + FirebaseStorage.getInstance().setMaxDownloadRetryTimeMillis((long)milliseconds); + } + + @ReactMethod + public void setMaxOperationRetryTime(final double milliseconds) { + FirebaseStorage.getInstance().setMaxOperationRetryTimeMillis((long)milliseconds); + } + + @ReactMethod + public void setMaxUploadRetryTime(final double milliseconds) { + FirebaseStorage.getInstance().setMaxUploadRetryTimeMillis((long)milliseconds); + } + + private StorageReference getReference(String path) { + if (path.startsWith("url::")) { + String url = path.substring(5); + return FirebaseStorage.getInstance().getReferenceFromUrl(url); + } else { + return FirebaseStorage.getInstance().getReference(path); + } + } + + private StorageMetadata buildMetadataFromMap(ReadableMap metadata) { + StorageMetadata.Builder metadataBuilder = new StorageMetadata.Builder(); + Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(metadata); + + for (Map.Entry<String, Object> entry : m.entrySet()) { + metadataBuilder.setCustomMetadata(entry.getKey(), entry.getValue().toString()); + } + + return metadataBuilder.build(); + } + + private WritableMap getMetadataAsMap(StorageMetadata storageMetadata) { + WritableMap metadata = Arguments.createMap(); + metadata.putString("bucket", storageMetadata.getBucket()); + metadata.putString("generation", storageMetadata.getGeneration()); + metadata.putString("metageneration", storageMetadata.getMetadataGeneration()); + metadata.putString("fullPath", storageMetadata.getPath()); + metadata.putString("name", storageMetadata.getName()); + metadata.putDouble("size", storageMetadata.getSizeBytes()); + metadata.putDouble("timeCreated", storageMetadata.getCreationTimeMillis()); + metadata.putDouble("updated", storageMetadata.getUpdatedTimeMillis()); + metadata.putString("md5hash", storageMetadata.getMd5Hash()); + metadata.putString("cacheControl", storageMetadata.getCacheControl()); + metadata.putString("contentDisposition", storageMetadata.getContentDisposition()); + metadata.putString("contentEncoding", storageMetadata.getContentEncoding()); + metadata.putString("contentLanguage", storageMetadata.getContentLanguage()); + metadata.putString("contentType", storageMetadata.getContentType()); + + WritableArray downloadURLs = Arguments.createArray(); + for (Uri uri : storageMetadata.getDownloadUrls()) { + downloadURLs.pushString(uri.getPath()); + } + metadata.putArray("downloadURLs", downloadURLs); + + WritableMap customMetadata = Arguments.createMap(); + for (String key : storageMetadata.getCustomMetadataKeys()) { + customMetadata.putString(key, storageMetadata.getCustomMetadata(key)); + } + metadata.putMap("customMetadata", customMetadata); + + return metadata; + } + + private String getRealPathFromURI(final String uri) { + Cursor cursor = null; + try { + String[] proj = {MediaStore.Images.Media.DATA}; + cursor = getReactApplicationContext().getContentResolver().query(Uri.parse(uri), proj, null, null, null); + int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA); + cursor.moveToFirst(); + return cursor.getString(column_index); + } finally { + if (cursor != null) { + cursor.close(); + } + } + } + + private WritableMap getDownloadTaskAsMap(final StreamDownloadTask.TaskSnapshot taskSnapshot) { + WritableMap resp = Arguments.createMap(); + resp.putDouble("bytesTransferred", taskSnapshot.getBytesTransferred()); + resp.putString("ref", taskSnapshot.getStorage().getPath()); + resp.putString("state", this.getTaskStatus(taskSnapshot.getTask())); + resp.putDouble("totalBytes", taskSnapshot.getTotalByteCount()); + + return resp; + } + + private WritableMap getUploadTaskAsMap(final UploadTask.TaskSnapshot taskSnapshot) { + StorageMetadata d = taskSnapshot.getMetadata(); + + WritableMap resp = Arguments.createMap(); + resp.putDouble("bytesTransferred", taskSnapshot.getBytesTransferred()); + resp.putString("downloadUrl", taskSnapshot.getDownloadUrl() != null ? taskSnapshot.getDownloadUrl().toString() : null); + resp.putString("ref", taskSnapshot.getStorage().getPath()); + resp.putString("state", this.getTaskStatus(taskSnapshot.getTask())); + resp.putDouble("totalBytes", taskSnapshot.getTotalByteCount()); + + if (taskSnapshot.getMetadata() != null) { + WritableMap metadata = getMetadataAsMap(taskSnapshot.getMetadata()); + resp.putMap("metadata", metadata); + } + + return resp; + } + + private String getTaskStatus(StorageTask<?> task) { + if (task.isInProgress()) { + return "RUNNING"; + } else if (task.isPaused()) { + return "PAUSED"; + } else if (task.isSuccessful() || task.isComplete()) { + return "SUCCESS"; + } else if (task.isCanceled()) { + return "CANCELLED"; + } else if (task.getException() != null) { + return "ERROR"; + } else { + return "UNKNOWN"; + } + } + + private void handleStorageEvent(final String name, final String path, WritableMap body) { + WritableMap evt = Arguments.createMap(); + evt.putString("eventName", name); + evt.putString("path", path); + evt.putMap("body", body); + + Utils.sendEvent(this.getReactApplicationContext(), STORAGE_EVENT, evt); + } + + private void handleStorageError(final String path, final StorageException error) { + WritableMap body = Arguments.createMap(); + body.putString("path", path); + body.putString("message", error.getMessage()); + + WritableMap evt = Arguments.createMap(); + evt.putString("eventName", STORAGE_ERROR); + evt.putMap("body", body); + + Utils.sendEvent(this.getReactApplicationContext(), STORAGE_ERROR, evt); + } + + private WritableMap makeErrorPayload(double code, Exception ex) { + WritableMap error = Arguments.createMap(); + error.putDouble("code", code); + error.putString("message", ex.getMessage()); + return error; + } + + + @Override + public Map<String, Object> getConstants() { + final Map<String, Object> constants = new HashMap<>(); + + constants.put(DocumentDirectory, 0); + constants.put(DocumentDirectoryPath, this.getReactApplicationContext().getFilesDir().getAbsolutePath()); + constants.put(TemporaryDirectoryPath, null); + constants.put(PicturesDirectoryPath, Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getAbsolutePath()); + constants.put(CachesDirectoryPath, this.getReactApplicationContext().getCacheDir().getAbsolutePath()); + constants.put(FileTypeRegular, 0); + constants.put(FileTypeDirectory, 1); + + File externalStorageDirectory = Environment.getExternalStorageDirectory(); + if (externalStorageDirectory != null) { + constants.put(ExternalStorageDirectoryPath, externalStorageDirectory.getAbsolutePath()); + } else { + constants.put(ExternalStorageDirectoryPath, null); + } + + File externalDirectory = this.getReactApplicationContext().getExternalFilesDir(null); + if (externalDirectory != null) { + constants.put(ExternalDirectoryPath, externalDirectory.getAbsolutePath()); + } else { + constants.put(ExternalDirectoryPath, null); + } + + return constants; + } +} diff --git a/docs/api/analytics.md b/docs/api/analytics.md new file mode 100644 index 0000000..daefe5c --- /dev/null +++ b/docs/api/analytics.md @@ -0,0 +1,80 @@ +# Analytics + +Integrating Firebase analytics is super simple using Firestack. A number of methods are provided to help tailor analytics specifically for your +own app. The Firebase SDK includes a number of pre-set events which are automatically handled, and cannot be used with custom events: + +``` + 'app_clear_data', + 'app_uninstall', + 'app_update', + 'error', + 'first_open', + 'in_app_purchase', + 'notification_dismiss', + 'notification_foreground', + 'notification_open', + 'notification_receive', + 'os_update', + 'session_start', + 'user_engagement', +``` + +#### `logEvent(event: string, params?: Object): void` + +Log a custom event with optional params. + +```javascript +firestack.analytics().logEvent('clicked_advert', { id: 1337 }); +``` + +#### `setAnalyticsCollectionEnabled(enabled: boolean): void` + +Sets whether analytics collection is enabled for this app on this device. + +```javascript +firestack.analytics().setAnalyticsCollectionEnabled(false); +``` + +#### `setCurrentScreen(screenName: string, screenClassOverride?: string): void` + +Sets the current screen name, which specifies the current visual context in your app. + +> Whilst `screenClassOverride` is optional, it is recommended it is always sent as your current class name, for example on Android it will always show as 'MainActivity' if not specified. + +```javascript +firestack.analytics().setCurrentScreen('user_profile'); +``` + +#### `setMinimumSessionDuration(miliseconds: number): void` + +Sets the minimum engagement time required before starting a session. The default value is 10000 (10 seconds). + +```javascript +firestack.analytics().setMinimumSessionDuration(15000); +``` + +#### `setSessionTimeoutDuration(miliseconds: number): void` + +Sets the duration of inactivity that terminates the current session. The default value is 1800000 (30 minutes). + +```javascript +firestack.analytics().setSessionTimeoutDuration(900000); +``` + +#### `setUserId(id: string): void` + +Gives a user a uniqiue identificaition. + +```javascript +const id = firestack.auth().currentUser.uid; + +firestack.analytics().setUserId(id); +``` + +#### `setUserProperty(name: string, value: string): void` + +Sets a key/value pair of data on the current user. + +```javascript +firestack.analytics().setUserProperty('nickname', 'foobar'); +``` diff --git a/docs/api/authentication.md b/docs/api/authentication.md new file mode 100644 index 0000000..84cb5bb --- /dev/null +++ b/docs/api/authentication.md @@ -0,0 +1,284 @@ +# Authentication + +Firestack handles authentication for us out of the box, both with email/password-based authentication and through oauth providers (with a separate library to handle oauth providers). + +> Authentication requires Google Play services to be installed on Android. + +## Auth + +### Properties + +##### `authenticated: boolean` - Returns the current Firebase authentication state. +##### `currentUser: User | null` - Returns the currently signed-in user (or null). See the [User](/docs/api/authentication.md#user) class documentation for further usage. + +### Methods + +#### [`onAuthStateChanged(event: Function): Function`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#onAuthStateChanged) + +Listen for changes in the users auth state (logging in and out). This method returns a unsubscribe function to stop listening to events. Always ensure you unsubscribe from the listener when no longer needed to prevent updates to components no longer in use. + +```javascript +class Example extends React.Component { + + constructor() { + super(); + this.unsubscribe = null; + } + + componentDidMount() { + this.unsubscribe = firestack.auth().onAuthStateChanged((user) => { + if (user) { + // User is signed in. + } + }); + } + + componentWillUnmount() { + if (this.unsubscribe) { + this.unsubscribe(); + } + } + +} +``` + +#### [`createUserWithEmailAndPassword(email: string, password: string): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#createUserWithEmailAndPassword) + +We can create a user by calling the `createUserWithEmailAndPassword()` function. +The method accepts two parameters, an email and a password. + +```javascript +firestack.auth().createUserWithEmailAndPassword('foo@bar.com', '123456') + .then((user) => { + console.log('user created', user) + }) + .catch((err) => { + console.error('An error occurred', err); + }); +``` + +#### [`signInWithEmailAndPassword(email: string, password: string): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#signInWithEmailAndPassword) + +To sign a user in with their email and password, use the `signInWithEmailAndPassword()` function. +It accepts two parameters, the user's email and password: + +```javascript +firestack.auth().signInWithEmailAndPassword('foo@bar.com', '123456') + .then((user) => { + console.log('User successfully logged in', user) + }) + .catch((err) => { + console.error('User signin error', err); + }); +``` + +#### [`signInAnonymously(): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#signInAnonymously) + +Sign an anonymous user. If the user has already signed in, that user will be returned. + +```javascript +firestack.auth().signInAnonymously() + .then((user) => { + console.log('Anonymous user successfully logged in', user) + }) + .catch((err) => { + console.error('Anonymous user signin error', err); + }); +``` + +#### [`signInWithCredential(credential: Object): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#signInWithCredential) + +Sign in the user with a 3rd party credential provider. `credential` requires the following properties: + +```javascript +{ + provider: string, + token: string, + secret: string +} +``` + +```javascript +const credential = { + provider: 'facebook', + token: '12345', + secret: '6789', +}; + +firestack.auth().signInWithCredential(credential) + .then((user) => { + console.log('User successfully signed in', user) + }) + .catch((err) => { + console.error('User signin error', err); + }); +``` + +#### [`signInWithCustomToken(token: string): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#signInWithCustomToken) + +Sign a user in with a self-signed [JWT](https://jwt.io) token. + +To sign a user using a self-signed custom token, use the `signInWithCustomToken()` function. It accepts one parameter, the custom token: + +```javascript +firestack.auth().signInWithCustomToken('12345') + .then((user) => { + console.log('User successfully logged in', user) + }) + .catch((err) => { + console.error('User signin error', err); + }); +``` + +#### [`sendPasswordResetEmail(email: string): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#sendPasswordResetEmail) + +Sends a password reset email to the given email address. Unlike the web SDK, the email will contain a password reset link rather than a code. + +```javascript +firestack.auth().sendPasswordResetEmail('foo@bar.com') + .then(() => { + console.log('Password reset email sent'); + }) + .catch((error) => { + console.error('Unable send password reset email', error); + }); +``` + +#### [`signOut(): Promise`](https://firebase.google.com/docs/reference/js/firebase.auth.Auth#confirmPasswordReset) + +Completes the password reset process, given a confirmation code and new password. + +```javascript +firestack.auth().signOut() + .then(() => { + console.log('User signed out successfully'); + }) + .catch(); +``` + +## User + +User class returned from `firestack.auth().currentUser`. + +### Properties + +##### `displayName: string | null` - The user's display name (if available). +##### `email: string | null` - The user's email address (if available). +##### `emailVerified: boolean` - True if the user's email address has been verified. +##### `isAnonymous: boolean` +##### `photoURL: string | null` - The URL of the user's profile picture (if available). +##### `providerData: Object | null` - Additional provider-specific information about the user. +##### `providerId: string | null` - The authentication provider ID for the current user. For example, 'facebook.com', or 'google.com'. +##### `uid: string` - The user's unique ID. + +### Methods + +#### [`delete(): Promise`](https://firebase.google.com/docs/reference/js/firebase.User#delete) + +Delete the current user. + +```javascript +firestack.auth().currentUser + .delete() + .then() + .catch(); +``` + +#### [`getToken(): Promise`](https://firebase.google.com/docs/reference/js/firebase.User#getToken) + +Returns the users authentication token. + +```javascript +firestack.auth().currentUser + .getToken() + .then((token) => {}) + .catch(); +``` + + +#### [`reauthenticate(credential: Object): Promise`](https://firebase.google.com/docs/reference/js/firebase.User#reauthenticate) + +Reauthenticate the current user with credentials: + +```javascript +{ + provider: string, + token: string, + secret: string +} +``` + +```javascript +const credentials = { + provider: 'facebook.com', + token: '12345', + secret: '6789', +}; + +firestack.auth().currentUser + .reauthenticate(credentials) + .then() + .catch(); +``` + +#### [`reload(): Promise`](https://firebase.google.com/docs/reference/js/firebase.User#reload) + +Refreshes the current user. + +```javascript +firestack.auth().currentUser + .reload() + .then((user) => {}) + .catch(); +``` + +#### [`sendEmailVerification(): Promise`](https://firebase.google.com/docs/reference/js/firebase.User#sendEmailVerification) + +Sends a verification email to a user. This will Promise reject is the user is anonymous. + +```javascript +firestack.auth().currentUser + .sendEmailVerification() + .then() + .catch(); +``` + +#### [updateEmail(email: string)](https://firebase.google.com/docs/reference/js/firebase.User#updateEmail) + +Updates the user's email address. See Firebase docs for more information on security & email validation. This will Promise reject is the user is anonymous. + +```javascript +firestack.auth().updateUserEmail('foo@bar.com') + .then() + .catch(); +``` + +#### [updatePassword(password: string)](https://firebase.google.com/docs/reference/js/firebase.User#updatePassword) + +Important: this is a security sensitive operation that requires the user to have recently signed in. If this requirement isn't met, ask the user to authenticate again and then call firebase.User#reauthenticate. This will Promise reject is the user is anonymous. + +```javascript +firestack.auth().updatePassword('foobar1234') + .then() + .catch(); +``` + +#### [updateProfile(profile: Object)](https://firebase.google.com/docs/reference/js/firebase.User#updateProfile) + +Updates a user's profile data. Profile data should be an object of fields to update: + +```javascript +{ + displayName: string, + photoURL: string, +} +``` + +```javascript +firestack.auth() + .updateProfile({ + displayName: 'Ari Lerner' + }) + .then() + .catch(); +``` diff --git a/docs/api/cloud-messaging.md b/docs/api/cloud-messaging.md new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/docs/api/cloud-messaging.md @@ -0,0 +1 @@ + diff --git a/docs/api/database.md b/docs/api/database.md new file mode 100644 index 0000000..8597984 --- /dev/null +++ b/docs/api/database.md @@ -0,0 +1,220 @@ + +# Realtime Database + +Firestack mimics the [Web Firebase SDK Realtime Database](https://firebase.google.com/docs/database/web/read-and-write), whilst +providing support for devices in low/no data connection state. + +All Realtime Database operations are accessed via `database()`. + +Basic read example: +```javascript +firestack.database() + .ref('posts') + .on('value', (snapshot) => { + const value = snapshot.val(); + }); +``` + +Read for export: +```javascript +firestack.database() + .ref('posts') + .on('value', (snapshot) => { + const value = snapshot.exportVal(); + }); +``` +This includes hidden properties like `.priority` + +Basic write example: +```javascript +firestack.database() + .ref('posts/1234') + .set({ + title: 'My awesome post', + content: 'Some awesome content', + }); +``` + +Test value exists at location: +```javascript +firestack.database() + .ref('posts/1234') + .on('value', (snapshot) => { + const exists = snapshot.exists(); + }); +``` + +Basic write with priority example: +```javascript +firestack.database() + .ref('posts/1235') + .setWithPriority({ + title: 'Another Awesome Post', + content: 'Some awesome content', + }, 10); +``` +Useful for `orderByPriority` queries. + + +Transaction Support: +```javascript +firestack.database() + .ref('posts/1234/title') + .transaction((title) => 'My Awesome Post'); +``` + +## Unmounted components + +Listening to database updates on unmounted components will trigger a warning: + +> Can only update a mounted or mounting component. This usually means you called setState() on an unmounted component. This is a no-op. Please check the code for the undefined component. + +It is important to always unsubscribe the reference from receiving new updates once the component is no longer in use. +This can be achived easily using [Reacts Component Lifecycle](https://facebook.github.io/react/docs/react-component.html#the-component-lifecycle) events: + +Always ensure the handler function provided is of the same reference so Firestack can unsubscribe the ref listener. + +```javascript +class MyComponent extends Component { + constructor() { + super(); + this.ref = null; + } + + // On mount, subscribe to ref updates + componentDidMount() { + this.ref = firestack.database().ref('posts/1234'); + this.ref.on('value', this.handlePostUpdate); + } + + // On unmount, ensure we no longer listen for updates + componentWillUnmount() { + if (this.ref) { + this.ref.off('value', this.handlePostUpdate); + } + } + + // Bind the method only once to keep the same reference + handlePostUpdate = (snapshot) => { + console.log('Post Content', snapshot.val()); + } + + render() { + return null; + } +} + +``` + +## Usage in offline environments + +### Reading data + +Firstack allows the database instance to [persist on disk](https://firebase.google.com/docs/database/android/offline-capabilities) if enabled. +To enable database persistence, call the following method before calls are made: + +```javascript +firestack.database().setPersistence(true); +``` + +Any subsequent calls to Firebase stores the data for the ref on disk. + +### Writing data + +Out of the box, Firebase has great support for writing operations in offline environments. Calling a write command whilst offline +will always trigger any subscribed refs with new data. Once the device reconnects to Firebase, it will be synced with the server. + +The following todo code snippet will work in both online and offline environments: + +```javascript +// Assume the todos are stored as an object value on Firebase as: +// { name: string, complete: boolean } + +class ToDos extends Component { + constructor() { + super(); + this.ref = null; + this.listView = new ListView.DataSource({ + rowHasChanged: (r1, r2) => r1 !== r2, + }); + + this.state = { + todos: this.listView.cloneWithRows({}), + }; + + // Keep a local reference of the TODO items + this.todos = {}; + } + + // Load the Todos on mount + componentDidMount() { + this.ref = firestack.database().ref('users/1234/todos'); + this.ref.on('value', this.handleToDoUpdate); + } + + // Unsubscribe from the todos on unmount + componentWillUnmount() { + if (this.ref) { + this.ref.off('value', this.handleToDoUpdate); + } + } + + // Handle ToDo updates + handleToDoUpdate = (snapshot) => { + this.todos = snapshot.val() || {}; + + this.setState({ + todos: this.listView.cloneWithRows(this.todos), + }); + } + + // Add a new ToDo onto Firebase + // If offline, this will still trigger an update to handleToDoUpdate + addToDo() { + firestack.database() + .ref('users/1234/todos') + .set({ + ...this.todos, { + name: 'Yet another todo...', + complete: false, + }, + }); + } + + // Render a ToDo row + renderToDo(todo) { + // Dont render the todo if its complete + if (todo.complete) { + return null; + } + + return ( + <View> + <Text>{todo.name}</Text> + </View> + ); + } + + // Render the list of ToDos with a Button + render() { + return ( + <View> + <ListView + dataSource={this.state.todos} + renderRow={(...args) => this.renderToDo(...args)} + /> + + <Button + title={'Add ToDo'} + onPress={() => this.addToDo} + /> + <View> + ); + } +``` + +#### Differences between `.on` & `.once` + +With persistence enabled, any calls to a ref with `.once` will always read the data from disk and not contact the server. +On behavious differently, by first checking for a connection and if none exists returns the persisted data. If it successfully connects +to the server, the new data will be returned and the disk data will be updated. diff --git a/docs/api/events.md b/docs/api/events.md new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/docs/api/events.md @@ -0,0 +1 @@ + diff --git a/docs/api/presence.md b/docs/api/presence.md new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/docs/api/presence.md @@ -0,0 +1 @@ + diff --git a/docs/api/remote-config.md b/docs/api/remote-config.md new file mode 100644 index 0000000..413c634 --- /dev/null +++ b/docs/api/remote-config.md @@ -0,0 +1 @@ +# Remote Config diff --git a/docs/api/server-value.md b/docs/api/server-value.md new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/docs/api/server-value.md @@ -0,0 +1 @@ + diff --git a/docs/api/storage.md b/docs/api/storage.md new file mode 100644 index 0000000..11c7bf4 --- /dev/null +++ b/docs/api/storage.md @@ -0,0 +1,91 @@ + +# Storage + +Firestack mimics the [Web Firebase SDK Storage](https://firebase.google.com/docs/storage/web/start), whilst +providing some iOS and Android specific functionality. + +All Storage operations are accessed via `storage()`. + +## Uploading files + +### Simple + +```javascript +firestack.storage() + .ref('/files/1234') + .putFile('/path/to/file/1234') + .then(uploadedFile => { + //success + }) + .catch(err => { + //Error + }); +``` + +### Listen to upload state + +```javascript +const unsubscribe = firestack.storage() + .ref('/files/1234') + .putFile('/path/to/file/1234') + .on('state_changed', snapshot => { + //Current upload state + }, err => { + //Error + unsubscribe(); + }, uploadedFile => { + //Success + unsubscribe(); + }); +``` + +## Downloading files + +### Simple + +```javascript +firestack.storage() + .ref('/files/1234') + .downloadFile('/path/to/save/file') + .then(downloadedFile => { + //success + }) + .catch(err => { + //Error + }); +``` + +### Listen to download state + +```javascript +const unsubscribe = firestack.storage() + .ref('/files/1234') + .downloadFile('/path/to/save/file') + .on('state_changed', snapshot => { + //Current download state + }, err => { + //Error + unsubscribe(); + }, downloadedFile => { + //Success + unsubscribe(); + }); +``` + +## TODO + +There are a few methods which have not yet been implemented for Storage: + +### Reference +- put() +- putString() + +### UploadTask +- cancel() +- pause() +- resume() + +### DownloadTask +- cancel() +- pause() +- resume() diff --git a/docs/firebase-setup.md b/docs/firebase-setup.md new file mode 100644 index 0000000..af3dc67 --- /dev/null +++ b/docs/firebase-setup.md @@ -0,0 +1,88 @@ +# Firebase Setup + +The Firestack library is intended on making it easy to work with [Firebase](https://firebase.google.com/) and provides a small native shim to the Firebase native code. + +To add Firebase to your project, make sure to create a project in the [Firebase console](https://firebase.google.com/console) + + + +Each platform uses a different setup method after creating the project. + +## iOS + +After creating a Firebase project, click on the [Add Firebase to your iOS app](http://d.pr/i/3sEL.png) and follow the steps from there to add the configuration file. You do _not_ need to set up a cocoapods project (this is already done through firestack). Make sure not to forget the `Copy Files` phase in iOS. + +[Download the Firebase config file](https://support.google.com/firebase/answer/7015592) and place it in your app directory next to your app source code: + + + +Once you download the configuration file, make sure you place it in the root of your Xcode project. Every different Bundle ID (aka, even different project variants needs their own configuration file). + +Lastly, due to some dependencies requirements, Firestack supports iOS versions 8.0 and up. Make sure to update the minimum version of your iOS app to `8.0`. + +## Android + +There are several ways to setup Firebase on Android. The _easiest_ way is to pass the configuration settings in JavaScript. In that way, there is no setup for the native platform. + +### google-services.json setup +If you prefer to include the default settings in the source of your app, download the `google-services.json` file provided by Firebase in the _Add Firebase to Android_ platform menu in your Firebase configuration console. + +Next you'll have to add the google-services gradle plugin in order to parse it. + +Add the google-services gradle plugin as a dependency in the *project* level build.gradle +`android/build.gradle` +```java +buildscript { + // ... + dependencies { + // ... + classpath 'com.google.gms:google-services:3.0.0' + } +} +``` + +In your app build.gradle file, add the gradle plugin at the VERY BOTTOM of the file (below all dependencies) +`android/app/build.gradle` +```java +apply plugin: 'com.google.gms.google-services' +``` + +## Usage + +After creating a Firebase project and installing the library, we can use it in our project by importing the library in our JavaScript: + +```javascript +import Firestack from 'react-native-firestack' +``` + +We need to tell the Firebase library we want to _configure_ the project. Firestack provides a way to configure both the native and the JavaScript side of the project at the same time with a single command: + +```javascript +const firestack = new Firestack(); +``` + +We can pass _custom_ options by passing an object with configuration options. The configuration object will be generated first by the native configuration object, if set and then will be overridden if passed in JS. That is, all of the following key/value pairs are optional if the native configuration is set. + +| option | type | Default Value | Description | +|----------------|----------|-------------------------|----------------------------------------| +| debug | bool | false | When set to true, Firestack will log messages to the console and fire `debug` events we can listen to in `js` | +| bundleID | string | Default from app `[NSBundle mainBundle]` | The bundle ID for the app to be bundled with | +| googleAppID | string | "" | The Google App ID that is used to uniquely identify an instance of an app. | +| databaseURL | string | "" | The database root (i.e. https://my-app.firebaseio.com) | +| deepLinkURLScheme | string | "" | URL scheme to set up durable deep link service | +| storageBucket | string | "" | The Google Cloud storage bucket name | +| androidClientID | string | "" | The Android client ID used in Google AppInvite when an iOS app has it's android version | +| GCMSenderID | string | "" | The Project number from the Google Developer's console used to configure Google Cloud Messaging | +| trackingID | string | "" | The tracking ID for Google Analytics | +| clientID | string | "" | The OAuth2 client ID for iOS application used to authenticate Google Users for signing in with Google | +| APIKey | string | "" | The secret iOS API key used for authenticating requests from our app | + +For instance: + +```javascript +const configurationOptions = { + debug: true +}; +const firestack = new Firestack(configurationOptions); +firestack.on('debug', msg => console.log('Received debug message', msg)) +``` diff --git a/docs/installation.android.md b/docs/installation.android.md new file mode 100644 index 0000000..2078ed7 --- /dev/null +++ b/docs/installation.android.md @@ -0,0 +1,55 @@ +# Android Installation + +The simplest way of installing on Android is to use React Native linker: + +``` +react-native link react-native-firestack +``` + +## Manually + +To install `react-native-firestack` manually in our project, we'll need to import the package from `io.fullstack.firestack` in our project's `android/app/src/main/java/com/[app name]/MainApplication.java` and list it as a package for ReactNative in the `getPackages()` function: + +```java +package com.appName; +// ... +import io.fullstack.firestack.FirestackPackage; +// ... +public class MainApplication extends Application implements ReactApplication { + // ... + + @Override + protected List<ReactPackage> getPackages() { + return Arrays.<ReactPackage>asList( + new MainReactPackage(), + new FirestackPackage() // <-- Add this line + ); + } + }; + // ... +} +``` + +We'll also need to list it in our `android/app/build.gradle` file as a dependency that we want React Native to compile. In the `dependencies` listing, add the `compile` line: + +```java +dependencies { + compile project(':react-native-firestack') +} +``` + +Add to `AndroidManifest.xml` file +```diff + <activity android:name="com.facebook.react.devsupport.DevSettingsActivity" /> ++ <service android:name="io.fullstack.firestack.FirestackMessagingService"> ++ <intent-filter> ++ <action android:name="com.google.firebase.MESSAGING_EVENT"/> ++ </intent-filter> ++ </service> + ++ <service android:name="io.fullstack.firestack.FirestackInstanceIdService" android:exported="false"> ++ <intent-filter> ++ <action android:name="com.google.firebase.INSTANCE_ID_EVENT"/> ++ </intent-filter> ++ </service> +``` diff --git a/docs/installation.ios.md b/docs/installation.ios.md new file mode 100644 index 0000000..688379a --- /dev/null +++ b/docs/installation.ios.md @@ -0,0 +1,67 @@ +#iOS Installation + +If you don't want to use cocoapods, you don't need to use it! Just make sure you link the Firebase libraries in your project manually. For more information, check out the relevant Firebase docs at [https://firebase.google.com/docs/ios/setup#frameworks](https://firebase.google.com/docs/ios/setup#frameworks). + +## cocoapods + +Unfortunately, due to AppStore restrictions, we currently do _not_ package Firebase libraries in with Firestack. However, the good news is we've automated the process (with many thanks to the Auth0 team for inspiration) of setting up with cocoapods. This will happen automatically upon linking the package with `react-native-cli`. + +**Remember to use the `ios/[YOUR APP NAME].xcworkspace` instead of the `ios/[YOUR APP NAME].xcproj` file from now on**. + +We need to link the package with our development packaging. We have two options to handle linking: + +#### Automatically with react-native-cli + +React native ships with a `link` command that can be used to link the projects together, which can help automate the process of linking our package environments. + +```bash +react-native link react-native-firestack +``` + +Update the newly installed pods once the linking is done: + +```bash +cd ios && pod update --verbose +``` + +#### Manually + +If you prefer not to use `react-native link`, we can manually link the package together with the following steps, after `npm install`: + +**A.** In XCode, right click on `Libraries` and find the `Add Files to [project name]`. + + + +**B.** Add the `node_modules/react-native-firestack/ios/Firestack.xcodeproj` + + + +**C.** Ensure that the `Build Settings` of the `Firestack.xcodeproj` project is ticked to _All_ and it's `Header Search Paths` include both of the following paths _and_ are set to _recursive_: + + 1. `$(SRCROOT)/../../react-native/React` + 2. `$(SRCROOT)/../node_modules/react-native/React` + 3. `${PROJECT_DIR}/../../../ios/Pods` + + + +**D.** Setting up cocoapods + +Since we're dependent upon cocoapods (or at least the Firebase libraries being available at the root project -- i.e. your application), we have to make them available for Firestack to find them. + +Using cocoapods is the easiest way to get started with this linking. Add or update a `Podfile` at `ios/Podfile` in your app with the following: + +```ruby +source 'https://github.com/CocoaPods/Specs.git' +[ + 'Firebase/Core', + 'Firebase/Auth', + 'Firebase/Storage', + 'Firebase/Database', + 'Firebase/RemoteConfig', + 'Firebase/Messaging' +].each do |lib| + pod lib +end +``` + +Then you can run `(cd ios && pod install)` to get the pods opened. If you do use this route, remember to use the `.xcworkspace` file. diff --git a/docs/redux.md b/docs/redux.md new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/docs/redux.md @@ -0,0 +1 @@ + diff --git a/firestack.android.js b/firestack.android.js index 15d7185..3ebc35e 100644 --- a/firestack.android.js +++ b/firestack.android.js @@ -1,7 +1,6 @@ /** - * @providesModule Firestack * @flow */ import Firestack from './lib/firestack' -export default Firestack \ No newline at end of file +export default Firestack diff --git a/firestack.ios.js b/firestack.ios.js index bc1e69b..3ebc35e 100644 --- a/firestack.ios.js +++ b/firestack.ios.js @@ -1,5 +1,4 @@ /** - * @providesModule Firestack * @flow */ import Firestack from './lib/firestack' diff --git a/ios/Firestack/Firestack.h b/ios/Firestack/Firestack.h index ffb5cc3..3b4af75 100644 --- a/ios/Firestack/Firestack.h +++ b/ios/Firestack/Firestack.h @@ -8,9 +8,9 @@ #define Firestack_h #import <UIKit/UIKit.h> -#import "RCTBridgeModule.h" -#import "RCTEventDispatcher.h" -#import "RCTEventEmitter.h" +#import <React/RCTBridgeModule.h> +#import <React/RCTEventDispatcher.h> +#import <React/RCTEventEmitter.h> @interface Firestack : RCTEventEmitter <RCTBridgeModule> { } diff --git a/ios/Firestack/Firestack.m b/ios/Firestack/Firestack.m index 284f38b..096d848 100644 --- a/ios/Firestack/Firestack.m +++ b/ios/Firestack/Firestack.m @@ -53,10 +53,12 @@ + (void) initializeFirestack:(Firestack *) instance dispatch_once(&onceToken, ^{ _sharedInstance = instance; + /* RCTReloadNotification is deprecated [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(reloadFirestack) name:RCTReloadNotification object:nil]; + */ [[NSNotificationCenter defaultCenter] postNotificationName:kFirestackInitialized diff --git a/ios/Firestack/FirestackAnalytics.h b/ios/Firestack/FirestackAnalytics.h index b5ed857..6ccabd7 100644 --- a/ios/Firestack/FirestackAnalytics.h +++ b/ios/Firestack/FirestackAnalytics.h @@ -9,7 +9,7 @@ #ifndef FirestackAnalytics_h #define FirestackAnalytics_h -#import "RCTBridgeModule.h" +#import <React/RCTBridgeModule.h> @interface FirestackAnalytics : NSObject <RCTBridgeModule> { diff --git a/ios/Firestack/FirestackAnalytics.m b/ios/Firestack/FirestackAnalytics.m index cf69e0b..202c912 100644 --- a/ios/Firestack/FirestackAnalytics.m +++ b/ios/Firestack/FirestackAnalytics.m @@ -21,38 +21,48 @@ - (void)dealloc RCT_EXPORT_MODULE(FirestackAnalytics); // Implementation -RCT_EXPORT_METHOD(logEventWithName:(NSString *)name - props:(NSDictionary *)props - callback:(RCTResponseSenderBlock) callback) +RCT_EXPORT_METHOD(logEvent:(NSString *)name + props:(NSDictionary *)props) { NSString *debugMsg = [NSString stringWithFormat:@"%@: %@ with %@", @"FirestackAnalytics", name, props]; [[Firestack sharedInstance] debugLog:@"logEventWithName called" msg:debugMsg]; - [FIRAnalytics logEventWithName:name parameters:props]; - callback(@[[NSNull null], @YES]); + [FIRAnalytics logEventWithName:name parameters:props]; } -RCT_EXPORT_METHOD(setEnabled:(BOOL) enabled - callback:(RCTResponseSenderBlock) callback) +RCT_EXPORT_METHOD(setAnalyticsCollectionEnabled:(BOOL) enabled) { [[FIRAnalyticsConfiguration sharedInstance] setAnalyticsCollectionEnabled:enabled]; - callback(@[[NSNull null], @YES]); } -RCT_EXPORT_METHOD(setUser: (NSString *) id - props:(NSDictionary *) props - callback:(RCTResponseSenderBlock) callback) +RCT_EXPORT_METHOD(setCurrentScreen:(NSString *) screenName + screenClass:(NSString *) screenClassOverriew) +{ + [FIRAnalytics setScreenName:screenName screenClass:screenClassOverriew]; +} + +RCT_EXPORT_METHOD(setMinimumSessionDuration:(NSNumber *) milliseconds) +{ + //Not implemented on iOS +} + +RCT_EXPORT_METHOD(setSessionTimeoutDuration:(NSNumber *) milliseconds) +{ + //Not implemented on iOS +} + +RCT_EXPORT_METHOD(setUserId: (NSString *) id + props:(NSDictionary *) props) { [FIRAnalytics setUserID:id]; - NSMutableArray *allKeys = [[props allKeys] mutableCopy]; - for (NSString *key in allKeys) { - NSString *val = [props valueForKey:key]; - [FIRAnalytics setUserPropertyString:val forName:key]; - } +} - callback(@[[NSNull null], @YES]); +RCT_EXPORT_METHOD(setUserProperty: (NSString *) name + value:(NSString *) value) +{ + [FIRAnalytics setUserPropertyString:value forName:name]; } @end diff --git a/ios/Firestack/FirestackAuth.h b/ios/Firestack/FirestackAuth.h index d6e6679..dad5d52 100644 --- a/ios/Firestack/FirestackAuth.h +++ b/ios/Firestack/FirestackAuth.h @@ -10,8 +10,8 @@ #define FirestackAuth_h #import "Firebase.h" -#import "RCTEventEmitter.h" -#import "RCTBridgeModule.h" +#import <React/RCTEventEmitter.h> +#import <React/RCTBridgeModule.h> @interface FirestackAuth : RCTEventEmitter <RCTBridgeModule> { FIRAuthStateDidChangeListenerHandle authListenerHandle; diff --git a/ios/Firestack/FirestackAuth.m b/ios/Firestack/FirestackAuth.m index bcd0919..55d3dd0 100644 --- a/ios/Firestack/FirestackAuth.m +++ b/ios/Firestack/FirestackAuth.m @@ -20,28 +20,27 @@ @implementation FirestackAuth (RCTResponseSenderBlock) callBack) { @try { - [[FIRAuth auth] signInAnonymouslyWithCompletion - :^(FIRUser *user, NSError *error) { - if (!user) { - NSDictionary *evt = @{ - @"eventName": AUTH_ANONYMOUS_ERROR_EVENT, - @"errorMessage": [error localizedDescription] - }; - - - [self sendJSEvent:AUTH_CHANGED_EVENT - props: evt]; - - callBack(@[evt]); - } else { - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callBack(@[[NSNull null], userProps]); - } - }]; + [[FIRAuth auth] signInAnonymouslyWithCompletion + :^(FIRUser *user, NSError *error) { + if (!user) { + NSDictionary *evt = @{ + @"eventName": AUTH_ANONYMOUS_ERROR_EVENT, + @"msg": [error localizedDescription] + }; + + + [self sendJSEvent:AUTH_CHANGED_EVENT + props: evt]; + + callBack(@[evt]); + } else { + [self userCallback:callBack user:user]; + } + }]; } @catch(NSException *ex) { NSDictionary *eventError = @{ @"eventName": AUTH_ANONYMOUS_ERROR_EVENT, - @"errorMessage": ex.reason + @"msg": ex.reason }; [self sendJSEvent:AUTH_ERROR_EVENT @@ -58,43 +57,33 @@ @implementation FirestackAuth [[FIRAuth auth] signInWithCustomToken:customToken completion:^(FIRUser *user, NSError *error) { - + if (user != nil) { - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], userProps]); + [self userCallback:callback user:user]; } else { - NSDictionary *err = - [FirestackErrors handleFirebaseError:AUTH_ERROR_EVENT - error:error - withUser:user]; - callback(@[err]); + [self userErrorCallback:callback error:error user:user msg:AUTH_ERROR_EVENT]; } }]; } RCT_EXPORT_METHOD(signInWithProvider: - (NSString *)provider - token:(NSString *)authToken - secret:(NSString *)authTokenSecret + (NSDictionary *)credentialData callback:(RCTResponseSenderBlock)callback) { - FIRAuthCredential *credential = [self getCredentialForProvider:provider - token:authToken - secret:authTokenSecret]; + FIRAuthCredential *credential = [self getCredentialForProvider:credentialData]; if (credential == nil) { NSDictionary *err = @{ @"error": @"Unhandled provider" }; return callback(@[err]); } - + @try { [[FIRAuth auth] signInWithCredential:credential completion:^(FIRUser *user, NSError *error) { if (user != nil) { // User is signed in. - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], userProps]); + [self userCallback:callback user:user]; } else { NSLog(@"An error occurred: %@", [error localizedDescription]); // No user is signed in. @@ -134,7 +123,7 @@ @implementation FirestackAuth self->authListenerHandle = [[FIRAuth auth] addAuthStateDidChangeListener:^(FIRAuth *_Nonnull auth, FIRUser *_Nullable user) { - + if (user != nil) { // User is signed in. [self userPropsFromFIRUserWithToken:user @@ -144,15 +133,14 @@ @implementation FirestackAuth sendJSEvent:AUTH_CHANGED_EVENT props: @{ @"eventName": @"userTokenError", - @"authenticated": @((BOOL)true), - @"errorMessage": [error localizedFailureReason] + @"msg": [error localizedDescription] }]; } else { [self sendJSEvent:AUTH_CHANGED_EVENT props: @{ @"eventName": @"user", - @"authenticated": @((BOOL)true), + @"authenticated": @(true), @"user": userProps }]; } @@ -165,7 +153,7 @@ @implementation FirestackAuth [self sendJSEvent:AUTH_CHANGED_EVENT props:@{ @"eventName": @"no_user", - @"authenticated": @((BOOL)false), + @"authenticated": @(false), @"error": err }]; } @@ -184,11 +172,9 @@ @implementation FirestackAuth RCT_EXPORT_METHOD(getCurrentUser:(RCTResponseSenderBlock)callback) { FIRUser *user = [FIRAuth auth].currentUser; - + if (user != nil) { - NSMutableDictionary *userProps = [self userPropsFromFIRUser:user]; - [userProps setValue: @((BOOL)true) forKey: @"authenticated"]; - callback(@[[NSNull null], userProps]); + [self userCallback:callback user:user]; } else { // No user is signed in. NSDictionary *err = @{ @@ -208,8 +194,7 @@ @implementation FirestackAuth completion:^(FIRUser *_Nullable user, NSError *_Nullable error) { if (user != nil) { - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], userProps]); + [self userCallback:callback user:user]; } else { NSDictionary *err = @{ @"error": @"createUserWithEmailError", @@ -229,17 +214,9 @@ @implementation FirestackAuth password:password completion:^(FIRUser *user, NSError *error) { if (user != nil) { - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - - callback(@[[NSNull null], @{ - @"user": userProps - }]); + [self userCallback:callback user:user]; } else { - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"signinError" - error:error - withUser:user]; - callback(@[err]); + [self userErrorCallback:callback error:error user:user msg:@"signinError"]; } }]; } @@ -248,49 +225,47 @@ @implementation FirestackAuth callback:(RCTResponseSenderBlock) callback) { FIRUser *user = [FIRAuth auth].currentUser; - - [user updateEmail:email completion:^(NSError *_Nullable error) { - if (error) { - // An error happened. - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"updateEmailError" - error:error - withUser:user]; - callback(@[err]); - } else { - // Email updated. - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], userProps]); - } - }]; + + if (user) { + [user updateEmail:email completion:^(NSError *_Nullable error) { + if (error) { + // An error happened. + [self userErrorCallback:callback error:error user:user msg:@"updateEmailError"]; + } else { + // Email updated. + [self userCallback:callback user:user]; + } + }]; + } else { + [self noUserCallback:callback isError:true]; + } } RCT_EXPORT_METHOD(updateUserPassword:(NSString *)newPassword callback:(RCTResponseSenderBlock) callback) { - + FIRUser *user = [FIRAuth auth].currentUser; - - [user updatePassword:newPassword completion:^(NSError *_Nullable error) { - if (error) { - // An error happened. - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"updateUserPasswordError" - error:error - withUser:user]; - callback(@[err]); - } else { - // Email updated. - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], userProps]); - } - }]; + + if (user) { + [user updatePassword:newPassword completion:^(NSError *_Nullable error) { + if (error) { + // An error happened. + [self userErrorCallback:callback error:error user:user msg:@"updateUserPasswordError"]; + } else { + // Email updated. + [self userCallback:callback user:user]; + } + }]; + } else { + [self noUserCallback:callback isError:true]; + } } RCT_EXPORT_METHOD(sendPasswordResetWithEmail:(NSString *)email callback:(RCTResponseSenderBlock) callback) { - + [[FIRAuth auth] sendPasswordResetWithEmail:email completion:^(NSError *_Nullable error) { if (error) { @@ -312,81 +287,71 @@ @implementation FirestackAuth RCT_EXPORT_METHOD(deleteUser:(RCTResponseSenderBlock) callback) { FIRUser *user = [FIRAuth auth].currentUser; - - [user deleteWithCompletion:^(NSError *_Nullable error) { - if (error) { - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"deleteUserError" - error:error - withUser:user]; - callback(@[err]); - } else { - callback(@[[NSNull null], @{@"result": @(true)}]); - } - }]; + + if (user) { + [user deleteWithCompletion:^(NSError *_Nullable error) { + if (error) { + [self userErrorCallback:callback error:error user:user msg:@"deleteUserError"]; + } else { + callback(@[[NSNull null], @{@"result": @(true)}]); + } + }]; + } else { + [self noUserCallback:callback isError:true]; + } } RCT_EXPORT_METHOD(getToken:(RCTResponseSenderBlock) callback) { FIRUser *user = [FIRAuth auth].currentUser; - - [user getTokenWithCompletion:^(NSString *token, NSError *_Nullable error) { - if (error) { - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"getTokenError" - error:error - withUser:user]; - callback(@[err]); - } else { - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], @{@"token": token, @"user": userProps}]); - } - }]; + + if (user) { + [user getTokenWithCompletion:^(NSString *token, NSError *_Nullable error) { + if (error) { + [self userErrorCallback:callback error:error user:user msg:@"getTokenError"]; + } else { + callback(@[[NSNull null], token]); + } + }]; + } else { + [self noUserCallback:callback isError:true]; + } } RCT_EXPORT_METHOD(getTokenWithCompletion:(RCTResponseSenderBlock) callback) { FIRUser *user = [FIRAuth auth].currentUser; - - [user getTokenWithCompletion:^(NSString *token , NSError *_Nullable error) { - if (error) { - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"getTokenWithCompletion" - error:error - withUser:user]; - callback(@[err]); - } else { - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], @{@"token": token, @"user": userProps}]); - } - }]; + + if (user) { + [user getTokenWithCompletion:^(NSString *token , NSError *_Nullable error) { + if (error) { + [self userErrorCallback:callback error:error user:user msg:@"getTokenWithCompletion"]; + } else { + callback(@[[NSNull null], token]); + } + }]; + } else { + [self noUserCallback:callback isError:true]; + } } RCT_EXPORT_METHOD(reauthenticateWithCredentialForProvider: - (NSString *)provider - token:(NSString *)authToken - secret:(NSString *)authTokenSecret + (NSDictionary *)credentialData callback:(RCTResponseSenderBlock)callback) { - FIRAuthCredential *credential = [self getCredentialForProvider:provider - token:authToken - secret:authTokenSecret]; + FIRAuthCredential *credential = [self getCredentialForProvider:credentialData]; if (credential == nil) { NSDictionary *err = @{ @"error": @"Unhandled provider" }; return callback(@[err]); } - + FIRUser *user = [FIRAuth auth].currentUser; - + [user reauthenticateWithCredential:credential completion:^(NSError *_Nullable error) { if (error) { - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"reauthenticateWithCredentialForProviderError" - error:error - withUser:user]; - callback(@[err]); + [self userErrorCallback:callback error:error user:user msg:@"reauthenticateWithCredentialForProviderError"]; } else { callback(@[[NSNull null], @{@"result": @(true)}]); } @@ -398,38 +363,38 @@ @implementation FirestackAuth callback:(RCTResponseSenderBlock) callback) { FIRUser *user = [FIRAuth auth].currentUser; - FIRUserProfileChangeRequest *changeRequest = [user profileChangeRequest]; - - NSMutableArray *allKeys = [[userProps allKeys] mutableCopy]; - for (NSString *key in allKeys) { - // i.e. changeRequest.displayName = userProps[displayName]; - @try { - if ([key isEqualToString:@"photoURL"]) { - NSURL *url = [NSURL URLWithString:[userProps valueForKey:key]]; - [changeRequest setValue:url forKey:key]; - } else { - [changeRequest setValue:[userProps objectForKey:key] forKey:key]; - } - } - @catch (NSException *exception) { - NSLog(@"Exception occurred while configuring: %@", exception); - } - @finally { - [changeRequest commitChangesWithCompletion:^(NSError *_Nullable error) { - if (error) { - // An error happened. - NSDictionary *err = - [FirestackErrors handleFirebaseError:@"updateEmailError" - error:error - withUser:user]; - callback(@[err]); + + if (user) { + FIRUserProfileChangeRequest *changeRequest = [user profileChangeRequest]; + + NSMutableArray *allKeys = [[userProps allKeys] mutableCopy]; + for (NSString *key in allKeys) { + // i.e. changeRequest.displayName = userProps[displayName]; + @try { + if ([key isEqualToString:@"photoURL"]) { + NSURL *url = [NSURL URLWithString:[userProps valueForKey:key]]; + [changeRequest setValue:url forKey:key]; } else { - // Profile updated. - NSDictionary *userProps = [self userPropsFromFIRUser:user]; - callback(@[[NSNull null], userProps]); + [changeRequest setValue:[userProps objectForKey:key] forKey:key]; } - }]; + } + @catch (NSException *exception) { + NSLog(@"Exception occurred while configuring: %@", exception); + } + @finally { + [changeRequest commitChangesWithCompletion:^(NSError *_Nullable error) { + if (error) { + // An error happened. + [self userErrorCallback:callback error:error user:user msg:@"updateEmailError"]; + } else { + // Profile updated. + [self userCallback:callback user:user]; + } + }]; + } } + } else { + [self noUserCallback:callback isError:true]; } } @@ -444,12 +409,12 @@ - (NSDictionary *) userPropsFromFIRUser:(FIRUser *) user @"refreshToken": user.refreshToken, @"providerID": user.providerID } mutableCopy]; - + if ([user valueForKey:@"photoURL"] != nil) { [userProps setValue: [NSString stringWithFormat:@"%@", user.photoURL] forKey:@"photoURL"]; } - + return userProps; } @@ -461,25 +426,29 @@ - (void) userPropsFromFIRUserWithToken:(FIRUser *) user if (error != nil) { return callback(nil, error); } - + [userProps setValue:token forKey:@"idToken"]; callback(userProps, nil); }]; } -- (FIRAuthCredential *)getCredentialForProvider:(NSString *)provider - token:(NSString *)authToken - secret:(NSString *)authTokenSecret +- (FIRAuthCredential *)getCredentialForProvider:(NSDictionary *)credentialData { FIRAuthCredential *credential; - if ([provider compare:@"twitter" options:NSCaseInsensitiveSearch] == NSOrderedSame) { - credential = [FIRTwitterAuthProvider credentialWithToken:authToken - secret:authTokenSecret]; - } else if ([provider compare:@"facebook" options:NSCaseInsensitiveSearch] == NSOrderedSame) { - credential = [FIRFacebookAuthProvider credentialWithAccessToken:authToken]; - } else if ([provider compare:@"google" options:NSCaseInsensitiveSearch] == NSOrderedSame) { - credential = [FIRGoogleAuthProvider credentialWithIDToken:authToken - accessToken:authTokenSecret]; + NSString *provider = [credentialData valueForKey:@"provider"]; + if ([provider compare:@"twitter.com" options:NSCaseInsensitiveSearch] == NSOrderedSame) { + NSString *accessToken = [credentialData valueForKey:@"accessToken"]; + NSString *secret = [credentialData valueForKey:@"secret"]; + credential = [FIRTwitterAuthProvider credentialWithToken:accessToken + secret:secret]; + } else if ([provider compare:@"facebook.com" options:NSCaseInsensitiveSearch] == NSOrderedSame) { + NSString *accessToken = [credentialData valueForKey:@"accessToken"]; + credential = [FIRFacebookAuthProvider credentialWithAccessToken:accessToken]; + } else if ([provider compare:@"google.com" options:NSCaseInsensitiveSearch] == NSOrderedSame) { + NSString *idToken = [credentialData valueForKey:@"idToken"]; + NSString *accessToken = [credentialData valueForKey:@"accessToken"]; + credential = [FIRGoogleAuthProvider credentialWithIDToken:idToken + accessToken:accessToken]; } else { NSLog(@"Provider not yet handled: %@", provider); } @@ -495,15 +464,44 @@ - (void) sendJSEvent:(NSString *)title props:(NSDictionary *)props { @try { - if (self->listening) { - [self sendEventWithName:title - body:props]; - } + if (self->listening) { + [self sendEventWithName:title + body:props]; + } } @catch (NSException *err) { NSLog(@"An error occurred in sendJSEvent: %@", [err debugDescription]); } } +- (void) userCallback:(RCTResponseSenderBlock) callback + user:(FIRUser *) user { + NSDictionary *userProps = [self userPropsFromFIRUser:user]; + callback(@[[NSNull null], userProps]); +} + +- (void) noUserCallback:(RCTResponseSenderBlock) callback + isError:(Boolean) isError { + if (isError) { + NSDictionary *err = @{ + @"error": @"Unhandled provider" + }; + return callback(@[err]); + + } + return callback(@[[NSNull null], [NSNull null]]); +} + +- (void) userErrorCallback:(RCTResponseSenderBlock) callback + error:(NSError *)error + user:(FIRUser *) user + msg:(NSString *) msg { + // An error happened. + NSDictionary *err = [FirestackErrors handleFirebaseError:msg + error:error + withUser:user]; + callback(@[err]); +} + @end diff --git a/ios/Firestack/FirestackCloudMessaging.h b/ios/Firestack/FirestackCloudMessaging.h index 3e7c98b..df46cfe 100644 --- a/ios/Firestack/FirestackCloudMessaging.h +++ b/ios/Firestack/FirestackCloudMessaging.h @@ -10,9 +10,9 @@ #define FirestackCloudMessaging_h #import "Firebase.h" -#import "RCTEventEmitter.h" -#import "RCTBridgeModule.h" -#import "RCTUtils.h" +#import <React/RCTEventEmitter.h> +#import <React/RCTBridgeModule.h> +#import <React/RCTUtils.h> @interface FirestackCloudMessaging : RCTEventEmitter <RCTBridgeModule> { diff --git a/ios/Firestack/FirestackCloudMessaging.m b/ios/Firestack/FirestackCloudMessaging.m index ea8a346..ca0f1ca 100644 --- a/ios/Firestack/FirestackCloudMessaging.m +++ b/ios/Firestack/FirestackCloudMessaging.m @@ -71,7 +71,7 @@ + (void) setup:(UIApplication *) application } #pragma mark Request permissions -- (void) requestPermissions(NSDictionary *)requestedPermissions +- (void) requestPermissions:(NSDictionary *)requestedPermissions callback:(RCTResponseSenderBlock) callback { if (SYSTEM_VERSION_LESS_THAN_OR_EQUAL_TO(@"9.0")) { diff --git a/ios/Firestack/FirestackDatabase.h b/ios/Firestack/FirestackDatabase.h index bab8105..850a36a 100644 --- a/ios/Firestack/FirestackDatabase.h +++ b/ios/Firestack/FirestackDatabase.h @@ -10,16 +10,17 @@ #define FirestackDatabase_h #import "Firebase.h" -#import "RCTEventEmitter.h" -#import "RCTBridgeModule.h" +#import <React/RCTEventEmitter.h> +#import <React/RCTBridgeModule.h> @interface FirestackDatabase : RCTEventEmitter <RCTBridgeModule> { } -@property (nonatomic) NSDictionary *_DBHandles; -@property (nonatomic, weak) FIRDatabaseReference *ref; +@property NSMutableDictionary *dbReferences; +@property NSMutableDictionary *transactions; +@property dispatch_queue_t transactionQueue; @end -#endif \ No newline at end of file +#endif diff --git a/ios/Firestack/FirestackDatabase.m b/ios/Firestack/FirestackDatabase.m index fb684fc..edb15f3 100644 --- a/ios/Firestack/FirestackDatabase.m +++ b/ios/Firestack/FirestackDatabase.m @@ -11,49 +11,216 @@ #import "FirestackEvents.h" @interface FirestackDBReference : NSObject +@property RCTEventEmitter *emitter; +@property FIRDatabaseQuery *query; @property NSString *path; -@property NSDictionary *listeners; +@property NSString *modifiersString; +@property NSMutableDictionary *listeners; @property FIRDatabaseHandle childAddedHandler; @property FIRDatabaseHandle childModifiedHandler; @property FIRDatabaseHandle childRemovedHandler; @property FIRDatabaseHandle childMovedHandler; @property FIRDatabaseHandle childValueHandler; ++ (NSDictionary *) snapshotToDict:(FIRDataSnapshot *) snapshot; + @end @implementation FirestackDBReference -- (id) initWithPath:(NSString *) path +- (id) initWithPathAndModifiers:(RCTEventEmitter *) emitter + database:(FIRDatabase *) database + path:(NSString *) path + modifiers:(NSArray *) modifiers + modifiersString:(NSString *) modifiersString { self = [super init]; if (self) { - _path = path; - _listeners = [[NSDictionary alloc] init]; + _emitter = emitter; + _path = path; + _modifiersString = modifiersString; + _query = [self buildQueryAtPathWithModifiers:database path:path modifiers:modifiers]; + _listeners = [[NSMutableDictionary alloc] init]; } return self; } -- (FIRDatabaseReference *) getRef +- (NSString *) absPath:(FIRDatabaseReference *) ref { + NSString *url = ref.URL; + NSString *rooturl = ref.root.URL; + return [[url substringFromIndex:rooturl.length] stringByReplacingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; +} + +- (void) addEventHandler:(NSString *) eventName { - FIRDatabaseReference *rootRef = [[FIRDatabase database] reference]; - return [rootRef child:self.path]; + if (![self isListeningTo:eventName]) { + id withBlock = ^(FIRDataSnapshot * _Nonnull snapshot) { + NSDictionary *props = [FirestackDBReference snapshotToDict:snapshot]; + [self sendJSEvent:DATABASE_DATA_EVENT + title:eventName + props: @{ + @"eventName": eventName, + @"path": [self absPath:[snapshot ref]], + @"modifiersString": _modifiersString, + @"snapshot": props, + @"handlePath": _path + }]; + }; + id errorBlock = ^(NSError * _Nonnull error) { + NSLog(@"Error onDBEvent: %@", [error debugDescription]); + [self getAndSendDatabaseError:error + path:_path + modifiersString:_modifiersString]; + }; + int eventType = [self eventTypeFromName:eventName]; + FIRDatabaseHandle handle = [_query observeEventType:eventType + withBlock:withBlock + withCancelBlock:errorBlock]; + [self setEventHandler:handle forName:eventName]; + } else { + NSLog(@"Warning Trying to add duplicate listener for type: %@ with modifiers: %@ for path: %@", eventName, _modifiersString, _path); + } +} + +- (void) addSingleEventHandler:(RCTResponseSenderBlock) callback + ofType:(NSString *) type +{ + int eventType = [self eventTypeFromName:type]; + [_query observeSingleEventOfType:eventType + withBlock:^(FIRDataSnapshot * _Nonnull snapshot) { + NSDictionary *props = [FirestackDBReference snapshotToDict:snapshot]; + callback(@[[NSNull null], @{ + @"eventName": type, + @"path": [self absPath:[snapshot ref]], + @"modifiersString": _modifiersString, + @"snapshot": props + }]); + } + withCancelBlock:^(NSError * _Nonnull error) { + NSLog(@"Error onDBEventOnce: %@", [error debugDescription]); + callback(@[@{ + @"error": @"onceError", + @"msg": [error debugDescription] + }]); + }]; } -- (FIRDatabaseQuery *) getQueryWithModifiers:(NSArray *) modifiers +- (void) removeEventHandler:(NSString *) name { - FIRDatabaseReference *rootRef = [self getRef]; - FIRDatabaseQuery *query = [rootRef queryOrderedByKey]; + int eventType = [self eventTypeFromName:name]; + switch (eventType) { + case FIRDataEventTypeValue: + if (self.childValueHandler != nil) { + [_query removeObserverWithHandle:self.childValueHandler]; + self.childValueHandler = nil; + } + break; + case FIRDataEventTypeChildAdded: + if (self.childAddedHandler != nil) { + [_query removeObserverWithHandle:self.childAddedHandler]; + self.childAddedHandler = nil; + } + break; + case FIRDataEventTypeChildChanged: + if (self.childModifiedHandler != nil) { + [_query removeObserverWithHandle:self.childModifiedHandler]; + self.childModifiedHandler = nil; + } + break; + case FIRDataEventTypeChildRemoved: + if (self.childRemovedHandler != nil) { + [_query removeObserverWithHandle:self.childRemovedHandler]; + self.childRemovedHandler = nil; + } + break; + case FIRDataEventTypeChildMoved: + if (self.childMovedHandler != nil) { + [_query removeObserverWithHandle:self.childMovedHandler]; + self.childMovedHandler = nil; + } + break; + default: + break; + } + [self unsetListeningOn:name]; +} + ++ (NSDictionary *) snapshotToDict:(FIRDataSnapshot *) snapshot +{ + NSMutableDictionary *dict = [[NSMutableDictionary alloc] init]; + [dict setValue:snapshot.key forKey:@"key"]; + NSDictionary *val = snapshot.value; + [dict setObject:val forKey:@"value"]; + NSDictionary *exportValue = snapshot.valueInExportFormat; + [dict setObject:exportValue forKey:@"exportValue"]; + // Snapshot ordering + NSMutableArray *childKeys = [NSMutableArray array]; + if (snapshot.childrenCount > 0) { + // Since JS does not respect object ordering of keys + // we keep a list of the keys and their ordering + // in the snapshot event + NSEnumerator *children = [snapshot children]; + FIRDataSnapshot *child; + while(child = [children nextObject]) { + [childKeys addObject:child.key]; + } + } + [dict setObject:childKeys forKey:@"childKeys"]; + [dict setValue:@(snapshot.hasChildren) forKey:@"hasChildren"]; + [dict setValue:@(snapshot.exists) forKey:@"exists"]; + [dict setValue:@(snapshot.childrenCount) forKey:@"childrenCount"]; + [dict setValue:snapshot.priority forKey:@"priority"]; + return dict; +} + +- (NSDictionary *) getAndSendDatabaseError:(NSError *) error + path:(NSString *) path + modifiersString:(NSString *) modifiersString +{ + NSDictionary *evt = @{ + @"eventName": DATABASE_ERROR_EVENT, + @"path": path, + @"modifiersString": modifiersString, + @"msg": [error debugDescription] + }; + [self sendJSEvent:DATABASE_ERROR_EVENT title:DATABASE_ERROR_EVENT props: evt]; + return evt; +} + +- (void) sendJSEvent:(NSString *)type + title:(NSString *)title + props:(NSDictionary *)props +{ + @try { + [_emitter sendEventWithName:type + body:@{ + @"eventName": title, + @"body": props + }]; + } + @catch (NSException *err) { + NSLog(@"An error occurred in sendJSEvent: %@", [err debugDescription]); + NSLog(@"Tried to send: %@ with %@", title, props); + } +} + + +- (FIRDatabaseQuery *) buildQueryAtPathWithModifiers:(FIRDatabase*) database + path:(NSString*) path + modifiers:(NSArray *) modifiers +{ + FIRDatabaseQuery *query = [[database reference] child:path]; for (NSString *str in modifiers) { if ([str isEqualToString:@"orderByKey"]) { - query = [rootRef queryOrderedByKey]; + query = [query queryOrderedByKey]; } else if ([str isEqualToString:@"orderByPriority"]) { - query = [rootRef queryOrderedByPriority]; + query = [query queryOrderedByPriority]; } else if ([str isEqualToString:@"orderByValue"]) { - query = [rootRef queryOrderedByValue]; + query = [query queryOrderedByValue]; } else if ([str containsString:@"orderByChild"]) { NSArray *args = [str componentsSeparatedByString:@":"]; NSString *key = args[1]; - query = [rootRef queryOrderedByChild:key]; + query = [query queryOrderedByChild:key]; } else if ([str containsString:@"limitToLast"]) { NSArray *args = [str componentsSeparatedByString:@":"]; NSString *key = args[1]; @@ -67,43 +234,35 @@ - (FIRDatabaseQuery *) getQueryWithModifiers:(NSArray *) modifiers } else if ([str containsString:@"equalTo"]) { NSArray *args = [str componentsSeparatedByString:@":"]; int size = (int)[args count];; - - if (size > 2) { - NSString *value = args[1]; - NSString *key = args[2]; - - query = [query queryEqualToValue:value + id value = [self getIdValue:args[1] type:args[2]]; + if (size > 3) { + NSString *key = args[3]; + query = [query queryEqualToValue:value childKey:key]; } else { - NSString *value = args[1]; - query = [query queryEqualToValue:value]; + query = [query queryEqualToValue:value]; } } else if ([str containsString:@"endAt"]) { NSArray *args = [str componentsSeparatedByString:@":"]; int size = (int)[args count];; - - if (size > 2) { - NSString *value = args[1]; - NSString *key = args[2]; - - query = [query queryEndingAtValue:value + id value = [self getIdValue:args[1] type:args[2]]; + if (size > 3) { + NSString *key = args[3]; + query = [query queryEndingAtValue:value childKey:key]; } else { - NSString *value = args[1]; - query = [query queryEndingAtValue:value]; + query = [query queryEndingAtValue:value]; } } else if ([str containsString:@"startAt"]) { NSArray *args = [str componentsSeparatedByString:@":"]; + id value = [self getIdValue:args[1] type:args[2]]; int size = (int)[args count];; - if (size > 2) { - NSString *value = args[1]; - NSString *key = args[2]; - - query = [query queryStartingAtValue:value + if (size > 3) { + NSString *key = args[3]; + query = [query queryStartingAtValue:value childKey:key]; } else { - NSString *value = args[1]; - query = [query queryStartingAtValue:value]; + query = [query queryStartingAtValue:value]; } } } @@ -111,6 +270,18 @@ - (FIRDatabaseQuery *) getQueryWithModifiers:(NSArray *) modifiers return query; } +- (id) getIdValue:(NSString *) value + type:(NSString *) type +{ + if ([type isEqualToString:@"number"]) { + return [NSNumber numberWithInteger:value.integerValue]; + } else if ([type isEqualToString:@"boolean"]) { + return [NSNumber numberWithBool:value.boolValue]; + } else { + return value; + } +} + - (void) setEventHandler:(FIRDatabaseHandle) handle forName:(NSString *) name { @@ -137,52 +308,20 @@ - (void) setEventHandler:(FIRDatabaseHandle) handle [self setListeningOn:name withHandle:handle]; } -- (void) removeEventHandler:(NSString *) name -{ - FIRDatabaseReference *ref = [self getRef]; - int eventType = [self eventTypeFromName:name]; - - switch (eventType) { - case FIRDataEventTypeValue: - [ref removeObserverWithHandle:self.childValueHandler]; - break; - case FIRDataEventTypeChildAdded: - [ref removeObserverWithHandle:self.childAddedHandler]; - break; - case FIRDataEventTypeChildChanged: - [ref removeObserverWithHandle:self.childModifiedHandler]; - break; - case FIRDataEventTypeChildRemoved: - [ref removeObserverWithHandle:self.childRemovedHandler]; - break; - case FIRDataEventTypeChildMoved: - [ref removeObserverWithHandle:self.childMovedHandler]; - break; - default: - break; - } - [self unsetListeningOn:name]; -} - - (void) setListeningOn:(NSString *) name withHandle:(FIRDatabaseHandle) handle { - NSMutableDictionary *listeners = [_listeners mutableCopy]; - [listeners setValue:@(handle) forKey:name]; - _listeners = listeners; + [_listeners setValue:@(handle) forKey:name]; } - (void) unsetListeningOn:(NSString *) name { - NSMutableDictionary *listeners = [_listeners mutableCopy]; - [listeners removeObjectForKey:name]; - _listeners = listeners; + [_listeners removeObjectForKey:name]; } - (BOOL) isListeningTo:(NSString *) name { - id listener = [_listeners valueForKey:name]; - return listener != nil; + return [_listeners valueForKey:name] != nil; } - (BOOL) hasListeners @@ -237,47 +376,57 @@ @implementation FirestackDatabase RCT_EXPORT_MODULE(FirestackDatabase); +- (id) init +{ + self = [super init]; + if (self != nil) { + _dbReferences = [[NSMutableDictionary alloc] init]; + _transactions = [[NSMutableDictionary alloc] init]; + _transactionQueue = dispatch_queue_create("com.fullstackreact.react-native-firestack", DISPATCH_QUEUE_CONCURRENT); + } + return self; +} + RCT_EXPORT_METHOD(enablePersistence:(BOOL) enable callback:(RCTResponseSenderBlock) callback) { - BOOL isEnabled = [FIRDatabase database].persistenceEnabled; - if ( isEnabled != enable) { - [FIRDatabase database].persistenceEnabled = enable; - } + [FIRDatabase database].persistenceEnabled = enable; callback(@[[NSNull null], @{ - @"result": @"success" - }]); + @"result": @"success" + }]); } RCT_EXPORT_METHOD(keepSynced:(NSString *) path withEnable:(BOOL) enable callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; - [ref keepSynced:enable]; - callback(@[[NSNull null], @{ - @"result": @"success", - @"path": path - }]); + FIRDatabaseReference *ref = [self getPathRef:path]; + [ref keepSynced:enable]; + callback(@[[NSNull null], @{ + @"status": @"success", + @"path": path + }]); } RCT_EXPORT_METHOD(set:(NSString *) path - value:(NSDictionary *)value + data:(NSDictionary *)data callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; + FIRDatabaseReference *ref = [self getPathRef:path]; + [ref setValue:[data valueForKey:@"value"] withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { + [self handleCallback:@"set" callback:callback databaseError:error]; + }]; +} - [ref setValue:value withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } +RCT_EXPORT_METHOD(setWithPriority:(NSString *) path + data:(NSDictionary *)data + priority: (NSDictionary *)priority + callback:(RCTResponseSenderBlock) callback) +{ + FIRDatabaseReference *ref = [self getPathRef:path]; + [ref setValue:[data valueForKey:@"value"] andPriority:[priority valueForKey:@"value"] withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { + [self handleCallback:@"setWithPriority" callback:callback databaseError:error]; }]; } @@ -285,166 +434,188 @@ @implementation FirestackDatabase value:(NSDictionary *)value callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; - + FIRDatabaseReference *ref = [self getPathRef:path]; [ref updateChildValues:value withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } + [self handleCallback:@"update" callback:callback databaseError:error]; }]; } RCT_EXPORT_METHOD(remove:(NSString *) path callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; + FIRDatabaseReference *ref = [self getPathRef:path]; [ref removeValueWithCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } + [self handleCallback:@"remove" callback:callback databaseError:error]; }]; } RCT_EXPORT_METHOD(push:(NSString *) path - props:(NSDictionary *) props + data:(NSDictionary *) data callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [[self getRefAtPath:path] childByAutoId]; + FIRDatabaseReference *ref = [self getPathRef:path]; + FIRDatabaseReference *newRef = [ref childByAutoId]; - NSURL *url = [NSURL URLWithString:ref.URL]; + NSURL *url = [NSURL URLWithString:newRef.URL]; NSString *newPath = [url path]; - if ([props count] > 0) { - [ref setValue:props withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { + if ([data count] > 0) { + [newRef setValue:[data valueForKey:@"value"] withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { if (error != nil) { // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; + NSDictionary *evt = @{ + @"errorCode": [NSNumber numberWithInt:[error code]], + @"errorDetails": [error debugDescription], + @"description": [error description] + }; + callback(@[evt]); } else { callback(@[[NSNull null], @{ - @"result": @"success", + @"status": @"success", @"ref": newPath }]); } }]; } else { callback(@[[NSNull null], @{ - @"result": @"success", + @"status": @"success", @"ref": newPath }]); } } +RCT_EXPORT_METHOD(beginTransaction:(NSString *) path + withIdentifier:(NSString *) identifier + applyLocally:(BOOL) applyLocally + onComplete:(RCTResponseSenderBlock) onComplete) +{ + dispatch_async(_transactionQueue, ^{ + NSMutableDictionary *transactionState = [NSMutableDictionary new]; + + dispatch_semaphore_t sema = dispatch_semaphore_create(0); + [transactionState setObject:sema forKey:@"semaphore"]; + + FIRDatabaseReference *ref = [self getPathRef:path]; + [ref runTransactionBlock:^FIRTransactionResult * _Nonnull(FIRMutableData * _Nonnull currentData) { + dispatch_barrier_async(_transactionQueue, ^{ + [_transactions setValue:transactionState forKey:identifier]; + [self sendEventWithName:DATABASE_TRANSACTION_EVENT + body:@{ + @"id": identifier, + @"originalValue": currentData.value + }]; + }); + // Wait for the event handler to call tryCommitTransaction + // WARNING: This wait occurs on the Firebase Worker Queue + // so if tryCommitTransaction fails to signal the semaphore + // no further blocks will be executed by Firebase until the timeout expires + dispatch_time_t delayTime = dispatch_time(DISPATCH_TIME_NOW, 30 * NSEC_PER_SEC); + BOOL timedout = dispatch_semaphore_wait(sema, delayTime) != 0; + BOOL abort = [transactionState valueForKey:@"abort"] || timedout; + id value = [transactionState valueForKey:@"value"]; + dispatch_barrier_async(_transactionQueue, ^{ + [_transactions removeObjectForKey:identifier]; + }); + if (abort) { + return [FIRTransactionResult abort]; + } else { + currentData.value = value; + return [FIRTransactionResult successWithValue:currentData]; + } + } andCompletionBlock:^(NSError * _Nullable databaseError, BOOL committed, FIRDataSnapshot * _Nullable snapshot) { + if (databaseError != nil) { + NSDictionary *evt = @{ + @"errorCode": [NSNumber numberWithInt:[databaseError code]], + @"errorDetails": [databaseError debugDescription], + @"description": [databaseError description] + }; + onComplete(@[evt]); + } else { + onComplete(@[[NSNull null], @{ + @"committed": [NSNumber numberWithBool:committed], + @"snapshot": [FirestackDBReference snapshotToDict:snapshot], + @"status": @"success", + @"method": @"transaction" + }]); + } + } withLocalEvents:applyLocally]; + }); +} +RCT_EXPORT_METHOD(tryCommitTransaction:(NSString *) identifier + withData:(NSDictionary *) data + orAbort:(BOOL) abort) +{ + __block NSMutableDictionary *transactionState; + dispatch_sync(_transactionQueue, ^{ + transactionState = [_transactions objectForKey: identifier]; + }); + if (!transactionState) { + NSLog(@"tryCommitTransaction for unknown ID %@", identifier); + return; + } + dispatch_semaphore_t sema = [transactionState valueForKey:@"semaphore"]; + if (abort) { + [transactionState setValue:@true forKey:@"abort"]; + } else { + id newValue = [data valueForKey:@"value"]; + [transactionState setValue:newValue forKey:@"value"]; + } + dispatch_semaphore_signal(sema); +} RCT_EXPORT_METHOD(on:(NSString *) path + modifiersString:(NSString *) modifiersString modifiers:(NSArray *) modifiers name:(NSString *) eventName callback:(RCTResponseSenderBlock) callback) { - FirestackDBReference *r = [self getDBHandle:path]; - FIRDatabaseQuery *query = [r getQueryWithModifiers:modifiers]; - - if (![r isListeningTo:eventName]) { - id withBlock = ^(FIRDataSnapshot * _Nonnull snapshot) { - NSDictionary *props = - [self snapshotToDict:snapshot]; - [self - sendJSEvent:DATABASE_DATA_EVENT - title:eventName - props: @{ - @"eventName": eventName, - @"path": path, - @"snapshot": props - }]; - }; - - id errorBlock = ^(NSError * _Nonnull error) { - NSLog(@"Error onDBEvent: %@", [error debugDescription]); - [self getAndSendDatabaseError:error withPath: path]; - }; - - int eventType = [r eventTypeFromName:eventName]; - FIRDatabaseHandle handle = [query observeEventType:eventType - withBlock:withBlock - withCancelBlock:errorBlock]; - [r setEventHandler:handle - forName:eventName]; - - // [self saveDBHandle:path dbRef:r]; - - callback(@[[NSNull null], @{ - @"result": @"success", - @"handle": @(handle) - }]); - } else { - callback(@[@{ - @"result": @"exists", - @"msg": @"Listener already exists" + FirestackDBReference *ref = [self getDBHandle:path modifiers:modifiers modifiersString:modifiersString]; + [ref addEventHandler:eventName]; + callback(@[[NSNull null], @{ + @"status": @"success", + @"handle": path }]); - } } RCT_EXPORT_METHOD(onOnce:(NSString *) path - modifiers:(NSArray *) modifiers - name:(NSString *) name - callback:(RCTResponseSenderBlock) callback) + modifiersString:(NSString *) modifiersString + modifiers:(NSArray *) modifiers + name:(NSString *) name + callback:(RCTResponseSenderBlock) callback) { - FirestackDBReference *r = [self getDBHandle:path]; - int eventType = [r eventTypeFromName:name]; - FIRDatabaseQuery *ref = [r getQueryWithModifiers:modifiers]; - - [ref observeSingleEventOfType:eventType - withBlock:^(FIRDataSnapshot * _Nonnull snapshot) { - NSDictionary *props = [self snapshotToDict:snapshot]; - callback(@[[NSNull null], @{ - @"eventName": name, - @"path": path, - @"snapshot": props - }]); - } - withCancelBlock:^(NSError * _Nonnull error) { - NSLog(@"Error onDBEventOnce: %@", [error debugDescription]); - callback(@[@{ - @"error": @"onceError", - @"msg": [error debugDescription] - }]); - }]; + FirestackDBReference *ref = [self getDBHandle:path modifiers:modifiers modifiersString:modifiersString]; + [ref addSingleEventHandler:callback ofType:name]; } RCT_EXPORT_METHOD(off:(NSString *)path + modifiersString:(NSString *) modifiersString eventName:(NSString *) eventName callback:(RCTResponseSenderBlock) callback) { - FirestackDBReference *r = [self getDBHandle:path]; - if (eventName == nil || [eventName isEqualToString:@""]) { - [r cleanup]; - [self removeDBHandle:path]; + NSString *key = [self getDBListenerKey:path withModifiers:modifiersString]; + NSArray *listenerKeys; + FirestackDBReference *ref = [_dbReferences objectForKey:key]; + if (ref == nil) { + listenerKeys = @[]; } else { - [r removeEventHandler:eventName]; - if (![r hasListeners]) { - [self removeDBHandle:path]; + if (eventName == nil || [eventName isEqualToString:@""]) { + [ref cleanup]; + [_dbReferences removeObjectForKey:key]; + } else { + [ref removeEventHandler:eventName]; + if (![ref hasListeners]) { + [_dbReferences removeObjectForKey:key]; + } } + listenerKeys = [ref listenerKeys]; } - - // [self saveDBHandle:path dbRef:r]; - callback(@[[NSNull null], @{ @"result": @"success", - @"path": path, - @"remainingListeners": [r listenerKeys], + @"handle": path, + @"modifiersString": modifiersString, + @"remainingListeners": listenerKeys, }]); } @@ -453,19 +624,10 @@ @implementation FirestackDatabase props:(NSDictionary *) props callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; - + FIRDatabaseReference *ref = [self getPathRef:path]; [ref onDisconnectSetValue:props withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } + [self handleCallback:@"onDisconnectSetObject" callback:callback databaseError:error]; }]; } @@ -473,35 +635,19 @@ @implementation FirestackDatabase val:(NSString *) val callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; + FIRDatabaseReference *ref = [self getPathRef:path]; [ref onDisconnectSetValue:val withCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } + [self handleCallback:@"onDisconnectSetString" callback:callback databaseError:error]; }]; } RCT_EXPORT_METHOD(onDisconnectRemove:(NSString *) path callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; + FIRDatabaseReference *ref = [self getPathRef:path]; [ref onDisconnectRemoveValueWithCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } + [self handleCallback:@"onDisconnectRemove" callback:callback databaseError:error]; }]; } @@ -510,150 +656,74 @@ @implementation FirestackDatabase RCT_EXPORT_METHOD(onDisconnectCancel:(NSString *) path callback:(RCTResponseSenderBlock) callback) { - FIRDatabaseReference *ref = [self getRefAtPath:path]; + FIRDatabaseReference *ref = [self getPathRef:path]; [ref cancelDisconnectOperationsWithCompletionBlock:^(NSError * _Nullable error, FIRDatabaseReference * _Nonnull ref) { - if (error != nil) { - // Error handling - NSDictionary *evt = [self getAndSendDatabaseError:error withPath: path]; - callback(@[evt]); - } else { - callback(@[[NSNull null], @{ - @"result": @"success" - }]); - } + [self handleCallback:@"onDisconnectCancel" callback:callback databaseError:error]; }]; } - - - -// Helpers -- (FIRDatabaseReference *) getRef -{ - if (self.ref == nil) { - FIRDatabaseReference *rootRef = [[FIRDatabase database] reference]; - self.ref = rootRef; - } - return self.ref; -} - -- (FIRDatabaseReference *) getRefAtPath:(NSString *) str -{ - FirestackDBReference *r = [self getDBHandle:str]; - return [r getRef]; -} - -// Handles -- (NSDictionary *) storedDBHandles +RCT_EXPORT_METHOD(goOffline) { - if (__DBHandles == nil) { - __DBHandles = [[NSDictionary alloc] init]; - } - return __DBHandles; + [FIRDatabase database].goOffline; } -- (FirestackDBReference *) getDBHandle:(NSString *) path +RCT_EXPORT_METHOD(goOnline) { - NSDictionary *stored = [self storedDBHandles]; - FirestackDBReference *r = [stored objectForKey:path]; - - if (r == nil) { - r = [[FirestackDBReference alloc] initWithPath:path]; - [self saveDBHandle:path dbRef:r]; - } - return r; + [FIRDatabase database].goOnline; } -- (void) saveDBHandle:(NSString *) path - dbRef:(FirestackDBReference *) dbRef +- (FIRDatabaseReference *) getPathRef:(NSString *) path { - NSMutableDictionary *stored = [[self storedDBHandles] mutableCopy]; - if ([stored objectForKey:path]) { - FirestackDBReference *r = [stored objectForKey:path]; - [r cleanup]; - } - - [stored setObject:dbRef forKey:path]; - self._DBHandles = stored; + return [[[FIRDatabase database] reference] child:path]; } -- (void) removeDBHandle:(NSString *) path +- (void) handleCallback:(NSString *) methodName + callback:(RCTResponseSenderBlock) callback + databaseError:(NSError *) databaseError { - NSMutableDictionary *stored = [[self storedDBHandles] mutableCopy]; - - FirestackDBReference *r = [stored objectForKey:path]; - if (r != nil) { - [r cleanup]; + if (databaseError != nil) { + NSDictionary *evt = @{ + @"errorCode": [NSNumber numberWithInt:[databaseError code]], + @"errorDetails": [databaseError debugDescription], + @"description": [databaseError description] + }; + callback(@[evt]); + } else { + callback(@[[NSNull null], @{ + @"status": @"success", + @"method": methodName + }]); } - [stored removeObjectForKey:path]; - self._DBHandles = [stored copy]; } -- (NSDictionary *) snapshotToDict:(FIRDataSnapshot *) snapshot -{ - NSMutableDictionary *dict = [[NSMutableDictionary alloc] init]; - [dict setValue:snapshot.key forKey:@"key"]; - NSDictionary *val = snapshot.value; - [dict setObject:val forKey:@"value"]; - - // Snapshot ordering - NSMutableArray *childKeys = [NSMutableArray array]; - if (snapshot.childrenCount > 0) { - // Since JS does not respect object ordering of keys - // we keep a list of the keys and their ordering - // in the snapshot event - NSEnumerator *children = [snapshot children]; - FIRDataSnapshot *child; - while(child = [children nextObject]) { - [childKeys addObject:child.key]; - } +- (FirestackDBReference *) getDBHandle:(NSString *) path + modifiers:modifiers + modifiersString:modifiersString +{ + NSString *key = [self getDBListenerKey:path withModifiers:modifiersString]; + FirestackDBReference *ref = [_dbReferences objectForKey:key]; + + if (ref == nil) { + ref = [[FirestackDBReference alloc] initWithPathAndModifiers:self + database:[FIRDatabase database] + path:path + modifiers:modifiers + modifiersString:modifiersString]; + [_dbReferences setObject:ref forKey:key]; } - - [dict setObject:childKeys forKey:@"childKeys"]; - [dict setValue:@(snapshot.hasChildren) forKey:@"hasChildren"]; - [dict setValue:@(snapshot.exists) forKey:@"exists"]; - [dict setValue:@(snapshot.childrenCount) forKey:@"childrenCount"]; - [dict setValue:snapshot.priority forKey:@"priority"]; - - return dict; + return ref; } -- (NSDictionary *) getAndSendDatabaseError:(NSError *) error - withPath:(NSString *) path +- (NSString *) getDBListenerKey:(NSString *) path + withModifiers:(NSString *) modifiersString { - NSDictionary *evt = @{ - @"eventName": DATABASE_ERROR_EVENT, - @"path": path, - @"msg": [error debugDescription] - }; - [self - sendJSEvent:DATABASE_ERROR_EVENT - title:DATABASE_ERROR_EVENT - props: evt]; - - return evt; + return [NSString stringWithFormat:@"%@ | %@", path, modifiersString, nil]; } // Not sure how to get away from this... yet - (NSArray<NSString *> *)supportedEvents { - return @[DATABASE_DATA_EVENT, DATABASE_ERROR_EVENT]; + return @[DATABASE_DATA_EVENT, DATABASE_ERROR_EVENT, DATABASE_TRANSACTION_EVENT]; } -- (void) sendJSEvent:(NSString *)type - title:(NSString *)title - props:(NSDictionary *)props -{ - @try { - [self sendEventWithName:type - body:@{ - @"eventName": title, - @"body": props - }]; - } - @catch (NSException *err) { - NSLog(@"An error occurred in sendJSEvent: %@", [err debugDescription]); - NSLog(@"Tried to send: %@ with %@", title, props); - } -} -@end \ No newline at end of file +@end diff --git a/ios/Firestack/FirestackErrors.h b/ios/Firestack/FirestackErrors.h index 5606baf..2711cf4 100644 --- a/ios/Firestack/FirestackErrors.h +++ b/ios/Firestack/FirestackErrors.h @@ -9,7 +9,7 @@ #ifndef FirestackErrors_h #define FirestackErrors_h -#import "RCTBridgeModule.h" +#import <React/RCTBridgeModule.h> #import "Firebase.h" @interface FirestackErrors : NSObject <RCTBridgeModule> { diff --git a/ios/Firestack/FirestackEvents.h b/ios/Firestack/FirestackEvents.h index ba976ef..68ead8b 100644 --- a/ios/Firestack/FirestackEvents.h +++ b/ios/Firestack/FirestackEvents.h @@ -29,6 +29,7 @@ static NSString *const DEBUG_EVENT = @"debug"; // Database static NSString *const DATABASE_DATA_EVENT = @"database_event"; static NSString *const DATABASE_ERROR_EVENT = @"database_error"; +static NSString *const DATABASE_TRANSACTION_EVENT = @"database_transaction_update"; static NSString *const DATABASE_VALUE_EVENT = @"value"; static NSString *const DATABASE_CHILD_ADDED_EVENT = @"child_added"; @@ -37,12 +38,14 @@ static NSString *const DATABASE_CHILD_REMOVED_EVENT = @"child_removed"; static NSString *const DATABASE_CHILD_MOVED_EVENT = @"child_moved"; // Storage -static NSString *const STORAGE_UPLOAD_PROGRESS = @"upload_progress"; -static NSString *const STORAGE_UPLOAD_PAUSED = @"upload_paused"; -static NSString *const STORAGE_UPLOAD_RESUMED = @"upload_resumed"; -static NSString *const STORAGE_DOWNLOAD_PROGRESS = @"download_progress"; -static NSString *const STORAGE_DOWNLOAD_PAUSED = @"download_paused"; -static NSString *const STORAGE_DOWNLOAD_RESUMED = @"download_resumed"; +static NSString *const STORAGE_EVENT = @"storage_event"; +static NSString *const STORAGE_ERROR = @"storage_error"; + +static NSString *const STORAGE_STATE_CHANGED = @"state_changed"; +static NSString *const STORAGE_UPLOAD_SUCCESS = @"upload_success"; +static NSString *const STORAGE_UPLOAD_FAILURE = @"upload_failure"; +static NSString *const STORAGE_DOWNLOAD_SUCCESS = @"download_success"; +static NSString *const STORAGE_DOWNLOAD_FAILURE = @"download_failure"; // Messaging static NSString *const MESSAGING_SUBSYSTEM_EVENT = @"messaging_event"; diff --git a/ios/Firestack/FirestackStorage.h b/ios/Firestack/FirestackStorage.h index 04e42ed..7aa218f 100644 --- a/ios/Firestack/FirestackStorage.h +++ b/ios/Firestack/FirestackStorage.h @@ -10,8 +10,8 @@ #define FirestackStorage_h #import "Firebase.h" -#import "RCTBridgeModule.h" -#import "RCTEventEmitter.h" +#import <React/RCTBridgeModule.h> +#import <React/RCTEventEmitter.h> @interface FirestackStorage : RCTEventEmitter <RCTBridgeModule> { diff --git a/ios/Firestack/FirestackStorage.m b/ios/Firestack/FirestackStorage.m index 2da5377..b2e9cd8 100644 --- a/ios/Firestack/FirestackStorage.m +++ b/ios/Firestack/FirestackStorage.m @@ -21,124 +21,213 @@ - (dispatch_queue_t)methodQueue return dispatch_queue_create("io.fullstack.firestack.storage", DISPATCH_QUEUE_SERIAL); } -RCT_EXPORT_METHOD(downloadUrl: (NSString *) storageUrl - path:(NSString *) path - callback:(RCTResponseSenderBlock) callback) +RCT_EXPORT_METHOD(delete: (NSString *) path + callback:(RCTResponseSenderBlock) callback) { - FIRStorageReference *storageRef; - if (storageUrl == nil ) { - storageRef = [[FIRStorage storage] reference]; - } else { - storageRef = [[FIRStorage storage] referenceForURL:storageUrl]; - } - FIRStorageReference *fileRef = [storageRef child:path]; + + FIRStorageReference *fileRef = [self getReference:path]; + [fileRef deleteWithCompletion:^(NSError * _Nullable error) { + if (error == nil) { + NSDictionary *resp = @{ + @"status": @"success", + @"path": path + }; + callback(@[[NSNull null], resp]); + } else { + NSDictionary *evt = @{ + @"status": @"error", + @"path": path, + @"message": [error debugDescription] + }; + callback(@[evt]); + } + }]; +} + +RCT_EXPORT_METHOD(getDownloadURL: (NSString *) path + callback:(RCTResponseSenderBlock) callback) +{ + FIRStorageReference *fileRef = [self getReference:path]; [fileRef downloadURLWithCompletion:^(NSURL * _Nullable URL, NSError * _Nullable error) { if (error != nil) { NSDictionary *evt = @{ @"status": @"error", @"path": path, - @"msg": [error debugDescription] + @"message": [error debugDescription] }; callback(@[evt]); } else { - NSDictionary *resp = @{ - @"status": @"success", - @"url": [URL absoluteString], - @"path": [URL path] - }; + callback(@[[NSNull null], [URL absoluteString]]); + } + }]; +} + +RCT_EXPORT_METHOD(getMetadata: (NSString *) path + callback:(RCTResponseSenderBlock) callback) +{ + FIRStorageReference *fileRef = [self getReference:path]; + [fileRef metadataWithCompletion:^(FIRStorageMetadata * _Nullable metadata, NSError * _Nullable error) { + if (error != nil) { + NSDictionary *evt = @{ + @"status": @"error", + @"path": path, + @"message": [error debugDescription] + }; + callback(@[evt]); + } else { + NSDictionary *resp = [metadata dictionaryRepresentation]; callback(@[[NSNull null], resp]); } }]; } -RCT_EXPORT_METHOD(uploadFile: (NSString *) urlStr - name: (NSString *) name - path:(NSString *)path - metadata:(NSDictionary *)metadata +RCT_EXPORT_METHOD(updateMetadata: (NSString *) path + metadata:(NSDictionary *) metadata + callback:(RCTResponseSenderBlock) callback) +{ + FIRStorageReference *fileRef = [self getReference:path]; + FIRStorageMetadata *firmetadata = [[FIRStorageMetadata alloc] initWithDictionary:metadata]; + [fileRef updateMetadata:firmetadata completion:^(FIRStorageMetadata * _Nullable metadata, NSError * _Nullable error) { + if (error != nil) { + NSDictionary *evt = @{ + @"status": @"error", + @"path": path, + @"message": [error debugDescription] + }; + callback(@[evt]); + } else { + NSDictionary *resp = [metadata dictionaryRepresentation]; + callback(@[[NSNull null], resp]); + } + }]; +} + +RCT_EXPORT_METHOD(downloadFile: (NSString *) path + localPath:(NSString *) localPath callback:(RCTResponseSenderBlock) callback) { - FIRStorageReference *storageRef; - if (urlStr == nil) { - storageRef = [[FIRStorage storage] reference]; - } else { - storageRef = [[FIRStorage storage] referenceForURL:urlStr]; - } + FIRStorageReference *fileRef = [self getReference:path]; + NSURL *localFile = [NSURL fileURLWithPath:localPath]; + + FIRStorageDownloadTask *downloadTask = [fileRef writeToFile:localFile]; + // Listen for state changes, errors, and completion of the download. + [downloadTask observeStatus:FIRStorageTaskStatusResume handler:^(FIRStorageTaskSnapshot *snapshot) { + // Download resumed, also fires when the upload starts + NSDictionary *event = [self getDownloadTaskAsDictionary:snapshot]; + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_STATE_CHANGED props:event]; + }]; + + [downloadTask observeStatus:FIRStorageTaskStatusPause handler:^(FIRStorageTaskSnapshot *snapshot) { + // Download paused + NSDictionary *event = [self getDownloadTaskAsDictionary:snapshot]; + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_STATE_CHANGED props:event]; + }]; + [downloadTask observeStatus:FIRStorageTaskStatusProgress handler:^(FIRStorageTaskSnapshot *snapshot) { + // Download reported progress + NSDictionary *event = [self getDownloadTaskAsDictionary:snapshot]; + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_STATE_CHANGED props:event]; + }]; + + [downloadTask observeStatus:FIRStorageTaskStatusSuccess handler:^(FIRStorageTaskSnapshot *snapshot) { + // Download completed successfully + NSDictionary *resp = [self getDownloadTaskAsDictionary:snapshot]; + + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_DOWNLOAD_SUCCESS props:resp]; + callback(@[[NSNull null], resp]); + }]; + + [downloadTask observeStatus:FIRStorageTaskStatusFailure handler:^(FIRStorageTaskSnapshot *snapshot) { + if (snapshot.error != nil) { + NSDictionary *errProps = [[NSMutableDictionary alloc] init]; + NSLog(@"Error in download: %@", snapshot.error); + + switch (snapshot.error.code) { + case FIRStorageErrorCodeObjectNotFound: + // File doesn't exist + [errProps setValue:@"File does not exist" forKey:@"message"]; + break; + case FIRStorageErrorCodeUnauthorized: + // User doesn't have permission to access file + [errProps setValue:@"You do not have permissions" forKey:@"message"]; + break; + case FIRStorageErrorCodeCancelled: + // User canceled the upload + [errProps setValue:@"Download canceled" forKey:@"message"]; + break; + case FIRStorageErrorCodeUnknown: + // Unknown error occurred, inspect the server response + [errProps setValue:@"Unknown error" forKey:@"message"]; + break; + } + + //TODO: Error event + callback(@[errProps]); + }}]; +} - FIRStorageReference *uploadRef = [storageRef child:name]; + +RCT_EXPORT_METHOD(putFile:(NSString *) path + localPath:(NSString *)localPath + metadata:(NSDictionary *)metadata + callback:(RCTResponseSenderBlock) callback) +{ + FIRStorageReference *fileRef = [self getReference:path]; FIRStorageMetadata *firmetadata = [[FIRStorageMetadata alloc] initWithDictionary:metadata]; - if ([path hasPrefix:@"assets-library://"]) { - NSURL *localFile = [[NSURL alloc] initWithString:path]; + if ([localPath hasPrefix:@"assets-library://"]) { + NSURL *localFile = [[NSURL alloc] initWithString:localPath]; PHFetchResult* assets = [PHAsset fetchAssetsWithALAssetURLs:@[localFile] options:nil]; PHAsset *asset = [assets firstObject]; [[PHImageManager defaultManager] requestImageDataForAsset:asset - options:nil + options:nil resultHandler:^(NSData * imageData, NSString * dataUTI, UIImageOrientation orientation, NSDictionary * info) { - FIRStorageUploadTask *uploadTask = [uploadRef putData:imageData - metadata:firmetadata]; + FIRStorageUploadTask *uploadTask = [fileRef putData:imageData + metadata:firmetadata]; [self addUploadObservers:uploadTask + path:path callback:callback]; }]; } else { - NSURL *imageFile = [NSURL fileURLWithPath:path]; - FIRStorageUploadTask *uploadTask = [uploadRef putFile:imageFile - metadata:firmetadata]; + NSURL *imageFile = [NSURL fileURLWithPath:localPath]; + FIRStorageUploadTask *uploadTask = [fileRef putFile:imageFile + metadata:firmetadata]; [self addUploadObservers:uploadTask + path:path callback:callback]; } } - (void) addUploadObservers:(FIRStorageUploadTask *) uploadTask + path:(NSString *) path callback:(RCTResponseSenderBlock) callback { // Listen for state changes, errors, and completion of the upload. [uploadTask observeStatus:FIRStorageTaskStatusResume handler:^(FIRStorageTaskSnapshot *snapshot) { // Upload resumed, also fires when the upload starts - [self sendJSEvent:STORAGE_UPLOAD_RESUMED props:@{ - @"eventName": STORAGE_UPLOAD_RESUMED, - @"ref": snapshot.reference.bucket - }]; + NSDictionary *event = [self getUploadTaskAsDictionary:snapshot]; + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_STATE_CHANGED props:event]; }]; [uploadTask observeStatus:FIRStorageTaskStatusPause handler:^(FIRStorageTaskSnapshot *snapshot) { // Upload paused - [self sendJSEvent:STORAGE_UPLOAD_PAUSED props:@{ - @"eventName": STORAGE_UPLOAD_PAUSED, - @"ref": snapshot.reference.bucket - }]; + NSDictionary *event = [self getUploadTaskAsDictionary:snapshot]; + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_STATE_CHANGED props:event]; }]; [uploadTask observeStatus:FIRStorageTaskStatusProgress handler:^(FIRStorageTaskSnapshot *snapshot) { // Upload reported progress - float percentComplete; - if (snapshot.progress.totalUnitCount == 0) { - percentComplete = 0.0; - } else { - percentComplete = 100.0 * (snapshot.progress.completedUnitCount) / (snapshot.progress.totalUnitCount); - } - - [self sendJSEvent:STORAGE_UPLOAD_PROGRESS props:@{ - @"eventName": STORAGE_UPLOAD_PROGRESS, - @"progress": @(percentComplete) - }]; - + NSDictionary *event = [self getUploadTaskAsDictionary:snapshot]; + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_STATE_CHANGED props:event]; }]; [uploadTask observeStatus:FIRStorageTaskStatusSuccess handler:^(FIRStorageTaskSnapshot *snapshot) { - [uploadTask removeAllObservers]; - // Upload completed successfully - FIRStorageReference *ref = snapshot.reference; - NSDictionary *props = @{ - @"fullPath": ref.fullPath, - @"bucket": ref.bucket, - @"name": ref.name, - @"metadata": [snapshot.metadata dictionaryRepresentation] - }; - - callback(@[[NSNull null], props]); + NSDictionary *resp = [self getUploadTaskAsDictionary:snapshot]; + + [self sendJSEvent:STORAGE_EVENT path:path title:STORAGE_UPLOAD_SUCCESS props:resp]; + callback(@[[NSNull null], resp]); }]; [uploadTask observeStatus:FIRStorageTaskStatusFailure handler:^(FIRStorageTaskSnapshot *snapshot) { @@ -148,125 +237,89 @@ - (void) addUploadObservers:(FIRStorageUploadTask *) uploadTask switch (snapshot.error.code) { case FIRStorageErrorCodeObjectNotFound: // File doesn't exist - [errProps setValue:@"File does not exist" forKey:@"description"]; + [errProps setValue:@"File does not exist" forKey:@"message"]; break; case FIRStorageErrorCodeUnauthorized: // User doesn't have permission to access file - [errProps setValue:@"You do not have permissions" forKey:@"description"]; + [errProps setValue:@"You do not have permissions" forKey:@"message"]; break; case FIRStorageErrorCodeCancelled: // User canceled the upload - [errProps setValue:@"Upload cancelled" forKey:@"description"]; + [errProps setValue:@"Upload cancelled" forKey:@"message"]; break; case FIRStorageErrorCodeUnknown: // Unknown error occurred, inspect the server response - [errProps setValue:@"Unknown error" forKey:@"description"]; - NSLog(@"Unknown error: %@", snapshot.error); + [errProps setValue:@"Unknown error" forKey:@"message"]; break; } + //TODO: Error event callback(@[errProps]); }}]; } -RCT_EXPORT_METHOD(downloadFile: (NSString *) urlStr - path:(NSString *) path - localFile:(NSString *) file - callback:(RCTResponseSenderBlock) callback) +//Firebase.Storage methods +RCT_EXPORT_METHOD(setMaxDownloadRetryTime:(NSNumber *) milliseconds) { - if (urlStr == nil) { - NSError *err = [[NSError alloc] init]; - [err setValue:@"Storage configuration error" forKey:@"name"]; - [err setValue:@"Call setStorageUrl() first" forKey:@"description"]; - return callback(@[err]); - } - - FIRStorageReference *storageRef = [[FIRStorage storage] referenceForURL:urlStr]; - FIRStorageReference *fileRef = [storageRef child:path]; - - NSURL *localFile = [NSURL fileURLWithPath:file]; - - FIRStorageDownloadTask *downloadTask = [fileRef writeToFile:localFile]; - // Listen for state changes, errors, and completion of the download. - [downloadTask observeStatus:FIRStorageTaskStatusResume handler:^(FIRStorageTaskSnapshot *snapshot) { - // Upload resumed, also fires when the upload starts - [self sendJSEvent:STORAGE_DOWNLOAD_RESUMED props:@{ - @"eventName": STORAGE_DOWNLOAD_RESUMED, - @"ref": snapshot.reference.bucket - }]; - }]; - - [downloadTask observeStatus:FIRStorageTaskStatusPause handler:^(FIRStorageTaskSnapshot *snapshot) { - // Upload paused - [self sendJSEvent:STORAGE_DOWNLOAD_PAUSED props:@{ - @"eventName": STORAGE_DOWNLOAD_PAUSED, - @"ref": snapshot.reference.bucket - }]; - }]; - [downloadTask observeStatus:FIRStorageTaskStatusProgress handler:^(FIRStorageTaskSnapshot *snapshot) { - // Upload reported progress - float percentComplete; - if (snapshot.progress.totalUnitCount == 0) { - percentComplete = 0.0; - } else { - percentComplete = 100.0 * (snapshot.progress.completedUnitCount) / (snapshot.progress.totalUnitCount); - } - - [self sendJSEvent:STORAGE_DOWNLOAD_PROGRESS props:@{ - @"eventName": STORAGE_DOWNLOAD_PROGRESS, - @"progress": @(percentComplete) - }]; - - }]; + [[FIRStorage storage] setMaxDownloadRetryTime:[milliseconds doubleValue]]; +} - [downloadTask observeStatus:FIRStorageTaskStatusSuccess handler:^(FIRStorageTaskSnapshot *snapshot) { - [downloadTask removeAllObservers]; +RCT_EXPORT_METHOD(setMaxOperationRetryTime:(NSNumber *) milliseconds) +{ + [[FIRStorage storage] setMaxOperationRetryTime:[milliseconds doubleValue]]; +} - // Upload completed successfully - FIRStorageReference *ref = snapshot.reference; - NSDictionary *props = @{ - @"fullPath": ref.fullPath, - @"bucket": ref.bucket, - @"name": ref.name - }; - - callback(@[[NSNull null], props]); - }]; +RCT_EXPORT_METHOD(setMaxUploadRetryTime:(NSNumber *) milliseconds) +{ + [[FIRStorage storage] setMaxUploadRetryTime:[milliseconds doubleValue]]; +} - [downloadTask observeStatus:FIRStorageTaskStatusFailure handler:^(FIRStorageTaskSnapshot *snapshot) { - if (snapshot.error != nil) { - NSDictionary *errProps = [[NSMutableDictionary alloc] init]; - NSLog(@"Error in download: %@", snapshot.error); +- (FIRStorageReference *)getReference:(NSString *)path +{ + if ([path hasPrefix:@"url::"]) { + NSString *url = [path substringFromIndex:5]; + return [[FIRStorage storage] referenceForURL:url]; + } else { + return [[FIRStorage storage] referenceWithPath:path]; + } +} - switch (snapshot.error.code) { - case FIRStorageErrorCodeObjectNotFound: - // File doesn't exist - [errProps setValue:@"File does not exist" forKey:@"description"]; - break; - case FIRStorageErrorCodeUnauthorized: - // User doesn't have permission to access file - [errProps setValue:@"You do not have permissions" forKey:@"description"]; - break; - case FIRStorageErrorCodeCancelled: - // User canceled the upload - [errProps setValue:@"Download canceled" forKey:@"description"]; - break; - case FIRStorageErrorCodeUnknown: - // Unknown error occurred, inspect the server response - [errProps setValue:@"Unknown error" forKey:@"description"]; - break; - } +- (NSDictionary *)getDownloadTaskAsDictionary:(FIRStorageTaskSnapshot *)task { + return @{ + @"bytesTransferred": @(task.progress.completedUnitCount), + @"ref": task.reference.fullPath, + @"status": [self getTaskStatus:task.status], + @"totalBytes": @(task.progress.totalUnitCount) + }; +} - callback(@[errProps]); - }}]; +- (NSDictionary *)getUploadTaskAsDictionary:(FIRStorageTaskSnapshot *)task +{ + NSString *downloadUrl = [task.metadata.downloadURL absoluteString]; + FIRStorageMetadata *metadata = [task.metadata dictionaryRepresentation]; + return @{ + @"bytesTransferred": @(task.progress.completedUnitCount), + @"downloadUrl": downloadUrl != nil ? downloadUrl : [NSNull null], + @"metadata": metadata != nil ? metadata : [NSNull null], + @"ref": task.reference.fullPath, + @"state": [self getTaskStatus:task.status], + @"totalBytes": @(task.progress.totalUnitCount) + }; } -// Compatibility with the android library -// For now, just passes the url path back -RCT_EXPORT_METHOD(getRealPathFromURI: (NSString *) urlStr - callback:(RCTResponseSenderBlock) callback) +- (NSString *)getTaskStatus:(FIRStorageTaskStatus)status { - callback(@[[NSNull null], urlStr]); + if (status == FIRStorageTaskStatusResume || status == FIRStorageTaskStatusProgress) { + return @"RUNNING"; + } else if (status == FIRStorageTaskStatusPause) { + return @"PAUSED"; + } else if (status == FIRStorageTaskStatusSuccess) { + return @"SUCCESS"; + } else if (status == FIRStorageTaskStatusFailure) { + return @"ERROR"; + } else { + return @"UNKNOWN"; + } } // This is just too good not to use, but I don't want to take credit for @@ -295,25 +348,36 @@ - (NSDictionary *)constantsToExport // Not sure how to get away from this... yet - (NSArray<NSString *> *)supportedEvents { - return @[ - STORAGE_UPLOAD_PAUSED, - STORAGE_UPLOAD_RESUMED, - STORAGE_UPLOAD_PROGRESS, - STORAGE_DOWNLOAD_PAUSED, - STORAGE_DOWNLOAD_RESUMED, - STORAGE_DOWNLOAD_PROGRESS - ]; + return @[STORAGE_EVENT, STORAGE_ERROR]; } -- (void) sendJSEvent:(NSString *)title +- (void) sendJSError:(NSError *) error + withPath:(NSString *) path +{ + NSDictionary *evt = @{ + @"path": path, + @"message": [error debugDescription] + }; + [self sendJSEvent:STORAGE_ERROR path:path title:STORAGE_ERROR props: evt]; +} + +- (void) sendJSEvent:(NSString *)type + path:(NSString *)path + title:(NSString *)title props:(NSDictionary *)props { @try { - [self sendEventWithName:title - body:props]; + [self sendEventWithName:type + body:@{ + @"eventName": title, + @"path": path, + @"body": props + }]; + } @catch (NSException *err) { NSLog(@"An error occurred in sendJSEvent: %@", [err debugDescription]); + NSLog(@"Tried to send: %@ with %@", title, props); } } diff --git a/lib/constants.js b/lib/constants.js new file mode 100644 index 0000000..83efd38 --- /dev/null +++ b/lib/constants.js @@ -0,0 +1,27 @@ +import { reverseKeyValues } from './utils'; + +export const ConnectionResult = { + SUCCESS: 0, + SERVICE_MISSING: 1, + SERVICE_VERSION_UPDATE_REQUIRED: 2, + SERVICE_DISABLED: 3, + SIGN_IN_REQUIRED: 4, + INVALID_ACCOUNT: 5, + RESOLUTION_REQUIRED: 6, + NETWORK_ERROR: 7, + INTERNAL_ERROR: 8, + SERVICE_INVALID: 9, + DEVELOPER_ERROR: 10, + LICENSE_CHECK_FAILED: 11, + CANCELED: 13, + TIMEOUT: 14, + INTERRUPTED: 15, + API_UNAVAILABLE: 16, + SIGN_IN_FAILED: 17, + SERVICE_UPDATING: 18, + SERVICE_MISSING_PERMISSION: 19, + RESTRICTED_PROFILE: 20, +}; + + +export const ConnectionResultReverse = reverseKeyValues(ConnectionResult); diff --git a/lib/firestack.js b/lib/firestack.js index 8bcbbfa..cfe2f83 100644 --- a/lib/firestack.js +++ b/lib/firestack.js @@ -2,38 +2,46 @@ * @providesModule Firestack * @flow */ -import Log from './utils/log' +import { NativeModules, NativeEventEmitter } from 'react-native'; -// const firebase = require('firebase'); +import Log from './utils/log'; +import { promisify } from './utils'; +import Singleton from './utils/singleton'; -// const app = require('firebase/app'); -// const storage = require('firebase/storage'); -// const db = require('firebase/database'); +// modules +import Auth from './modules/auth'; +import Storage from './modules/storage'; +import Database from './modules/database'; +import Presence from './modules/presence'; +import Messaging from './modules/messaging'; +import Analytics from './modules/analytics'; +import RemoteConfig from './modules/remoteConfig'; -import {NativeModules, NativeEventEmitter, AsyncStorage} from 'react-native'; -// TODO: Break out modules into component pieces -// i.e. auth component, storage component, etc. +let log; +const instances = { default: null }; const FirestackModule = NativeModules.Firestack; const FirestackModuleEvt = new NativeEventEmitter(FirestackModule); -import promisify from './utils/promisify' -import Singleton from './utils/singleton' - -import RemoteConfig from './modules/remoteConfig' -import {Authentication} from './modules/authentication' -import {Database} from './modules/database' -import {Analytics} from './modules/analytics' -import {Storage} from './modules/storage' -import {Presence} from './modules/presence' -import {CloudMessaging} from './modules/cloudmessaging' +type GoogleApiAvailabilityType = { + status: number, + isAvailable: boolean, + isUserResolvableError?: boolean, + error?: string +}; -let log; -export class Firestack extends Singleton { +/** + * @class Firestack + */ +export default class Firestack extends Singleton { - constructor(options) { - var instance = super(options); + /** + * + * @param options + */ + constructor(options: Object = {}) { + const instance = super(options); - instance.options = options || {}; + instance.options = Object.assign({ errorOnMissingPlayServices: true }, options); instance._debug = instance.options.debug || false; Log.enable(instance._debug); @@ -45,20 +53,59 @@ export class Firestack extends Singleton { delete instance.options.remoteConfig; instance.configured = instance.options.configure || false; - instance.auth = null; instance.eventHandlers = {}; log.info('Calling configure with options', instance.options); instance.configurePromise = instance.configure(instance.options); - instance._auth = new Authentication(instance, instance.options); + instance._auth = new Auth(instance, instance.options); + + if (instance.options.errorOnMissingPlayServices && !this.googleApiAvailability.isAvailable) { + throw new Error(`Google Play Services is required to run this application but no valid installation was found (Code ${this.googleApiAvailability.status}).`); + } + + const database = { + ServerValue: { + TIMESTAMP: FirestackModule.serverValueTimestamp || { '.sv': 'timestamp' }, + } + }; + database.__proto__ = instance.database.__proto__; + instance.database.__proto__ = database; + } + + _db: ?Object; + _log: ?Object; + _auth: ?Object; + _store: ?Object; + _storage: ?Object; + _presence: ?Object; + _analytics: ?Object; + _constants: ?Object; + _messaging: ?Object; + _remoteConfig: ?Object; + + /** + * Support web version of initApp. + * @param options + * @param name + * @returns {*} + */ + static initializeApp(options: Object = {}, name: string = 'default') { + if (!instances[name]) instances[name] = new Firestack(options); + return instances[name]; } - configure(opts = {}) { + + /** + * + * @param opts + * @returns {Promise.<TResult>|*|Promise.<T>} + */ + configure(opts: Object = {}) { if (!this.configurePromise) { const firestackOptions = Object.assign({}, this.options, opts); - + this.configurePromise = promisify('configureWithOptions', FirestackModule)(firestackOptions) .then((configuredProperties) => { log.info('Native configureWithOptions success', configuredProperties); @@ -67,12 +114,13 @@ export class Firestack extends Singleton { return configuredProperties; }).catch((err) => { log.info('Native error occurred while calling configure', err); - }) + }); } return this.configurePromise; } - onReady(cb) { + onReady(cb: Function) { + // TODO wut o.O return this.configurePromise = this.configurePromise.then(cb); } @@ -82,78 +130,88 @@ export class Firestack extends Singleton { * when they are needed. Not sure if this is a good * idea or not (imperative vs. direct manipulation/proxy) */ - get auth() { - if (!this._auth) { this._auth = new Authentication(this); } + auth() { return this._auth; } - // database - get database() { - if (!this._db) { this._db = new Database(this); } + + + database() { + if (!this._db) { + this._db = new Database(this); + } return this._db; - // db.enableLogging(this._debug); - // return this.appInstance.database(); } - // analytics - get analytics() { - if (!this._analytics) { this._analytics = new Analytics(this); } + + analytics() { + if (!this._analytics) { + this._analytics = new Analytics(this); + } return this._analytics; } // storage - get storage() { - if (!this._storage) { this._storage = new Storage(this); } + storage() { + if (!this._storage) { + this._storage = new Storage(this); + } return this._storage; } - // presence - get presence() { - if (!this._presence) { this._presence = new Presence(this); } + presence() { + if (!this._presence) { + this._presence = new Presence(this); + } return this._presence; } - // CloudMessaging - get cloudMessaging() { - if (!this._cloudMessaging) { this._cloudMessaging = new CloudMessaging(this); } - return this._cloudMessaging; + + messaging() { + if (!this._messaging) { + this._messaging = new Messaging(this); + } + return this._messaging; + } + + remoteConfig() { + if (!this._remoteConfig) { + this._remoteConfig = new RemoteConfig(this); + } + return this._remoteConfig; } - // other - get ServerValue() { + // // TODO remove as deprecated and its in the wrong place anyway + get ServerValue(): Promise<*> { return promisify('serverValue', FirestackModule)(); } - /** - * remote config - */ - get remoteConfig() { - if (!this.remoteConfig) { - this.remoteConfig = new RemoteConfig(this._remoteConfig); - } - return this.remoteConfig; + get apps(): Array<string> { + return Object.keys(instances); } /** - * app instance - **/ - get app() { - return this.appInstance; + * Returns androids GoogleApiAvailability status and message if available. + * @returns {GoogleApiAvailabilityType|{isAvailable: boolean, status: number}} + */ + get googleApiAvailability(): GoogleApiAvailabilityType { + // if not available then return a fake object for ios - saves doing platform specific logic. + return FirestackModule.googleApiAvailability || { isAvailable: true, status: 0 }; } /** * Logger */ - get log() { + get log(): Log { return this._log; } /** * Redux store **/ - get store() { + get store(): ?Object { return this._store; } - get constants() { + get constants(): Object { if (!this._constants) { this._constants = Object.assign({}, Storage.constants) } @@ -163,7 +221,7 @@ export class Firestack extends Singleton { /** * Set the redux store helper */ - setStore(store) { + setStore(store: Object) { if (store) { this.log.info('Setting the store for Firestack instance'); this._store = store; @@ -173,24 +231,20 @@ export class Firestack extends Singleton { /** * Global event handlers for the single Firestack instance */ - on(name, cb, nativeModule) { + on(name: string, cb: Function, nativeModule: Object = FirestackModuleEvt) { if (!this.eventHandlers[name]) { this.eventHandlers[name] = []; } - if (!nativeModule) { - nativeModule = FirestackModuleEvt; - } + const sub = nativeModule.addListener(name, cb); this.eventHandlers[name].push(sub); return sub; } - off(name) { + off(name: string) { if (this.eventHandlers[name]) { this.eventHandlers[name] .forEach(subscription => subscription.remove()); } } } - -export default Firestack diff --git a/lib/firestackModule.js b/lib/firestackModule.js index 94cda50..a2e31a3 100644 --- a/lib/firestackModule.js +++ b/lib/firestackModule.js @@ -56,7 +56,7 @@ export class FirestackModule { makeRef(path) { const refName = [this._refName, path] - const ref = this._firestack.database.ref(...refName); + const ref = this._firestack.database().ref(...refName); return this._makeRef(ref); } diff --git a/lib/flow.js b/lib/flow.js new file mode 100644 index 0000000..2db69a4 --- /dev/null +++ b/lib/flow.js @@ -0,0 +1,4 @@ +declare module 'react-native' { + // noinspection ES6ConvertVarToLetConst + declare var exports: any; +} diff --git a/lib/modules/analytics.js b/lib/modules/analytics.js index 70fac43..4cd1b90 100644 --- a/lib/modules/analytics.js +++ b/lib/modules/analytics.js @@ -1,42 +1,118 @@ -import {NativeModules, NativeAppEventEmitter} from 'react-native'; +// @flow +import { NativeModules } from 'react-native'; +import { Base } from './base'; + const FirestackAnalytics = NativeModules.FirestackAnalytics; +const AlphaNumericUnderscore = /^[a-zA-Z0-9_]+$/; + +const ReservedEventNames = [ + 'app_clear_data', + 'app_uninstall', + 'app_update', + 'error', + 'first_open', + 'in_app_purchase', + 'notification_dismiss', + 'notification_foreground', + 'notification_open', + 'notification_receive', + 'os_update', + 'session_start', + 'user_engagement', +]; + +export default class Analytics extends Base { + /** + * Logs an app event. + * @param {string} name + * @param params + * @return {Promise} + */ + logEvent(name: string, params: Object): void { + // check name is not a reserved event name + if (ReservedEventNames.includes(name)) { + throw new Error(`event name '${name}' is a reserved event name and can not be used.`); + } -import promisify from '../utils/promisify' -import { Base } from './base' + // name format validation + if (!AlphaNumericUnderscore.test(name)) { + throw new Error(`Event name '${name}' is invalid. Names should contain 1 to 32 alphanumeric characters or underscores.`); + } -export class Analytics extends Base { - constructor(firestack, options={}) { - super(firestack, options); + // maximum number of allowed params check + if (params && Object.keys(params).length > 25) throw new Error('Maximum number of parameters exceeded (25).'); - this._addToFirestackInstance( - 'logEventWithName' - ) + // TODO validate param names and values + // Parameter names can be up to 24 characters long and must start with an alphabetic character + // and contain only alphanumeric characters and underscores. Only String, long and double param + // types are supported. String parameter values can be up to 36 characters long. The "firebase_" + // prefix is reserved and should not be used for parameter names. + + return FirestackAnalytics.logEvent(name, params); } + /** - * Log an event - * @param {string} name The name of the event - * @param {object} props An object containing string-keys - * @return {Promise} + * Sets whether analytics collection is enabled for this app on this device. + * @param enabled */ - logEventWithName(name, props) { - return promisify('logEventWithName', FirestackAnalytics)(name, props); + setAnalyticsCollectionEnabled(enabled: boolean): void { + return FirestackAnalytics.setAnalyticsCollectionEnabled(enabled); } - enable() { - return promisify('setEnabled', FirestackAnalytics)(true); + /** + * Sets the current screen name, which specifies the current visual context in your app. + * @param screenName + * @param screenClassOverride + */ + setCurrentScreen(screenName: string, screenClassOverride: string): void { + return FirestackAnalytics.setCurrentScreen(screenName, screenClassOverride); } - disable() { - return promisify('setEnabled', FirestackAnalytics)(false); + /** + * Sets the minimum engagement time required before starting a session. The default value is 10000 (10 seconds). + * @param milliseconds + */ + setMinimumSessionDuration(milliseconds: number = 10000): void { + return FirestackAnalytics.setMinimumSessionDuration(milliseconds); } - setUser(id, properties={}) { - return promisify('setUserId', FirestackAnalytics)(id, properties); + /** + * Sets the duration of inactivity that terminates the current session. The default value is 1800000 (30 minutes). + * @param milliseconds + */ + setSessionTimeoutDuration(milliseconds: number = 1800000): void { + return FirestackAnalytics.setSessionTimeoutDuration(milliseconds); } - get namespace() { - return 'firestack:analytics' + /** + * Sets the user ID property. + * @param id + */ + setUserId(id: string): void { + return FirestackAnalytics.setUserId(id); + } + + /** + * Sets a user property to a given value. + * @param name + * @param value + */ + setUserProperty(name: string, value: string): void { + return FirestackAnalytics.setUserProperty(name, value); } -} -export default Analytics \ No newline at end of file + /** + * Sets a user property to a given value. + * @param object + */ + setUserProperties(object: Object): void { + for (const property of Object.keys(object)) { + FirestackAnalytics.setUserProperty(property, object[property]); + } + } + + + get namespace(): string { + return 'firestack:analytics'; + } +} diff --git a/lib/modules/auth/Email.js b/lib/modules/auth/Email.js new file mode 100644 index 0000000..92f4d68 --- /dev/null +++ b/lib/modules/auth/Email.js @@ -0,0 +1,9 @@ +export default { + credential(email, password) { + return { + token: email, + secret: password, + provider: 'password', + }; + }, +}; diff --git a/lib/modules/auth/index.js b/lib/modules/auth/index.js new file mode 100644 index 0000000..61ef65e --- /dev/null +++ b/lib/modules/auth/index.js @@ -0,0 +1,233 @@ +// @flow +import { NativeModules, NativeEventEmitter } from 'react-native'; + +import User from './../user'; +import { Base } from './../base'; +import EmailAuthProvider from './Email'; +import { promisify } from './../../utils'; + +const FirestackAuth = NativeModules.FirestackAuth; +const FirestackAuthEvt = new NativeEventEmitter(FirestackAuth); + +type AuthResultType = { authenticated: boolean, user: Object|null }; +type CredentialType = { provider: string, token: string, secret: string }; + +export default class Auth extends Base { + _user: User|null; + _authResult: AuthResultType | null; + authenticated: boolean; + + constructor(firestack: Object, options: Object = {}) { + super(firestack, options); + this._user = null; + this._authResult = null; + this.authenticated = false; + + // attach auth providers + // TODO add missing providers + this.EmailAuthProvider = EmailAuthProvider; + // start listening straight away + // generally though the initial event fired will get ignored + // but this is ok as we fake it with the getCurrentUser below + FirestackAuthEvt.addListener('listenForAuth', this._onAuthStateChanged.bind(this)); + FirestackAuth.listenForAuth(); + } + + /** + * Internal auth changed listener + * @param auth + * @private + */ + _onAuthStateChanged(auth: AuthResultType) { + this._authResult = auth; + this.authenticated = auth ? auth.authenticated || false : false; + if (auth && auth.user && !this._user) this._user = new User(this, auth); + else if ((!auth || !auth.user) && this._user) this._user = null; + else if (this._user) this._user._updateValues(auth); + this.emit('onAuthStateChanged', this._authResult.user || null); + } + + /* + * WEB API + */ + + /** + * Listen for auth changes. + * @param listener + */ + onAuthStateChanged(listener: Function) { + this.log.info('Creating onAuthStateChanged listener'); + this.on('onAuthStateChanged', listener); + if (this._authResult) listener(this._authResult.user || null); + return this._offAuthStateChanged.bind(this, listener); + } + + /** + * Remove auth change listener + * @param listener + */ + _offAuthStateChanged(listener: Function) { + this.log.info('Removing onAuthStateChanged listener'); + this.removeListener('onAuthStateChanged', listener); + } + + /** + * Create a user with the email/password functionality + * @param {string} email The user's email + * @param {string} password The user's password + * @return {Promise} A promise indicating the completion + */ + createUserWithEmailAndPassword(email: string, password: string): Promise<Object> { + this.log.info('Creating user with email and password', email); + return promisify('createUserWithEmail', FirestackAuth, 'auth/')(email, password); + } + + /** + * Sign a user in with email/password + * @param {string} email The user's email + * @param {string} password The user's password + * @return {Promise} A promise that is resolved upon completion + */ + signInWithEmailAndPassword(email: string, password: string): Promise<Object> { + this.log.info('Signing in user with email and password', email); + return promisify('signInWithEmail', FirestackAuth, 'auth/')(email, password); + } + + // TODO move user methods to User class + + /** + * Update the current user's email + * @param {string} email The user's _new_ email + * @return {Promise} A promise resolved upon completion + */ + updateEmail(email: string): Promise<Object> { + return promisify('updateUserEmail', FirestackAuth, 'auth/')(email); + } + + /** + * Send verification email to current user. + */ + sendEmailVerification(): Promise<Object> { + return promisify('sendEmailVerification', FirestackAuth, 'auth/')(); + } + + /** + * Update the current user's password + * @param {string} password the new password + * @return {Promise} + */ + updatePassword(password: string): Promise<Object> { + return promisify('updateUserPassword', FirestackAuth, 'auth/')(password); + } + + /** + * Update the current user's profile + * @param {Object} updates An object containing the keys listed [here](https://firebase.google.com/docs/auth/ios/manage-users#update_a_users_profile) + * @return {Promise} + */ + updateProfile(updates: Object = {}): Promise<Object> { + return promisify('updateUserProfile', FirestackAuth, 'auth/')(updates); + } + + /** + * + * @param credential + */ + link(credential: CredentialType) { + return promisify('link', FirestackAuth, 'auth/')(credential.provider, credential.token, credential.secret); + } + + /** + * Sign the user in with a custom auth token + * @param {string} customToken A self-signed custom auth token. + * @return {Promise} A promise resolved upon completion + */ + signInWithCustomToken(customToken: string): Promise<Object> { + return promisify('signInWithCustomToken', FirestackAuth)(customToken); + } + + /** + * Sign the user in with a third-party authentication provider + * @return {Promise} A promise resolved upon completion + */ + signInWithCredential(credential: any): Promise<Object> { + return promisify('signInWithProvider', FirestackAuth)(credential); + } + + /** + * Re-authenticate a user with a third-party authentication provider + * @return {Promise} A promise resolved upon completion + */ + reauthenticateUser(credential: any): Promise<Object> { + return promisify('reauthenticateWithCredentialForProvider', FirestackAuth)(credential); + } + + /** + * Sign a user in anonymously + * @return {Promise} A promise resolved upon completion + */ + signInAnonymously(): Promise<Object> { + return promisify('signInAnonymously', FirestackAuth, 'auth/')(); + } + + /** + * Send reset password instructions via email + * @param {string} email The email to send password reset instructions + */ + sendPasswordResetEmail(email: string): Promise<Object> { + return promisify('sendPasswordResetWithEmail', FirestackAuth, 'auth/')(email); + } + + /** + * Delete the current user + * @return {Promise} + */ + deleteUser(): Promise<Object> { + return promisify('deleteUser', FirestackAuth, 'auth/')(); + } + + /** + * Delete the current user + * @return {Promise} + */ + reloadUser(): Promise<Object> { + return promisify('reloadUser', FirestackAuth, 'auth/')(); + } + + /** + * get the token of current user + * @return {Promise} + */ + getToken(): Promise<Object> { + return promisify('getToken', FirestackAuth, 'auth/')(); + } + + + /** + * Sign the current user out + * @return {Promise} + */ + signOut(): Promise<Object> { + return promisify('signOut', FirestackAuth, 'auth/')(); + } + + /** + * Get the currently signed in user + * @return {Promise} + */ + getCurrentUser(): Promise<Object> { + return promisify('getCurrentUser', FirestackAuth, 'auth/')(); + } + + /** + * Get the currently signed in user + * @return {Promise} + */ + get currentUser(): User|null { + return this._user; + } + + get namespace(): string { + return 'firestack:auth'; + } +} diff --git a/lib/modules/authentication.js b/lib/modules/authentication.js deleted file mode 100644 index c06edfc..0000000 --- a/lib/modules/authentication.js +++ /dev/null @@ -1,160 +0,0 @@ - -import {NativeModules, NativeEventEmitter} from 'react-native'; -const FirestackAuth = NativeModules.FirestackAuth -const FirestackAuthEvt = new NativeEventEmitter(FirestackAuth); - -import promisify from '../utils/promisify' -import { Base } from './base' - -export class Authentication extends Base { - constructor(firestack, options={}) { - super(firestack, options); - } - - // Auth - listenForAuth(callback) { - this.log.info('Setting up listenForAuth callback'); - const sub = this._on('listenForAuth', callback, FirestackAuthEvt); - FirestackAuth.listenForAuth(); - this.log.info('Listening for auth...'); - return promisify(() => sub, FirestackAuth)(sub); - } - - unlistenForAuth() { - this.log.info('Unlistening for auth'); - this._off('listenForAuth'); - return promisify('unlistenForAuth', FirestackAuth)(); - } - - /** - * Create a user with the email/password functionality - * @param {string} email The user's email - * @param {string} password The user's password - * @return {Promise} A promise indicating the completion - */ - createUserWithEmail(email, password) { - this.log.info('Creating user with email', email); - return promisify('createUserWithEmail', FirestackAuth)(email, password); - } - - /** - * Sign a user in with email/password - * @param {string} email The user's email - * @param {string} password The user's password - * @return {Promise} A promise that is resolved upon completion - */ - signInWithEmail(email, password) { - return promisify('signInWithEmail', FirestackAuth)(email, password) - } - - /** - * Sign the user in with a third-party authentication provider - * @param {string} provider The name of the provider to use for login - * @param {string} authToken The authToken granted by the provider - * @param {string} authSecret The authToken secret granted by the provider - * @return {Promise} A promise resolved upon completion - */ - signInWithProvider(provider, authToken, authSecret) { - return promisify('signInWithProvider', FirestackAuth)(provider, authToken, authSecret) - } - - /** - * Sign the user in with a custom auth token - * @param {string} customToken A self-signed custom auth token. - * @return {Promise} A promise resolved upon completion - */ - signInWithCustomToken(customToken) { - return promisify('signInWithCustomToken', FirestackAuth)(customToken) - } - - /** - * Sign a user in anonymously - * @return {Promise} A promise resolved upon completion - */ - signInAnonymously() { - return promisify('signInAnonymously', FirestackAuth)(); - } - - /** - * Reauthenticate a user with a third-party authentication provider - * @param {string} provider The provider name - * @param {string} token The authToken granted by the provider - * @param {string} secret The authTokenSecret granted by the provider - * @return {Promise} A promise resolved upon completion - */ - reauthenticateWithCredentialForProvider(provider, token, secret) { - return promisify('reauthenticateWithCredentialForProvider', FirestackAuth)(provider, token, secret) - } - - /** - * Update the current user's email - * @param {string} email The user's _new_ email - * @return {Promise} A promise resolved upon completion - */ - updateUserEmail(email) { - return promisify('updateUserEmail', FirestackAuth)(email); - } - - /** - * Update the current user's password - * @param {string} email the new password - * @return {Promise} - */ - updatePassword(password) { - return promisify('updateUserPassword', FirestackAuth)(password); - } - - /** - * Send reset password instructions via email - * @param {string} email The email to send password reset instructions - */ - sendPasswordResetWithEmail(email) { - return promisify('sendPasswordResetWithEmail', FirestackAuth)(email); - } - - /** - * Delete the current user - * @return {Promise} - */ - deleteUser() { - return promisify('deleteUser', FirestackAuth)() - } - /** - * get the token of current user - * @return {Promise} - */ - getToken() { - return promisify('getToken', FirestackAuth)() - } - - /** - * Update the current user's profile - * @param {Object} obj An object containing the keys listed [here](https://firebase.google.com/docs/auth/ios/manage-users#update_a_users_profile) - * @return {Promise} - */ - updateUserProfile(obj) { - return promisify('updateUserProfile', FirestackAuth)(obj); - } - - /** - * Sign the current user out - * @return {Promise} - */ - signOut() { - return promisify('signOut', FirestackAuth)(); - } - - /** - * Get the currently signed in user - * @return {Promise} - */ - getCurrentUser() { - return promisify('getCurrentUser', FirestackAuth)(); - } - - get namespace() { - return 'firestack:auth'; - } -} - -export default Authentication diff --git a/lib/modules/base.js b/lib/modules/base.js index 56bf538..ef25fc2 100644 --- a/lib/modules/base.js +++ b/lib/modules/base.js @@ -1,17 +1,26 @@ /** * @flow */ -import Log from '../utils/log' +import { NativeModules, NativeEventEmitter } from 'react-native'; + +import Log from '../utils/log'; +import EventEmitter from './../utils/eventEmitter'; -import {NativeModules, NativeEventEmitter, AsyncStorage} from 'react-native'; const FirestackModule = NativeModules.Firestack; const FirestackModuleEvt = new NativeEventEmitter(FirestackModule); -import promisify from '../utils/promisify' +const logs = {}; + +// single event emitter for all classes extending base +// TODO +// const EE = new EventEmitter(); + +type FirestackOptions = {}; -let logs = {}; -export class Base { - constructor(firestack, options={}) { +// TODO cleanup +export class Base extends EventEmitter { + constructor(firestack: Object, options: FirestackOptions = {}) { + super(); this.firestack = firestack; this.eventHandlers = {}; @@ -19,40 +28,24 @@ export class Base { this.options = Object.assign({}, firestack.options, options); } + // Logger - get log() { - if (!logs[this.namespace]) { - const debug = this.firestack._debug; - logs[this.namespace] = new Log(this.namespace, debug); - } + get log(): Log { + if (!logs[this.namespace]) logs[this.namespace] = new Log(this.namespace, this.firestack._debug); return logs[this.namespace]; } - _addConstantExports(constants) { - Object.keys(constants).forEach(name => { - FirestackModule[name] = constants[name]; - }); - } - - _addToFirestackInstance(...methods) { - methods.forEach(name => { - this.firestack[name] = this[name].bind(this); - }) - } - /** * app instance **/ - get app() { + get app(): Object { return this.firestack.app; } - whenReady(fn) { - return this.firestack.configurePromise.then(fn); - } - - get namespace() { - return 'firestack:base'; + whenReady(promise: Promise<*>): Promise<*> { + return this.firestack.configurePromise.then((result) => { + return promise; + }); } // Event handlers @@ -68,46 +61,28 @@ export class Base { const sub = nativeModule.addListener(name, cb); this.eventHandlers[name] = sub; resolve(sub); - }) + }); } _off(name) { - return new Promise((resolve, reject) => { + return new Promise((resolve) => { if (this.eventHandlers[name]) { const subscription = this.eventHandlers[name]; subscription.remove(); // Remove subscription delete this.eventHandlers[name]; - resolve(subscription) + resolve(subscription); } }); } } export class ReferenceBase extends Base { - constructor(firestack, path) { + constructor(firestack: Object, path: string) { super(firestack); - - this.path = Array.isArray(path) ? - path : - (typeof path == 'string' ? - [path] : []); - - // sanitize path, just in case - this.path = this.path - .filter(str => str !== "" ); - } - - get key() { - const path = this.path; - return path.length === 0 ? '/' : path[path.length - 1]; + this.path = path || '/'; } - pathToString() { - let path = this.path; - let pathStr = (path.length > 0 ? path.join('/') : '/'); - if (pathStr[0] != '/') { - pathStr = `/${pathStr}` - } - return pathStr; + get key(): string { + return this.path === '/' ? null : this.path.substring(this.path.lastIndexOf('/') + 1); } -} \ No newline at end of file +} diff --git a/lib/modules/cloudmessaging.js b/lib/modules/cloudmessaging.js index 73b390b..bfba997 100644 --- a/lib/modules/cloudmessaging.js +++ b/lib/modules/cloudmessaging.js @@ -2,7 +2,7 @@ import {Platform, NativeModules, NativeEventEmitter} from 'react-native'; const FirestackCloudMessaging = NativeModules.FirestackCloudMessaging; const FirestackCloudMessagingEvt = new NativeEventEmitter(FirestackCloudMessaging); -import promisify from '../utils/promisify' +import { promisify } from '../utils' import { Base, ReferenceBase } from './base' const defaultPermissions = { diff --git a/lib/modules/database.js b/lib/modules/database.js deleted file mode 100644 index cbf3bc4..0000000 --- a/lib/modules/database.js +++ /dev/null @@ -1,518 +0,0 @@ -/** - * Database representation wrapper - */ -import {NativeModules, NativeEventEmitter} from 'react-native'; -const FirestackDatabase = NativeModules.FirestackDatabase; -const FirestackDatabaseEvt = new NativeEventEmitter(FirestackDatabase); - -import promisify from '../utils/promisify' -import { Base, ReferenceBase } from './base' - -let dbSubscriptions = {}; - -class DataSnapshot { - static key:String; - static value:Object; - static exists:boolean; - static hasChildren:boolean; - static childrenCount:Number; - static childKeys:String[]; - - constructor(ref, snapshot) { - this.ref = ref; - this.key = snapshot.key; - this.value = snapshot.value; - this.exists = snapshot.exists || true; - this.priority = snapshot.priority; - this.hasChildren = snapshot.hasChildren || false; - this.childrenCount = snapshot.childrenCount || 0; - this.childKeys = snapshot.childKeys || []; - } - - val() { - return this.value; - } - - forEach(fn) { - (this.childKeys || []) - .forEach(key => fn({key: key, value: this.value[key]})) - } - - map(fn) { - let arr = []; - this.forEach(item => arr.push(fn(item))) - return arr; - } - - reverseMap(fn) { - return this.map(fn).reverse(); - } -} - -class DatabaseOnDisconnect { - constructor(ref) { - this.ref = ref; - } - - setValue(val) { - const path = this.ref.dbPath(); - if (typeof val == 'string') { - return promisify('onDisconnectSetString', FirestackDatabase)(path, val); - } else if (typeof val == 'object') { - return promisify('onDisconnectSetObject', FirestackDatabase)(path, val); - } - } - - remove() { - const path = this.ref.dbPath(); - return promisify('onDisconnectRemove', FirestackDatabase)(path); - } - - cancel() { - const path = this.ref.dbPath(); - return promisify('onDisconnectCancel', FirestackDatabase)(path); - } -} - -class DatabaseQuery { - static ref: DatabaseRef; - static orderBy: String[]; - static limit: String[]; - static filters: Object; - - constructor(ref) { - this.ref = ref; - this.reset(); - } - - setOrderBy(name, ...args) { - this.orderBy = [name].concat(args); - return this.ref; - } - - setLimit(name, ...args) { - this.limit = [name].concat(args); - return this.ref; - } - - setFilter(name, ...args) { - this.filters[name] = args.filter(n => n != undefined); - return this.ref; - } - - build() { - const argsSeparator = ':' - let modifiers = []; - if (this.orderBy) { - modifiers.push(this.orderBy.join(argsSeparator)); - } - if (this.limit) { - modifiers.push(this.limit.join(argsSeparator)); - } - Object.keys(this.filters) - .forEach(key => { - let filter = this.filters[key]; - if (filter) { - const cleanFilters = filter.filter((f) => typeof f !== "undefined"); - const filterArgs = ([key].concat(cleanFilters)).join(argsSeparator); - modifiers.push(filterArgs); - } - }) - return modifiers; - } - - reset() { - this.orderBy = null; - this.limit = null; - this.filters = {}; - ['startAt', 'endAt', 'equalTo'] - .forEach(key => this.filters[key] = null); - return this.ref; - } -} - -// https://firebase.google.com/docs/reference/js/firebase.database.Reference -const separator = '/'; -class DatabaseRef extends ReferenceBase { - constructor(db, path) { - super(db.firestack, path); - - this.db = db; - this.query = new DatabaseQuery(this); - this.listeners = {}; - - // Aliases - this.get = this.getAt; - this.set = this.setAt; - this.update = this.updateAt; - this.remove = this.removeAt; - - this.log.debug('Created new DatabaseRef', this.dbPath()); - } - - // Parent roots - parent() { - const parentPaths = this.path.slice(0, -1); - return new DatabaseRef(this.db, parentPaths); - } - - root() { - return new DatabaseRef(this.db, []); - } - - child(...paths) { - return new DatabaseRef(this.db, this.path.concat(paths)); - } - - keepSynced(bool) { - const path = this.dbPath(); - return promisify('keepSynced', FirestackDatabase)(path, bool); - } - - // Get the value of a ref either with a key - getAt() { - const path = this.dbPath(); - const modifiers = this.dbModifiers(); - return promisify('onOnce', FirestackDatabase)(path, modifiers, 'value'); - } - - setAt(val) { - const path = this.dbPath(); - const value = this._serializeValue(val); - return promisify('set', FirestackDatabase)(path, value) - } - - updateAt(val) { - const path = this.dbPath(); - const value = this._serializeValue(val); - return promisify('update', FirestackDatabase)(path, value) - } - - removeAt(key) { - const path = this.dbPath(); - return promisify('remove', FirestackDatabase)(path) - } - - push(val={}) { - const path = this.dbPath(); - const value = this._serializeValue(val); - return promisify('push', FirestackDatabase)(path, value) - .then(({ref}) => { - return new DatabaseRef(this.db, ref.split(separator)) - }) - } - - on(evt, cb) { - const path = this.dbPath(); - const modifiers = this.dbModifiers(); - return this.db.on(path, evt, cb) - .then(({callback, subscriptions}) => { - return promisify('on', FirestackDatabase)(path, modifiers, evt) - .then(() => { - this.listeners[evt] = subscriptions; - callback(this); - return subscriptions; - }) - }); - } - - once(evt='once', cb) { - const path = this.dbPath(); - const modifiers = this.dbModifiers(); - return promisify('onOnce', FirestackDatabase)(path, modifiers, evt) - .then(({snapshot}) => new DataSnapshot(this, snapshot)) - .then(snapshot => { - if (cb && typeof cb === 'function') { - cb(snapshot); - } - return snapshot; - }) - } - - off(evt='', origCB) { - const path = this.dbPath(); - return this.db.off(path, evt, origCB) - .then(({callback, subscriptions}) => { - if (dbSubscriptions[path] && dbSubscriptions[path][evt] && dbSubscriptions[path][evt].length > 0) { - return subscriptions; - } - - return promisify('off', FirestackDatabase)(path, evt) - .then(() => { - // subscriptions.forEach(sub => sub.remove()); - delete this.listeners[evt]; - callback(this); - return subscriptions; - }) - }) - .catch(err => { - console.error('Never get here', err); - }) - } - - cleanup() { - let promises = Object.keys(this.listeners) - .map(key => this.off(key)) - return Promise.all(promises); - } - - // Sanitize value - // As Firebase cannot store date objects. - _serializeValue(obj={}) { - return Object.keys(obj).reduce((sum, key) => { - let val = obj[key]; - if (val instanceof Date) { - val = val.toISOString(); - } - return { - ...sum, - [key]: val - } - }, {}); - } - - _deserializeValue(obj={}) { - return Object.keys(obj).reduce((sum, key) => { - let val = obj[key]; - if (val instanceof Date) { - val = val.getTime(); - } - return { - ...sum, - [key]: val - } - }, {}); - } - - // Modifiers - orderByKey() { - return this.query.setOrderBy('orderByKey'); - } - - orderByPriority() { - return this.query.setOrderBy('orderByPriority'); - } - - orderByValue() { - return this.query.setOrderBy('orderByValue'); - } - - orderByChild(key) { - return this.query.setOrderBy('orderByChild', key); - } - - // Limits - limitToLast(limit) { - return this.query.setLimit('limitToLast', limit); - } - - limitToFirst(limit) { - return this.query.setLimit('limitToFirst', limit); - } - - // Filters - equalTo(value, key) { - return this.query.setFilter('equalTo', value, key); - } - - endAt(value, key) { - return this.query.setFilter('endAt', value, key); - } - - startAt(value, key) { - return this.query.setFilter('startAt', value, key); - } - - presence(path) { - const presence = this.firestack.presence; - const ref = path ? this.child(path) : this; - return presence.ref(ref, this.dbPath()); - } - - // onDisconnect - onDisconnect() { - return new DatabaseOnDisconnect(this); - } - - // attributes - get fullPath() { - return this.dbPath(); - } - - get name() { - return this.path.splice(-1); - } - - dbPath() { - let path = this.path; - let pathStr = (path.length > 0 ? path.join('/') : '/'); - if (pathStr[0] != '/') { - pathStr = `/${pathStr}` - } - return pathStr; - } - - dbModifiers() { - const modifiers = this.query.build(); - this.query.reset(); // should we reset this - return modifiers; - } - - get namespace() { - return `firestack:dbRef` - } -} - -export class Database extends Base { - - constructor(firestack, options={}) { - super(firestack, options); - this.log.debug('Created new Database instance', this.options); - - this.persistenceEnabled = false; - this.successListener = null; - this.errorListener = null; - this.refs = {}; - } - - ref(...path) { - const key = this._pathKey(path); - if (!this.refs[key]) { - const ref = new DatabaseRef(this, path); - this.refs[key] = ref; - } - return this.refs[key]; - } - - setPersistence(enable=true) { - let promise; - if (this.persistenceEnabled !== enable) { - this.log.debug(`${enable ? 'Enabling' : 'Disabling'} persistence`); - promise = this.whenReady(promisify('enablePersistence', FirestackDatabase)(enable)); - this.persistenceEnabled = enable; - } else { - promise = this.whenReady(Promise.resolve({status: "Already enabled"})) - } - - return promise; - } - - handleDatabaseEvent(evt) { - const body = evt.body; - const path = body.path; - const evtName = body.eventName; - - const subscriptions = dbSubscriptions[path]; - - if (subscriptions) { - const cbs = subscriptions[evtName]; - cbs.forEach(cb => { - if (cb && typeof(cb) === 'function') { - const snap = new DataSnapshot(this, body.snapshot); - this.log.debug('database_event received', path, evtName); - cb(snap, body); - } - }); - } - } - - handleDatabaseError(evt) { - this.log.debug('handleDatabaseError ->', evt); - } - - on(path, evt, cb) { - const key = this._pathKey(path); - - if (!dbSubscriptions[key]) { - dbSubscriptions[key] = {}; - } - - if (!dbSubscriptions[key][evt]) { - dbSubscriptions[key][evt] = []; - } - dbSubscriptions[key][evt].push(cb); - - if (!this.successListener) { - this.successListener = FirestackDatabaseEvt - .addListener( - 'database_event', - this.handleDatabaseEvent.bind(this)); - } - - if (!this.errorListener) { - this.errorListener = FirestackDatabaseEvt - .addListener( - 'database_error', - this.handleDatabaseError.bind(this)); - } - - const callback = (ref) => { - const key = this._pathKey(ref.path); - this.refs[key] = ref; - } - const subscriptions = [this.successListener, this.errorListener]; - return Promise.resolve({callback, subscriptions}); - } - - off(path, evt, origCB) { - const key = this._pathKey(path); - // Remove subscription - if (dbSubscriptions[key]) { - if (!evt || evt === "") { - dbSubscriptions[key] = {}; - } else if (dbSubscriptions[key][evt]) { - if (origCB) { - dbSubscriptions[key][evt].splice(dbSubscriptions[key][evt].indexOf(origCB), 1); - } else { - delete dbSubscriptions[key][evt]; - } - } - - if (Object.keys(dbSubscriptions[key]).length <= 0) { - // there are no more subscriptions - // so we can unwatch - delete dbSubscriptions[key] - } - if (Object.keys(dbSubscriptions).length == 0) { - if (this.successListener) { - this.successListener.remove(); - this.successListener = null; - } - if (this.errorListener) { - this.errorListener.remove(); - this.errorListener = null; - } - } - } - const callback = (ref) => { - const key = this._pathKey(ref.path); - delete this.refs[key]; - } - const subscriptions = [this.successListener, this.errorListener]; - return Promise.resolve({callback, subscriptions}); - } - - cleanup() { - let promises = Object.keys(this.refs) - .map(key => this.refs[key]) - .map(ref => ref.cleanup()) - return Promise.all(promises); - } - - release(...path) { - const key = this._pathKey(path); - if (this.refs[key]) { - delete this.refs[key]; - } - } - - _pathKey(...path) { - return path.join('-'); - } - - get namespace() { - return 'firestack:database' - } -} - -export default Database diff --git a/lib/modules/database/disconnect.js b/lib/modules/database/disconnect.js new file mode 100644 index 0000000..bce7f83 --- /dev/null +++ b/lib/modules/database/disconnect.js @@ -0,0 +1,35 @@ +/* @flow */ + +import { NativeModules } from 'react-native'; +import { promisify } from './../../utils'; +import Reference from './reference'; + +const FirestackDatabase = NativeModules.FirestackDatabase; + +/** + * @class Disconnect + */ +export default class Disconnect { + ref: Reference; + + constructor(ref: Reference) { + this.ref = ref; + } + + setValue(val: string | Object) { + const path = this.ref._dbPath(); + if (typeof val === 'string') { + return promisify('onDisconnectSetString', FirestackDatabase)(path, val); + } else if (typeof val === 'object') { + return promisify('onDisconnectSetObject', FirestackDatabase)(path, val); + } + } + + remove() { + return promisify('onDisconnectRemove', FirestackDatabase)(this.ref._dbPath()); + } + + cancel() { + return promisify('onDisconnectCancel', FirestackDatabase)(this.ref._dbPath()); + } +} diff --git a/lib/modules/database/index.js b/lib/modules/database/index.js new file mode 100644 index 0000000..f3eecdd --- /dev/null +++ b/lib/modules/database/index.js @@ -0,0 +1,259 @@ +/** + * @flow + * Database representation wrapper + */ +import { NativeModules, NativeEventEmitter } from 'react-native'; + +import { Base } from './../base'; +import Snapshot from './snapshot.js'; +import Reference from './reference.js'; +import { promisify } from './../../utils'; + +const FirestackDatabase = NativeModules.FirestackDatabase; +const FirestackDatabaseEvt = new NativeEventEmitter(FirestackDatabase); + +/** + * @class Database + */ +export default class Database extends Base { + constructor(firestack: Object, options: Object = {}) { + super(firestack, options); + this.subscriptions = {}; + + this.transactions = {}; + this.errorSubscriptions = {}; + + this.serverTimeOffset = 0; + this.persistenceEnabled = false; + this.namespace = 'firestack:database'; + + this.successListener = FirestackDatabaseEvt.addListener( + 'database_event', + event => this._handleDatabaseEvent(event) + ); + + this.errorListener = FirestackDatabaseEvt.addListener( + 'database_error', + err => this._handleDatabaseError(err) + ); + + this.transactionListener = FirestackDatabaseEvt.addListener( + 'database_transaction_update', + event => this._handleDatabaseTransaction(event) + ); + + this.offsetRef = this.ref('.info/serverTimeOffset'); + + this.offsetRef.on('value', (snapshot) => { + this.serverTimeOffset = snapshot.val() || this.serverTimeOffset; + }); + + this.log.debug('Created new Database instance', this.options); + } + + /** + * https://firebase.google.com/docs/reference/js/firebase.database.ServerValue + * @returns {{TIMESTAMP: (*|{[.sv]: string})}} + * @constructor + */ + get ServerValue(): Object { + return { + TIMESTAMP: FirestackDatabase.serverValueTimestamp || { '.sv': 'timestamp' }, + }; + } + + /** + * Returns a new firestack reference instance + * @param path + * @returns {Reference} + */ + ref(path: string) { + return new Reference(this, path); + } + + /** + * Enabled / disable database persistence + * @param enable + * @returns {*} + */ + setPersistence(enable: boolean = true) { + if (this.persistenceEnabled !== enable) { + this.log.debug(`${enable ? 'Enabling' : 'Disabling'} persistence`); + this.persistenceEnabled = enable; + return this.whenReady(promisify('enablePersistence', FirestackDatabase)(enable)); + } + + return this.whenReady(Promise.resolve({ status: 'Already enabled' })); + } + + /** + * + * @param path + * @param modifiersString + * @param modifiers + * @param eventName + * @param cb + * @returns {*} + */ + on(path: string, modifiersString: string, modifiers: Array<string>, eventName: string, cb: () => void, errorCb: () => void) { + const handle = this._handle(path, modifiersString); + this.log.debug('adding on listener', handle); + + if (!this.subscriptions[handle]) this.subscriptions[handle] = {}; + if (!this.subscriptions[handle][eventName]) this.subscriptions[handle][eventName] = []; + this.subscriptions[handle][eventName].push(cb); + if (errorCb) { + if (!this.errorSubscriptions[handle]) this.errorSubscriptions[handle] = []; + this.errorSubscriptions[handle].push(errorCb); + } + + return promisify('on', FirestackDatabase)(path, modifiersString, modifiers, eventName); + } + + /** + * + * @param path + * @param modifiersString + * @param eventName + * @param origCB + * @returns {*} + */ + off(path: string, modifiersString: string, eventName?: string, origCB?: () => void) { + const handle = this._handle(path, modifiersString); + this.log.debug('off() : ', handle, eventName); + + if (!this.subscriptions[handle] || (eventName && !this.subscriptions[handle][eventName])) { + this.log.warn('off() called, but not currently listening at that location (bad path)', handle, eventName); + return Promise.resolve(); + } + + if (eventName && origCB) { + const i = this.subscriptions[handle][eventName].indexOf(origCB); + + if (i === -1) { + this.log.warn('off() called, but the callback specified is not listening at that location (bad path)', handle, eventName); + return Promise.resolve(); + } + + this.subscriptions[handle][eventName].splice(i, 1); + if (this.subscriptions[handle][eventName].length > 0) return Promise.resolve(); + } else if (eventName) { + this.subscriptions[handle][eventName] = []; + } else { + this.subscriptions[handle] = {}; + } + this.errorSubscriptions[handle] = []; + return promisify('off', FirestackDatabase)(path, modifiersString, eventName); + } + + /** + * Removes all event handlers and their native subscriptions + * @returns {Promise.<*>} + */ + cleanup() { + const promises = []; + Object.keys(this.subscriptions).forEach((handle) => { + Object.keys(this.subscriptions[handle]).forEach((eventName) => { + const separator = handle.indexOf('|'); + const path = handle.substring(0, separator); + const modifiersString = handle.substring(separator + 1); + promises.push(this.off(path, modifiersString, eventName)); + }); + }); + + return Promise.all(promises); + } + + goOnline() { + FirestackDatabase.goOnline(); + } + + goOffline() { + FirestackDatabase.goOffline(); + } + + addTransaction(path, updateCallback, applyLocally) { + let id = this._generateTransactionID(); + this.transactions[id] = updateCallback; + return promisify('beginTransaction', FirestackDatabase)(path, id, applyLocally || false) + .then((v) => {delete this.transactions[id]; return v;}, + (e) => {delete this.transactions[id]; throw e;}); + } + + _generateTransactionID() { + // 10 char random alphanumeric + return Math.random().toString(36).substr(2, 10); + } + + _handleDatabaseTransaction(event) { + const {id, originalValue} = event; + let newValue; + try { + const updateCallback = this.transactions[id]; + newValue = updateCallback(originalValue); + } finally { + let abort = false; + if (newValue === undefined) { + abort = true; + } + FirestackDatabase.tryCommitTransaction(id, {value: newValue}, abort); + } + } + + /** + * INTERNALS + */ + _getServerTime() { + return new Date().getTime() + this.serverTimeOffset; + } + + /** + * + * @param path + * @param modifiersString + * @returns {string} + * @private + */ + _handle(path: string = '', modifiersString: string = '') { + return `${path}|${modifiersString}`; + } + + + /** + * + * @param event + * @private + */ + _handleDatabaseEvent(event: Object) { + const body = event.body || {}; + const { path, modifiersString, eventName, snapshot, handlePath } = body; + const handle = this._handle(handlePath, modifiersString); + + this.log.debug('_handleDatabaseEvent: ', handle, eventName, snapshot && snapshot.key); + + if (this.subscriptions[handle] && this.subscriptions[handle][eventName]) { + this.subscriptions[handle][eventName].forEach((cb) => { + cb(new Snapshot(new Reference(this, path, modifiersString.split('|')), snapshot), body); + }); + } else { + FirestackDatabase.off(path, modifiersString, eventName, () => { + this.log.debug('_handleDatabaseEvent: No JS listener registered, removed native listener', handle, eventName); + }); + } + } + + /** + * + * @param err + * @private + */ + _handleDatabaseError(err: Object) { + const body = err.body || {}; + const { path, modifiersString, eventName, msg } = body; + const handle = this._handle(path, modifiersString); + + this.log.debug('_handleDatabaseError ->', handle, eventName, err); + + if (this.errorSubscriptions[handle]) this.errorSubscriptions[handle].forEach((cb) => cb(new Error(msg))); + } +} diff --git a/lib/modules/database/query.js b/lib/modules/database/query.js new file mode 100644 index 0000000..90f451f --- /dev/null +++ b/lib/modules/database/query.js @@ -0,0 +1,56 @@ +/** + * @flow + */ +'use strict'; + +import { ReferenceBase } from './../base'; +import Reference from './reference.js'; + +/** + * @class Query + */ +export default class Query extends ReferenceBase { + static ref: Reference; + + static modifiers: Array<string>; + + ref: Reference; + + constructor(ref: Reference, path: string, existingModifiers?: Array<string>) { + super(ref.db, path); + this.log.debug('creating Query ', path, existingModifiers); + this.ref = ref; + this.modifiers = existingModifiers ? [...existingModifiers] : []; + } + + setOrderBy(name: string, key?: string) { + if (key) { + this.modifiers.push(name + ':' + key); + } else { + this.modifiers.push(name); + } + } + + setLimit(name: string, limit: number) { + this.modifiers.push(name + ':' + limit); + } + + setFilter(name: string, value: any, key?:string) { + if (key) { + this.modifiers.push(name + ':' + value + ':' + (typeof value) + ':' + key); + } else { + this.modifiers.push(name + ':' + value + ':' + (typeof value)); + } + } + + getModifiers(): Array<string> { + return [...this.modifiers]; + } + + getModifiersString(): string { + if (!this.modifiers || !Array.isArray(this.modifiers)) { + return ''; + } + return this.modifiers.join('|'); + } +} diff --git a/lib/modules/database/reference.js b/lib/modules/database/reference.js new file mode 100644 index 0000000..adc6952 --- /dev/null +++ b/lib/modules/database/reference.js @@ -0,0 +1,358 @@ +/** + * @flow + */ +import { NativeModules } from 'react-native'; + +import Query from './query.js'; +import Snapshot from './snapshot'; +import Disconnect from './disconnect'; +import { ReferenceBase } from './../base'; +import { promisify, isFunction, isObject, tryJSONParse, tryJSONStringify, generatePushID } from './../../utils'; + +const FirestackDatabase = NativeModules.FirestackDatabase; + +// https://firebase.google.com/docs/reference/js/firebase.database.Reference + +/** + * @class Reference + */ +export default class Reference extends ReferenceBase { + + db: FirestackDatabase; + query: Query; + + constructor(db: FirestackDatabase, path: string, existingModifiers?: Array<string>) { + super(db.firestack, path); + this.db = db; + this.namespace = 'firestack:db:ref'; + this.query = new Query(this, path, existingModifiers); + this.log.debug('Created new Reference', this.db._handle(path, existingModifiers)); + } + + /** + * + * @param bool + * @returns {*} + */ + keepSynced(bool: boolean) { + const path = this._dbPath(); + return promisify('keepSynced', FirestackDatabase)(path, bool); + } + + /** + * + * @param value + * @returns {*} + */ + set(value: any) { + const path = this._dbPath(); + const _value = this._serializeAnyType(value); + return promisify('set', FirestackDatabase)(path, _value); + } + + /** + * + * @param value + * @returns {*} + */ + setWithPriority(value: any, priority: any) { + const path = this._dbPath(); + const _value = this._serializeAnyType(value); + const _priority = this._serializeAnyType(priority); + return promisify('setWithPriority', FirestackDatabase)(path, _value, _priority); + } + + /** + * + * @param val + * @returns {*} + */ + update(val: Object) { + const path = this._dbPath(); + const value = this._serializeObject(val); + return promisify('update', FirestackDatabase)(path, value); + } + + /** + * + * @returns {*} + */ + remove() { + return promisify('remove', FirestackDatabase)(this._dbPath()); + } + + /** + * + * @param value + * @param onComplete + * @returns {*} + */ + push(value: any, onComplete: Function) { + if (value === null || value === undefined) { + const _path = this.path + '/' + generatePushID(this.db.serverTimeOffset); + return new Reference(this.db, _path); + } + + const path = this._dbPath(); + const _value = this._serializeAnyType(value); + return promisify('push', FirestackDatabase)(path, _value) + .then(({ ref }) => { + const newRef = new Reference(this.db, ref); + if (isFunction(onComplete)) return onComplete(null, newRef); + return newRef; + }).catch((e) => { + if (isFunction(onComplete)) return onComplete(e, null); + return e; + }); + } + + on(eventName: string, cb: () => any, errorCb: () => any) { + if (!isFunction(cb)) throw new Error('The specified callback must be a function'); + if (errorCb && !isFunction(errorCb)) throw new Error('The specified error callback must be a function'); + const path = this._dbPath(); + const modifiers = this.query.getModifiers(); + const modifiersString = this.query.getModifiersString(); + this.log.debug('adding reference.on', path, modifiersString, eventName); + return this.db.on(path, modifiersString, modifiers, eventName, cb, errorCb); + } + + once(eventName: string = 'once', cb: (snapshot: Object) => void) { + const path = this._dbPath(); + const modifiers = this.query.getModifiers(); + const modifiersString = this.query.getModifiersString(); + return promisify('onOnce', FirestackDatabase)(path, modifiersString, modifiers, eventName) + .then(({ snapshot, path, modifiersString}) => new Snapshot( + new Reference(this.db, path, modifiersString.split('|')), + snapshot)) + .then((snapshot) => { + if (isFunction(cb)) cb(snapshot); + return snapshot; + }); + } + + off(eventName?: string = '', origCB?: () => any) { + const path = this._dbPath(); + const modifiersString = this.query.getModifiersString(); + this.log.debug('ref.off(): ', path, modifiersString, eventName); + return this.db.off(path, modifiersString, eventName, origCB); + } + + transaction(transactionUpdate, onComplete, applyLocally) { + const path = this._dbPath(); + return this.db.addTransaction(path, transactionUpdate, applyLocally) + .then((({ snapshot, committed }) => {return {snapshot: new Snapshot(this, snapshot), committed}}).bind(this)) + .then(({ snapshot, committed }) => { + if (isFunction(onComplete)) onComplete(null, snapshot); + return {snapshot, committed}; + }).catch((e) => { + if (isFunction(onComplete)) return onComplete(e, null); + throw e; + }); + } + + /** + * MODIFIERS + */ + + /** + * + * @returns {Reference} + */ + orderByKey(): Reference { + return this.orderBy('orderByKey'); + } + + /** + * + * @returns {Reference} + */ + orderByPriority(): Reference { + return this.orderBy('orderByPriority'); + } + + /** + * + * @returns {Reference} + */ + orderByValue(): Reference { + return this.orderBy('orderByValue'); + } + + /** + * + * @param key + * @returns {Reference} + */ + orderByChild(key: string): Reference { + return this.orderBy('orderByChild', key); + } + + /** + * + * @param name + * @param key + * @returns {Reference} + */ + orderBy(name: string, key?: string): Reference { + const newRef = new Reference(this.db, this.path, this.query.getModifiers()); + newRef.query.setOrderBy(name, key); + return newRef; + } + + /** + * LIMITS + */ + + /** + * + * @param limit + * @returns {Reference} + */ + limitToLast(limit: number): Reference { + return this.limit('limitToLast', limit); + } + + /** + * + * @param limit + * @returns {Reference} + */ + limitToFirst(limit: number): Reference { + return this.limit('limitToFirst', limit); + } + + /** + * + * @param name + * @param limit + * @returns {Reference} + */ + limit(name: string, limit: number): Reference { + const newRef = new Reference(this.db, this.path, this.query.getModifiers()); + newRef.query.setLimit(name, limit); + return newRef; + } + + /** + * FILTERS + */ + + /** + * + * @param value + * @param key + * @returns {Reference} + */ + equalTo(value: any, key?: string): Reference { + return this.filter('equalTo', value, key); + } + + /** + * + * @param value + * @param key + * @returns {Reference} + */ + endAt(value: any, key?: string): Reference { + return this.filter('endAt', value, key); + } + + /** + * + * @param value + * @param key + * @returns {Reference} + */ + startAt(value: any, key?: string): Reference { + return this.filter('startAt', value, key); + } + + /** + * + * @param name + * @param value + * @param key + * @returns {Reference} + */ + filter(name: string, value: any, key?: string): Reference { + const newRef = new Reference(this.db, this.path, this.query.getModifiers()); + newRef.query.setFilter(name, value, key); + return newRef; + } + + onDisconnect() { + return new Disconnect(this); + } + + child(path: string) { + return new Reference(this.db, this.path + '/' + path); + } + + toString(): string { + return this._dbPath(); + } + + /** + * GETTERS + */ + + /** + * Returns the parent ref of the current ref i.e. a ref of /foo/bar would return a new ref to '/foo' + * @returns {*} + */ + get parent(): Reference|null { + if (this.path === '/') return null; + return new Reference(this.db, this.path.substring(0, this.path.lastIndexOf('/'))); + } + + + /** + * Returns a ref to the root of db - '/' + * @returns {Reference} + */ + get root(): Reference { + return new Reference(this.db, '/'); + } + + /** + * INTERNALS + */ + + _dbPath(): string { + return this.path; + } + + /** + * + * @param obj + * @returns {Object} + * @private + */ + _serializeObject(obj: Object) { + if (!isObject(obj)) return obj; + + // json stringify then parse it calls toString on Objects / Classes + // that support it i.e new Date() becomes a ISO string. + return tryJSONParse(tryJSONStringify(obj)); + } + + /** + * + * @param value + * @returns {*} + * @private + */ + _serializeAnyType(value: any) { + if (isObject(value)) { + return { + type: 'object', + value: this._serializeObject(value), + }; + } + + return { + type: typeof value, + value, + }; + } +} diff --git a/lib/modules/database/snapshot.js b/lib/modules/database/snapshot.js new file mode 100644 index 0000000..f05bcd2 --- /dev/null +++ b/lib/modules/database/snapshot.js @@ -0,0 +1,93 @@ +/** + * @flow + */ +import Reference from './reference.js'; +import { isObject, deepGet, deepExists } from './../../utils'; + +export default class Snapshot { + static key: String; + static value: Object; + static exists: boolean; + static hasChildren: boolean; + static childKeys: String[]; + + ref: Object; + key: string; + value: any; + exists: boolean; + priority: any; + childKeys: Array<string>; + + constructor(ref: Reference, snapshot: Object) { + this.ref = ref; + this.key = snapshot.key; + this.value = snapshot.value; + this.exportValue = snapshot.exportValue; + this.priority = snapshot.priority === undefined ? null : snapshot.priority; + this.childKeys = snapshot.childKeys || []; + } + + /* + * DEFAULT API METHODS + */ + + val() { + return this.value; + } + + exportVal() { + return this.exportValue; + } + + child(path: string) { + const value = deepGet(this.value, path); + const exportValue = deepGet(this.exportValue, path); + const childRef = this.ref.child(path); + return new Snapshot(childRef, { + value, + exportValue, + key: childRef.key, + exists: value !== null, + priority: (exportValue && Object.prototype.hasOwnProperty.call(exportValue, '.priority')) ? exportValue['.priority'] : undefined, + childKeys: isObject(value) ? Object.keys(value) : [], + }); + } + + exists() { + return this.value !== null; + } + + forEach(fn: (key: any) => any) { + return this.childKeys.forEach((key, i) => fn(this.child(key), i)); + } + + getPriority() { + return this.priority; + } + + hasChild(path: string) { + return deepExists(this.value, path); + } + + hasChildren() { + return this.numChildren() > 0; + } + + numChildren() { + if (!isObject(this.value)) return 0; + return Object.keys(this.value).length; + } + + /* + * EXTRA API METHODS + */ + map(fn: (key: string) => mixed) { + const arr = []; + this.forEach((item, i) => arr.push(fn(item, i))); + return arr; + } + + reverseMap(fn: (key: string) => mixed) { + return this.map(fn).reverse(); + } +} diff --git a/lib/modules/messaging.js b/lib/modules/messaging.js new file mode 100644 index 0000000..1a59d70 --- /dev/null +++ b/lib/modules/messaging.js @@ -0,0 +1,115 @@ +import { NativeModules, NativeEventEmitter } from 'react-native'; +import { Base } from './base'; +import { promisify } from '../utils'; + +const FirestackMessaging = NativeModules.FirestackMessaging || NativeModules.FirestackCloudMessaging; +const FirestackMessagingEvt = new NativeEventEmitter(FirestackMessaging); + +type RemoteMessage = { + id: string, + type: string, + ttl?: number, + sender: string, + collapseKey?: string, + data: Object, +}; + +/** + * @class Messaging + */ +export default class Messaging extends Base { + constructor(firestack, options = {}) { + super(firestack, options); + this.namespace = 'firestack:messaging'; + } + + /* + * WEB API + */ + // TODO move to new event emitter logic + onMessage(callback) { + this.log.info('Setting up onMessage callback'); + const sub = this._on('FirestackReceiveNotification', callback, FirestackMessagingEvt); + return promisify(() => sub, FirestackMessaging)(sub); + } + + // TODO this is wrong - also there is no 'off' onMessage should return the unsubscribe function + offMessage() { + this.log.info('Unlistening from onMessage (offMessage)'); + this._off('FirestackReceiveNotification'); + } + + offMessageReceived(...args) { + return this.offMessage(...args); + } + + getToken() { + this.log.info('getToken for cloudMessaging'); + return promisify('getToken', FirestackMessaging)(); + } + + // sendMessage(details: Object = {}, type: string = 'local') { + // const methodName = `send${type == 'local' ? 'Local' : 'Remote'}`; + // this.log.info('sendMessage', methodName, details); + // return promisify(methodName, FirestackMessaging)(details); + // } + // + // scheduleMessage(details: Object = {}, type: string = 'local') { + // const methodName = `schedule${type == 'local' ? 'Local' : 'Remote'}`; + // return promisify(methodName, FirestackMessaging)(details); + // } + + // OLD + send(remoteMessage: RemoteMessage) { + if (!remoteMessage || !remoteMessage.data) return Promise.reject(new Error('Invalid remote message format provided.')); + return promisify('send', FirestackMessaging)(remoteMessage); + } + + // + listenForTokenRefresh(callback) { + this.log.info('Setting up listenForTokenRefresh callback'); + const sub = this._on('FirestackRefreshToken', callback, FirestackMessagingEvt); + return promisify(() => sub, FirestackMessaging)(sub); + } + + unlistenForTokenRefresh() { + this.log.info('Unlistening for TokenRefresh'); + this._off('FirestackRefreshToken'); + } + + subscribeToTopic(topic) { + this.log.info(`subscribeToTopic ${topic}`); + const finalTopic = `/topics/${topic}`; + return promisify('subscribeToTopic', FirestackMessaging)(finalTopic); + } + + unsubscribeFromTopic(topic) { + this.log.info(`unsubscribeFromTopic ${topic}`); + const finalTopic = `/topics/${topic}`; + return promisify('unsubscribeFromTopic', FirestackMessaging)(finalTopic); + } + + // New api + onRemoteMessage(callback) { + this.log.info('On remote message callback'); + const sub = this._on('messaging_remote_event_received', callback, FirestackMessagingEvt); + return promisify(() => sub, FirestackMessaging)(sub); + } + + onLocalMessage(callback) { + this.log.info('on local callback'); + const sub = this._on('messaging_local_event_received', callback, FirestackMessagingEvt); + return promisify(() => sub, FirestackMessaging)(sub); + } + + listenForReceiveUpstreamSend(callback) { + this.log.info('Setting up send callback'); + const sub = this._on('FirestackUpstreamSend', callback, FirestackMessagingEvt); + return promisify(() => sub, FirestackMessaging)(sub); + } + + unlistenForReceiveUpstreamSend() { + this.log.info('Unlistening for send'); + this._off('FirestackUpstreamSend'); + } +} diff --git a/lib/modules/presence.js b/lib/modules/presence.js index cac57ae..e44b6b6 100644 --- a/lib/modules/presence.js +++ b/lib/modules/presence.js @@ -1,41 +1,39 @@ -import invariant from 'invariant' -import promisify from '../utils/promisify' -import { Base, ReferenceBase } from './base' +import { Base, ReferenceBase } from './base'; class PresenceRef extends ReferenceBase { constructor(presence, ref, pathParts) { super(presence.firestack); - - this.presence = presence; - const db = this.firestack.database; this.ref = ref; - this.lastOnlineRef = this.ref.child('lastOnline'); - - this._connectedRef = db.ref('.info/connected'); - this._pathParts = pathParts; - this._onConnect = []; + this.presence = presence; + this._pathParts = pathParts; + this.lastOnlineRef = this.ref.child('lastOnline'); + this._connectedRef = this.firestack.database().ref('.info/connected'); } setOnline() { - this.ref.setAt({online: true}) + this.ref.set({ online: true }); + + // todo cleanup - creating a ref every time? this._connectedRef.on('value', (snapshot) => { const val = snapshot.val(); if (val) { // add self to connection list // this.ref.push() - this.ref.setAt({ - online: true - }) - .then(() => { - this._disconnect(); - - this._onConnect.forEach(fn => { - if (fn && typeof fn === 'function') { - fn.bind(this)(this.ref); - } - }) + this.ref.set({ + online: true, }) + .then(() => { + this._disconnect(); + + // todo switch to event emitter + // todo this will leak + this._onConnect.forEach((fn) => { + if (fn && typeof fn === 'function') { + fn.bind(this)(this.ref); + } + }); + }); } }); return this; @@ -43,8 +41,7 @@ class PresenceRef extends ReferenceBase { setOffline() { if (this.ref) { - this.ref.setAt({online: false}) - .then(() => this.ref.off('value')) + this.ref.set({ online: false }).then(() => this.ref.off('value')); this.presence.off(this._pathParts); } return this; @@ -52,11 +49,9 @@ class PresenceRef extends ReferenceBase { _disconnect() { if (this.ref) { - this.ref.onDisconnect() - .setValue({online: false}); - // set last online time - this.lastOnlineRef.onDisconnect() - .setValue(this.firestack.ServerValue.TIMESTAMP) + this.ref.onDisconnect().setValue({ online: false }); + // todo ServerValue is a promise? so this should be broken..? + this.lastOnlineRef.onDisconnect().setValue(this.firestack.ServerValue.TIMESTAMP); } } @@ -64,6 +59,8 @@ class PresenceRef extends ReferenceBase { return this._pathParts.join('/'); } + // todo switch to event emitter + // todo this will leak onConnect(cb) { this._onConnect.push(cb); return this; @@ -71,24 +68,21 @@ class PresenceRef extends ReferenceBase { } -export class Presence extends Base { - constructor(firestack, options={}) { +export default class Presence extends Base { + constructor(firestack, options = {}) { super(firestack, options); - this.instances = {}; this.path = ['/presence/connections']; } on(key) { - invariant(key, 'You must supply a key for presence'); + if (!key || !key.length) throw new Error('You must supply a key for presence'); const path = this.path.concat(key); const pathKey = this._presenceKey(path); if (!this.instances[pathKey]) { - const _ref = this.firestack.database.ref(pathKey); - this.log.debug('Created new presence object for ', pathKey) - const inst = new PresenceRef(this, _ref, path); - - this.instances[pathKey] = inst; + const _ref = this.firestack.database().ref(pathKey); + this.log.debug('Created new presence object for ', pathKey); + this.instances[pathKey] = new PresenceRef(this, _ref, path); } return this.instances[pathKey]; @@ -110,8 +104,6 @@ export class Presence extends Base { } get namespace() { - return 'firestack:presence' + return 'firestack:presence'; } } - -export default Presence; \ No newline at end of file diff --git a/lib/modules/remoteConfig.js b/lib/modules/remoteConfig.js index 385caf5..cd17d64 100644 --- a/lib/modules/remoteConfig.js +++ b/lib/modules/remoteConfig.js @@ -2,19 +2,20 @@ * Configuration class */ const defaultExpiration = 60 * 60 * 24; // one day -export class RemoteConfig { + +export default class RemoteConfig { constructor(options) { this.config = options || {}; this.setDefaultRemoteConfig(options) - .then(() => this.configured = true); + .then(() => this.configured = true); } setDefaultRemoteConfig(options) { return promisify('setDefaultRemoteConfig')(options); } - fetchWithExpiration(expirationSeconds=defaultExpiration) { + fetchWithExpiration(expirationSeconds = defaultExpiration) { return promisify('fetchWithExpiration')(expirationSeconds) } @@ -26,5 +27,3 @@ export class RemoteConfig { return promisify('setDev')(); } } - -export default RemoteConfig; \ No newline at end of file diff --git a/lib/modules/storage.js b/lib/modules/storage.js deleted file mode 100644 index 6e5f46a..0000000 --- a/lib/modules/storage.js +++ /dev/null @@ -1,134 +0,0 @@ - -import {NativeModules, NativeEventEmitter} from 'react-native'; -const FirestackStorage = NativeModules.FirestackStorage; -const FirestackStorageEvt = new NativeEventEmitter(FirestackStorage); - -import promisify from '../utils/promisify' -import { Base, ReferenceBase } from './base' - -class StorageRef extends ReferenceBase { - constructor(storage, path) { - super(storage.firestack, path); - - this.storage = storage; - } - - downloadUrl() { - const path = this.pathToString(); - return promisify('downloadUrl', FirestackStorage)(this.storage.storageUrl, path); - } - - /** - * Downloads a reference to the device - * @param {String} downloadPath Where to store the file - * @return {Promise} - */ - download (downloadPath, cb) { - let callback = cb; - if (!callback || typeof callback !== 'function') { - callback = (evt) => {}; - } - - const listeners = []; - listeners.push(this.storage._addListener('download_progress', callback)); - listeners.push(this.storage._addListener('download_paused', callback)); - listeners.push(this.storage._addListener('download_resumed', callback)); - - const path = this.pathToString(); - return promisify('downloadFile', FirestackStorage)(this.storage.storageUrl, path, downloadPath) - .then((res) => { - console.log('res --->', res); - listeners.forEach(this.storage._removeListener); - return res; - }) - .catch(err => { - console.log('Got an error ->', err); - }) - } -} - -export class Storage extends Base { - constructor(firestack, options={}) { - super(firestack, options); - - if (this.options.storageBucket) { - this.setStorageUrl(this.options.storageBucket); - } - - this.refs = {}; - } - - ref(...path) { - const key = this._pathKey(path); - if (!this.refs[key]) { - const ref = new StorageRef(this, path); - this.refs[key] = ref; - } - return this.refs[key]; - } - - /** - * Upload a filepath - * @param {string} name The destination for the file - * @param {string} filepath The local path of the file - * @param {object} metadata An object containing metadata - * @return {Promise} - */ - uploadFile(name, filepath, metadata={}, cb) { - let callback = cb; - if (!callback || typeof callback !== 'function') { - callback = (evt) => {} - } - - filepath = filepath.replace("file://", ""); - - const listeners = []; - listeners.push(this._addListener('upload_progress', callback)); - listeners.push(this._addListener('upload_paused', callback)); - listeners.push(this._addListener('upload_resumed', callback)); - return promisify('uploadFile', FirestackStorage)(this.storageUrl, name, filepath, metadata) - .then((res) => { - listeners.forEach(this._removeListener); - return res; - }); - } - - getRealPathFromURI(uri) { - return promisify('getRealPathFromURI', FirestackStorage)(uri); - } - - _addListener(evt, cb) { - return FirestackStorageEvt.addListener(evt, cb); - } - - _removeListener(evt) { - return FirestackStorageEvt.removeListener(evt); - } - - setStorageUrl(url) { - // return promisify('setStorageUrl', FirestackStorage)(url); - this.storageUrl = `gs://${url}`; - } - - _pathKey(...path) { - return path.join('-'); - } - - static constants = { - 'MAIN_BUNDLE_PATH': FirestackStorage.MAIN_BUNDLE_PATH, - 'CACHES_DIRECTORY_PATH': FirestackStorage.CACHES_DIRECTORY_PATH, - 'DOCUMENT_DIRECTORY_PATH': FirestackStorage.DOCUMENT_DIRECTORY_PATH, - 'EXTERNAL_DIRECTORY_PATH': FirestackStorage.EXTERNAL_DIRECTORY_PATH, - 'EXTERNAL_STORAGE_DIRECTORY_PATH': FirestackStorage.EXTERNAL_STORAGE_DIRECTORY_PATH, - 'TEMP_DIRECTORY_PATH': FirestackStorage.TEMP_DIRECTORY_PATH, - 'LIBRARY_DIRECTORY_PATH': FirestackStorage.LIBRARY_DIRECTORY_PATH, - 'FILETYPE_REGULAR': FirestackStorage.FILETYPE_REGULAR, - 'FILETYPE_DIRECTORY': FirestackStorage.FILETYPE_DIRECTORY - }; - - get namespace() { - return 'firestack:storage' - } -} - -export default Storage diff --git a/lib/modules/storage/index.js b/lib/modules/storage/index.js new file mode 100644 index 0000000..f3f2417 --- /dev/null +++ b/lib/modules/storage/index.js @@ -0,0 +1,112 @@ +/* @flow */ +import { NativeModules, NativeEventEmitter } from 'react-native'; + +import { Base } from './../base'; +import StorageRef from './reference'; +import { promisify, noop } from './../../utils'; + +const FirestackStorage = NativeModules.FirestackStorage; +const FirestackStorageEvt = new NativeEventEmitter(FirestackStorage); + +type StorageOptionsType = { + storageBucket?: ?string, +}; + +export default class Storage extends Base { + constructor(firestack: Object, options: StorageOptionsType = {}) { + super(firestack, options); + this.subscriptions = {}; + + this.successListener = FirestackStorageEvt.addListener( + 'storage_event', + event => this._handleStorageEvent(event) + ); + + this.errorListener = FirestackStorageEvt.addListener( + 'storage_error', + err => this._handleStorageError(err) + ); + } + + ref(path: string): StorageRef { + return new StorageRef(this, path); + } + + refFromURL(url: string): Promise<StorageRef> { + return new StorageRef(this, `url::${url}`); + } + + setMaxOperationRetryTime(time: number) { + FirestackStorage.setMaxOperationRetryTime(time); + } + + setMaxUploadRetryTime(time: number) { + FirestackStorage.setMaxUploadRetryTime(time); + } + + //Additional methods compared to Web API + setMaxDownloadRetryTime(time: number) { + FirestackStorage.setMaxDownloadRetryTime(time); + } + + _handleStorageEvent(event: Object) { + const { path, eventName } = event; + const body = event.body || {}; + + this.log.debug('_handleStorageEvent: ', path, eventName, body); + + if (this.subscriptions[path] && this.subscriptions[path][eventName]) { + this.subscriptions[path][eventName].forEach((cb) => { + cb(body); + }) + } + } + + _handleStorageError(err: Object) { + this.log.debug('_handleStorageError ->', err); + } + + _addListener(path: string, eventName: string, cb: (evt: Object) => Object) { + if (!this.subscriptions[path]) this.subscriptions[path] = {}; + if (!this.subscriptions[path][eventName]) this.subscriptions[path][eventName] = []; + this.subscriptions[path][eventName].push(cb); + } + + _removeListener(path: string, eventName: string, origCB: (evt: Object) => Object) { + if (!this.subscriptions[path] || (eventName && !this.subscriptions[path][eventName])) { + this.log.warn('_removeListener() called, but not currently listening at that location (bad path)', path, eventName); + return Promise.resolve(); + } + + if (eventName && origCB) { + const i = this.subscriptions[path][eventName].indexOf(origCB); + + if (i === -1) { + this.log.warn('_removeListener() called, but the callback specified is not listening at this location (bad path)', path, eventName); + } else { + this.subscriptions[path][eventName].splice(i, 1); + } + } else if (eventName) { + this.subscriptions[path][eventName] = []; + } else { + this.subscriptions[path] = {} + } + } + + static constants = { + MAIN_BUNDLE_PATH: FirestackStorage.MAIN_BUNDLE_PATH, + CACHES_DIRECTORY_PATH: FirestackStorage.CACHES_DIRECTORY_PATH, + DOCUMENT_DIRECTORY_PATH: FirestackStorage.DOCUMENT_DIRECTORY_PATH, + EXTERNAL_DIRECTORY_PATH: FirestackStorage.EXTERNAL_DIRECTORY_PATH, + EXTERNAL_STORAGE_DIRECTORY_PATH: FirestackStorage.EXTERNAL_STORAGE_DIRECTORY_PATH, + TEMP_DIRECTORY_PATH: FirestackStorage.TEMP_DIRECTORY_PATH, + LIBRARY_DIRECTORY_PATH: FirestackStorage.LIBRARY_DIRECTORY_PATH, + FILETYPE_REGULAR: FirestackStorage.FILETYPE_REGULAR, + FILETYPE_DIRECTORY: FirestackStorage.FILETYPE_DIRECTORY, + }; + + get namespace(): string { + return 'firestack:storage'; + } +} + diff --git a/lib/modules/storage/reference.js b/lib/modules/storage/reference.js new file mode 100644 index 0000000..6abfe99 --- /dev/null +++ b/lib/modules/storage/reference.js @@ -0,0 +1,143 @@ +/* @flow */ +import { NativeModules } from 'react-native'; + +import { promisify, noop } from '../../utils'; +import { ReferenceBase } from './../base'; +import Storage from './'; + +const FirestackStorage = NativeModules.FirestackStorage; + +export default class StorageRef extends ReferenceBase { + constructor(storage: Storage, path: string) { + super(storage.firestack, path); + this.storage = storage; + } + + child(path: string) { + return new StorageRef(this.storage, this.path + '/' + path); + } + + delete(): Promise { + return promisify('delete', FirestackStorage)(this.path) + .catch(err => { + this.log.error('Error deleting reference ', this.path, '. Error: ', err); + throw err; + }) + } + + getDownloadURL(): Promise<String> { + this.log.debug('getDownloadURL(', this.path, ')'); + return promisify('getDownloadURL', FirestackStorage)(this.path) + .catch((err) => { + this.log.error('Error downloading URL for ', this.path, '. Error: ', err); + throw err; + }); + } + + getMetadata(): Promise<Object> { + return promisify('getMetadata', FirestackStorage)(this.path) + .catch(err => { + this.log.error('Error getting metadata for ', this.path, '. Error: ', err); + throw err; + }) + } + + //TODO: Figure out the best way to do this on iOS/Android + put(data: Object, metadata: Object = {}): /*UploadTask*/Promise<Object> { + throw new Error('put() is not currently supported by react-native-firestack') + } + + //TODO: Figure out the best way to do this on iOS/Android + putString(data: string, format: String, metadata: Object = {}): /*UploadTask*/Promise<Object> { + throw new Error('putString() is not currently supported by react-native-firestack') + } + + toString(): String { + //TODO: Return full gs://bucket/path + return this.path; + } + + updateMetadata(metadata: Object = {}): Promise<Object> { + return promisify('updateMetadata', FirestackStorage)(this.path, metadata) + .catch(err => { + this.log.error('Error updating metadata for ', this.path, '. Error: ', err); + throw err; + }) + } + + //Additional methods compared to Web API + /** + * Downloads a reference to the device + * @param {String} filePath Where to store the file + * @return {Promise} + */ + downloadFile(filePath: string): Promise<Object> { + this.log.debug('download(', this.path, ') -> ', filePath); + + let downloadTask = promisify('downloadFile', FirestackStorage)(this.path, filePath); + downloadTask.cancel = () => { + //TODO + throw new Error('.cancel() is not currently supported by react-native-firestack'); + } + downloadTask.on = (event, nextOrObserver, error, complete) => { + //TODO: nextOrObserver as an object + if (nextOrObserver) this.storage._addListener(this.path, 'state_changed', nextOrObserver); + if (error) this.storage._addListener(this.path, 'download_failure', error); + if (complete) this.storage._addListener(this.path, 'download_success', complete); + return () => { + if (nextOrObserver) this.storage._removeListener(this.path, 'state_changed', nextOrObserver); + if (error) this.storage._removeListener(this.path, 'download_failure', error); + if (complete) this.storage._removeListener(this.path, 'download_success', complete); + } + } + downloadTask.pause = () => { + //TODO + throw new Error('.pause() is not currently supported by react-native-firestack'); + } + downloadTask.resume = () => { + //TODO + throw new Error('.resume() is not currently supported by react-native-firestack'); + } + + return downloadTask; + } + + /** + * Upload a file path + * @param {string} filePath The local path of the file + * @param {object} metadata An object containing metadata + * @return {Promise} + */ + putFile(filePath: Object, metadata: Object = {}): Promise<Object> { + const _filePath = filePath.replace('file://', ''); + this.log.debug('putFile(', _filePath, ') -> ', this.path); + + //TODO: There's probably a better way of doing this, but I couldn't figure out the best way to extend a promise + let uploadTask = promisify('putFile', FirestackStorage)(this.path, _filePath, metadata); + uploadTask.cancel = () => { + //TODO + throw new Error('.cancel() is not currently supported by react-native-firestack'); + } + uploadTask.on = (event, nextOrObserver, error, complete) => { + //TODO: nextOrObserver as an object + if (nextOrObserver) this.storage._addListener(this.path, 'state_changed', nextOrObserver); + if (error) this.storage._addListener(this.path, 'upload_failure', error); + if (complete) this.storage._addListener(this.path, 'upload_success', complete); + return () => { + if (nextOrObserver) this.storage._removeListener(this.path, 'state_changed', nextOrObserver); + if (error) this.storage._removeListener(this.path, 'upload_failure', error); + if (complete) this.storage._removeListener(this.path, 'upload_success', complete); + } + } + uploadTask.pause = () => { + //TODO + throw new Error('.pause() is not currently supported by react-native-firestack'); + } + uploadTask.resume = () => { + //TODO + throw new Error('.resume() is not currently supported by react-native-firestack'); + } + + return uploadTask; + } +} diff --git a/lib/modules/user.js b/lib/modules/user.js new file mode 100644 index 0000000..a86f36f --- /dev/null +++ b/lib/modules/user.js @@ -0,0 +1,127 @@ +// TODO refreshToken property +// TODO reload() method +export default class User { + constructor(authClass, authObj) { + this._auth = authClass; + this._user = null; + this._updateValues(authObj); + } + + /** + * INTERNALS + */ + + /** + * + * @param authObj + * @private + */ + _updateValues(authObj) { + this._authObj = authObj; + if (authObj.user) { + this._user = authObj.user; + } else { + this._user = null; + } + } + + /** + * Returns a user property or null if does not exist + * @param prop + * @returns {*} + * @private + */ + _valueOrNull(prop) { + if (!this._user) return null; + if (!Object.hasOwnProperty.call(this._user, prop)) return null; + return this._user[prop]; + } + + /** + * PROPERTIES + */ + + get displayName() { + return this._valueOrNull('displayName'); + } + + get email() { + return this._valueOrNull('email'); + } + + get emailVerified() { + return this._valueOrNull('emailVerified'); + } + + get isAnonymous() { + return !this._valueOrNull('email') && this._valueOrNull('providerId') === 'firebase'; + } + + get photoURL() { + return this._valueOrNull('photoURL'); + } + + get photoUrl() { + return this._valueOrNull('photoURL'); + } + + // TODO no android method yet, the SDK does have .getProviderData but returns as a List. + // get providerData() { + // return this._valueOrNull('providerData'); + // } + + get providerId() { + return this._valueOrNull('providerId'); + } + + // TODO no android method? + // get refreshToken() { + // return this._valueOrNull('refreshToken'); + // } + + get uid() { + return this._valueOrNull('uid'); + } + + // noinspection ReservedWordAsName + /** + * METHODS + */ + + delete(...args) { + return this._auth.deleteUser(...args); + } + + reload(...args) { + return this._auth.reloadUser(...args); + } + + // TODO valueOrNul token - optional promise + getToken(...args) { + return this._auth.getToken(...args); + } + + get reauthenticate() { + return this._auth.reauthenticateUser; + } + + // TODO match errors to auth/something errors from firebase web api + get updateEmail() { + if (this.isAnonymous) return () => Promise.reject(new Error('Can not update email on an anonymous user.')); + return this._auth.updateEmail; + } + + get updateProfile() { + return this._auth.updateProfile; + } + + get updatePassword() { + if (this.isAnonymous) return () => Promise.reject(new Error('Can not update password on an anonymous user.')); + return this._auth.updatePassword; + } + + get sendEmailVerification() { + if (this.isAnonymous) return () => Promise.reject(new Error('Can not verify email on an anonymous user.')); + return this._auth.sendEmailVerification; + } +} diff --git a/lib/utils/eventEmitter.js b/lib/utils/eventEmitter.js new file mode 100644 index 0000000..67c15ba --- /dev/null +++ b/lib/utils/eventEmitter.js @@ -0,0 +1,313 @@ +// TODO - this is just a raw copy of eventEmitter3 - until i can implement a lightweight version + +'use strict'; + +var has = Object.prototype.hasOwnProperty + , prefix = '~'; + +/** + * Constructor to create a storage for our `EE` objects. + * An `Events` instance is a plain object whose properties are event names. + * + * @constructor + * @api private + */ +function Events() {} + +// +// We try to not inherit from `Object.prototype`. In some engines creating an +// instance in this way is faster than calling `Object.create(null)` directly. +// If `Object.create(null)` is not supported we prefix the event names with a +// character to make sure that the built-in object properties are not +// overridden or used as an attack vector. +// +if (Object.create) { + Events.prototype = Object.create(null); + + // + // This hack is needed because the `__proto__` property is still inherited in + // some old browsers like Android 4, iPhone 5.1, Opera 11 and Safari 5. + // + if (!new Events().__proto__) prefix = false; +} + +/** + * Representation of a single event listener. + * + * @param {Function} fn The listener function. + * @param {Mixed} context The context to invoke the listener with. + * @param {Boolean} [once=false] Specify if the listener is a one-time listener. + * @constructor + * @api private + */ +function EE(fn, context, once) { + this.fn = fn; + this.context = context; + this.once = once || false; +} + +/** + * Minimal `EventEmitter` interface that is molded against the Node.js + * `EventEmitter` interface. + * + * @constructor + * @api public + */ +function EventEmitter() { + this._events = new Events(); + this._eventsCount = 0; +} + +/** + * Return an array listing the events for which the emitter has registered + * listeners. + * + * @returns {Array} + * @api public + */ +EventEmitter.prototype.eventNames = function eventNames() { + var names = [] + , events + , name; + + if (this._eventsCount === 0) return names; + + for (name in (events = this._events)) { + if (has.call(events, name)) names.push(prefix ? name.slice(1) : name); + } + + if (Object.getOwnPropertySymbols) { + return names.concat(Object.getOwnPropertySymbols(events)); + } + + return names; +}; + +/** + * Return the listeners registered for a given event. + * + * @param {String|Symbol} event The event name. + * @param {Boolean} exists Only check if there are listeners. + * @returns {Array|Boolean} + * @api public + */ +EventEmitter.prototype.listeners = function listeners(event, exists) { + var evt = prefix ? prefix + event : event + , available = this._events[evt]; + + if (exists) return !!available; + if (!available) return []; + if (available.fn) return [available.fn]; + + for (var i = 0, l = available.length, ee = new Array(l); i < l; i++) { + ee[i] = available[i].fn; + } + + return ee; +}; + +/** + * Calls each of the listeners registered for a given event. + * + * @param {String|Symbol} event The event name. + * @returns {Boolean} `true` if the event had listeners, else `false`. + * @api public + */ +EventEmitter.prototype.emit = function emit(event, a1, a2, a3, a4, a5) { + var evt = prefix ? prefix + event : event; + + if (!this._events[evt]) return false; + + var listeners = this._events[evt] + , len = arguments.length + , args + , i; + + if (listeners.fn) { + if (listeners.once) this.removeListener(event, listeners.fn, undefined, true); + + switch (len) { + case 1: return listeners.fn.call(listeners.context), true; + case 2: return listeners.fn.call(listeners.context, a1), true; + case 3: return listeners.fn.call(listeners.context, a1, a2), true; + case 4: return listeners.fn.call(listeners.context, a1, a2, a3), true; + case 5: return listeners.fn.call(listeners.context, a1, a2, a3, a4), true; + case 6: return listeners.fn.call(listeners.context, a1, a2, a3, a4, a5), true; + } + + for (i = 1, args = new Array(len -1); i < len; i++) { + args[i - 1] = arguments[i]; + } + + listeners.fn.apply(listeners.context, args); + } else { + var length = listeners.length + , j; + + for (i = 0; i < length; i++) { + if (listeners[i].once) this.removeListener(event, listeners[i].fn, undefined, true); + + switch (len) { + case 1: listeners[i].fn.call(listeners[i].context); break; + case 2: listeners[i].fn.call(listeners[i].context, a1); break; + case 3: listeners[i].fn.call(listeners[i].context, a1, a2); break; + case 4: listeners[i].fn.call(listeners[i].context, a1, a2, a3); break; + default: + if (!args) for (j = 1, args = new Array(len -1); j < len; j++) { + args[j - 1] = arguments[j]; + } + + listeners[i].fn.apply(listeners[i].context, args); + } + } + } + + return true; +}; + +/** + * Add a listener for a given event. + * + * @param {String|Symbol} event The event name. + * @param {Function} fn The listener function. + * @param {Mixed} [context=this] The context to invoke the listener with. + * @returns {EventEmitter} `this`. + * @api public + */ +EventEmitter.prototype.on = function on(event, fn, context) { + var listener = new EE(fn, context || this) + , evt = prefix ? prefix + event : event; + + if (!this._events[evt]) this._events[evt] = listener, this._eventsCount++; + else if (!this._events[evt].fn) this._events[evt].push(listener); + else this._events[evt] = [this._events[evt], listener]; + + return this; +}; + +/** + * Add a one-time listener for a given event. + * + * @param {String|Symbol} event The event name. + * @param {Function} fn The listener function. + * @param {Mixed} [context=this] The context to invoke the listener with. + * @returns {EventEmitter} `this`. + * @api public + */ +EventEmitter.prototype.once = function once(event, fn, context) { + var listener = new EE(fn, context || this, true) + , evt = prefix ? prefix + event : event; + + if (!this._events[evt]) this._events[evt] = listener, this._eventsCount++; + else if (!this._events[evt].fn) this._events[evt].push(listener); + else this._events[evt] = [this._events[evt], listener]; + + return this; +}; + +/** + * Remove the listeners of a given event. + * + * @param {String|Symbol} event The event name. + * @param {Function} fn Only remove the listeners that match this function. + * @param {Mixed} context Only remove the listeners that have this context. + * @param {Boolean} once Only remove one-time listeners. + * @returns {EventEmitter} `this`. + * @api public + */ +EventEmitter.prototype.removeListener = function removeListener(event, fn, context, once) { + var evt = prefix ? prefix + event : event; + + if (!this._events[evt]) return this; + if (!fn) { + if (--this._eventsCount === 0) this._events = new Events(); + else delete this._events[evt]; + return this; + } + + var listeners = this._events[evt]; + + if (listeners.fn) { + if ( + listeners.fn === fn + && (!once || listeners.once) + && (!context || listeners.context === context) + ) { + if (--this._eventsCount === 0) this._events = new Events(); + else delete this._events[evt]; + } + } else { + for (var i = 0, events = [], length = listeners.length; i < length; i++) { + if ( + listeners[i].fn !== fn + || (once && !listeners[i].once) + || (context && listeners[i].context !== context) + ) { + events.push(listeners[i]); + } + } + + // + // Reset the array, or remove it completely if we have no more listeners. + // + if (events.length) this._events[evt] = events.length === 1 ? events[0] : events; + else if (--this._eventsCount === 0) this._events = new Events(); + else delete this._events[evt]; + } + + return this; +}; + +/** + * Remove all listeners, or those of the specified event. + * + * @param {String|Symbol} [event] The event name. + * @returns {EventEmitter} `this`. + * @api public + */ +EventEmitter.prototype.removeAllListeners = function removeAllListeners(event) { + var evt; + + if (event) { + evt = prefix ? prefix + event : event; + if (this._events[evt]) { + if (--this._eventsCount === 0) this._events = new Events(); + else delete this._events[evt]; + } + } else { + this._events = new Events(); + this._eventsCount = 0; + } + + return this; +}; + +// +// Alias methods names because people roll like that. +// +EventEmitter.prototype.off = EventEmitter.prototype.removeListener; +EventEmitter.prototype.addListener = EventEmitter.prototype.on; + +// +// This function doesn't apply anymore. +// +EventEmitter.prototype.setMaxListeners = function setMaxListeners() { + return this; +}; + +// +// Expose the prefix. +// +EventEmitter.prefixed = prefix; + +// +// Allow `EventEmitter` to be imported as module namespace. +// +EventEmitter.EventEmitter = EventEmitter; + +// +// Expose the module. +// +if ('undefined' !== typeof module) { + module.exports = EventEmitter; +} diff --git a/lib/utils/index.js b/lib/utils/index.js new file mode 100644 index 0000000..02f8f21 --- /dev/null +++ b/lib/utils/index.js @@ -0,0 +1,278 @@ +// modeled after base64 web-safe chars, but ordered by ASCII +const PUSH_CHARS = '-0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ_abcdefghijklmnopqrstuvwxyz'; +const hasOwnProperty = Object.hasOwnProperty; +const DEFAULT_CHUNK_SIZE = 50; + +// internal promise handler +const _handler = (resolve, reject, errorPrefix, err, resp) => { + // resolve / reject after events etc + setImmediate(() => { + if (err) return reject(errorPrefix ? { code: toWebSDKErrorCode(err.code || err.errCode || err.errorCode || '', errorPrefix), message: err.message } : err); + return resolve(resp); + }); +}; + +export function toWebSDKErrorCode(code, prefix) { + return code.toLowerCase().replace('error_', prefix).replace(/_/g, '-'); +} + +/** + * Deep get a value from an object. + * @website https://github.com/Salakar/deeps + * @param object + * @param path + * @param joiner + * @returns {*} + */ +export function deepGet(object, path, joiner = '/') { + const keys = path.split(joiner); + + let i = 0; + let tmp = object; + const len = keys.length; + + while (i < len) { + const key = keys[i++]; + if (!tmp || !hasOwnProperty.call(tmp, key)) return null; + tmp = tmp[key]; + } + + return tmp; +} + +/** + * Deep check if a key exists. + * @website https://github.com/Salakar/deeps + * @param object + * @param path + * @param joiner + * @returns {*} + */ +export function deepExists(object, path, joiner = '/') { + const keys = path.split(joiner); + + let i = 0; + let tmp = object; + const len = keys.length; + + while (i < len) { + const key = keys[i++]; + if (!tmp || !hasOwnProperty.call(tmp, key)) return false; + tmp = tmp[key]; + } + + return tmp !== undefined; +} + +/** + * Simple is object check. + * @param item + * @returns {boolean} + */ +export function isObject(item) { + return (item && typeof item === 'object' && !Array.isArray(item) && item !== null); +} + +/** + * Simple is function check + * @param item + * @returns {*|boolean} + */ +export function isFunction(item) { + return (item && typeof item === 'function'); +} + +/** + * + * @param string + * @returns {*} + */ +export function tryJSONParse(string) { + try { + return JSON.parse(string); + } catch (jsonError) { + return string; + } +} + +/** + * + * @param data + * @returns {*} + */ +export function tryJSONStringify(data) { + try { + return JSON.stringify(data); + } catch (jsonError) { + return null; + } +} + + +// noinspection Eslint +export const windowOrGlobal = (typeof self === 'object' && self.self === self && self) || (typeof global === 'object' && global.global === global && global) || this; + +/** + * Makes an objects keys it's values + * @param object + * @returns {{}} + */ +export function reverseKeyValues(object: Object): Object { + const output = {}; + for (const key in object) { + output[object[key]] = key; + } + return output; +} + +/** + * No operation func + */ +export function noop(): void { +} + +/** + * Wraps a native module method to support promises. + * @param fn + * @param NativeModule + * @param errorPrefix + */ +export function promisify(fn: Function, NativeModule: Object, errorPrefix): Function<Promise> { + return (...args) => { + return new Promise((resolve, reject) => { + const _fn = typeof fn === 'function' ? fn : NativeModule[fn]; + if (!_fn || typeof _fn !== 'function') return reject(new Error('Missing function for promisify.')); + return _fn.apply(NativeModule, [...args, _handler.bind(_handler, resolve, reject, errorPrefix)]); + }); + }; +} + + +/** + * Delays chunks based on sizes per event loop. + * @param collection + * @param chunkSize + * @param operation + * @param callback + * @private + */ +function _delayChunk(collection, chunkSize, operation, callback): void { + const length = collection.length; + const iterations = Math.ceil(length / chunkSize); + + // noinspection ES6ConvertVarToLetConst + let thisIteration = 0; + + setImmediate(function next() { + const start = thisIteration * chunkSize; + const _end = start + chunkSize; + const end = _end >= length ? length : _end; + const result = operation(collection.slice(start, end), start, end); + + if (thisIteration++ > iterations) { + callback(null, result); + } else { + setImmediate(next); + } + }); +} + +/** + * Async each with optional chunk size limit + * @param array + * @param chunkSize + * @param iterator + * @param cb + */ +export function each(array: Array, chunkSize?: number, iterator: Function, cb: Function): void { + if (typeof chunkSize === 'function') { + cb = iterator; + iterator = chunkSize; + chunkSize = DEFAULT_CHUNK_SIZE; + } + + _delayChunk(array, chunkSize, (slice, start) => { + for (let ii = 0, jj = slice.length; ii < jj; ii += 1) { + iterator(slice[ii], start + ii); + } + }, cb); +} + +/** + * Async map with optional chunk size limit + * @param array + * @param chunkSize + * @param iterator + * @param cb + * @returns {*} + */ +export function map(array: Array, chunkSize?: number, iterator: Function, cb: Function): void { + if (typeof chunkSize === 'function') { + cb = iterator; + iterator = chunkSize; + chunkSize = DEFAULT_CHUNK_SIZE; + } + + const result = []; + _delayChunk(array, chunkSize, (slice, start) => { + for (let ii = 0, jj = slice.length; ii < jj; ii += 1) { + result.push(iterator(slice[ii], start + ii, array)); + } + return result; + }, () => cb(result)); +} + + +// timestamp of last push, used to prevent local collisions if you push twice in one ms. +let lastPushTime = 0; + +// we generate 72-bits of randomness which get turned into 12 characters and appended to the +// timestamp to prevent collisions with other clients. We store the last characters we +// generated because in the event of a collision, we'll use those same characters except +// "incremented" by one. +const lastRandChars = []; + +/** + * Generate a firebase id - for use with ref().push(val, cb) - e.g. -KXMr7k2tXUFQqiaZRY4' + * @param serverTimeOffset - pass in server time offset from native side + * @returns {string} + */ +export function generatePushID(serverTimeOffset?: number = 0): string { + const timeStampChars = new Array(8); + let now = new Date().getTime() + serverTimeOffset; + const duplicateTime = (now === lastPushTime); + + lastPushTime = now; + + for (let i = 7; i >= 0; i -= 1) { + timeStampChars[i] = PUSH_CHARS.charAt(now % 64); + now = Math.floor(now / 64); + } + + if (now !== 0) throw new Error('We should have converted the entire timestamp.'); + + let id = timeStampChars.join(''); + + if (!duplicateTime) { + for (let i = 0; i < 12; i += 1) { + lastRandChars[i] = Math.floor(Math.random() * 64); + } + } else { + // if the timestamp hasn't changed since last push, + // use the same random number, but increment it by 1. + let i; + for (i = 11; i >= 0 && lastRandChars[i] === 63; i -= 1) { + lastRandChars[i] = 0; + } + + lastRandChars[i] += 1; + } + + for (let i = 0; i < 12; i++) { + id += PUSH_CHARS.charAt(lastRandChars[i]); + } + + if (id.length !== 20) throw new Error('Length should be 20.'); + + return id; +} diff --git a/lib/utils/log.js b/lib/utils/log.js index 12b0258..3584ce1 100644 --- a/lib/utils/log.js +++ b/lib/utils/log.js @@ -1,44 +1,40 @@ -// document hack -import root from './window-or-global' +import { windowOrGlobal } from './'; -let bows; -(function (base) { - window = base || window - if(!window.localStorage) window.localStorage = {}; -})(root); +((base) => { + window = base || window; + if (!window.localStorage) window.localStorage = {}; +})(windowOrGlobal); -const levels = [ - 'warn', 'info', 'error', 'debug' -]; - -export class Log { +export default class Log { constructor(namespace) { this._namespace = namespace || 'firestack'; + require('bows').config({ padLength: 20 }); this.loggers = {}; - // Add the logging levels for each level - levels - .forEach(level => this[level] = (...args) => this._log(level)(...args)); } - static enable(booleanOrStringDebug) { - window.localStorage.debug = - typeof booleanOrStringDebug === 'string' ? - (booleanOrStringDebug === '*' ? true : booleanOrStringDebug) : - (booleanOrStringDebug instanceof RegExp ? booleanOrStringDebug.toString() : booleanOrStringDebug); + get warn() { + return this._createOrGetLogger('warn'); + } + get info() { + return this._createOrGetLogger('info'); + } + + get error() { + return this._createOrGetLogger('error'); + } + + get debug() { + return this._createOrGetLogger('debug'); + } + + static enable(booleanOrStringDebug) { + window.localStorage.debug = booleanOrStringDebug; window.localStorage.debugColors = !!window.localStorage.debug; } - _log(level) { - if (!this.loggers[level]) { - (function() { - const bows = require('bows'); - bows.config({ padLength: 20 }); - this.loggers[level] = bows(this._namespace, `[${level}]`); - }.bind(this))(); - } + _createOrGetLogger(level) { + if (!this.loggers[level]) this.loggers[level] = require('bows')(this._namespace, `[${level}]`); return this.loggers[level]; } } - -export default Log; \ No newline at end of file diff --git a/lib/utils/promisify.js b/lib/utils/promisify.js deleted file mode 100644 index b8fbd4b..0000000 --- a/lib/utils/promisify.js +++ /dev/null @@ -1,12 +0,0 @@ -export const promisify = (fn, NativeModule) => (...args) => { - return new Promise((resolve, reject) => { - const handler = (err, resp) => { - err ? reject(err) : resolve(resp); - } - args.push(handler); - (typeof fn === 'function' ? fn : NativeModule[fn]) - .call(NativeModule, ...args); - }); -}; - -export default promisify \ No newline at end of file diff --git a/lib/utils/window-or-global.js b/lib/utils/window-or-global.js deleted file mode 100644 index 7b64020..0000000 --- a/lib/utils/window-or-global.js +++ /dev/null @@ -1,5 +0,0 @@ -'use strict' -// https://github.com/purposeindustries/window-or-global -module.exports = (typeof self === 'object' && self.self === self && self) || - (typeof global === 'object' && global.global === global && global) || - {} \ No newline at end of file diff --git a/package.json b/package.json index e611094..6466074 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "react-native-firestack", - "version": "2.3.3", + "version": "3.0.0-rc.3", "author": "Ari Lerner <ari@fullstack.io> (https://fullstackreact.com)", "description": "A firebase v3 adapter", "main": "index", @@ -10,7 +10,8 @@ "dev": "npm run compile -- --watch", "lint": "eslint ./src", "publish_pages": "gh-pages -d public/", - "test": "./node_modules/.bin/mocha" + "test": "./node_modules/.bin/mocha", + "watchcpx": "echo 'See https://github.com/wix/wml for watching changes. \r\n'" }, "repository": { "type": "git", @@ -51,10 +52,19 @@ } }, "devDependencies": { + "babel-eslint": "^7.0.0", "babel-jest": "^14.1.0", "babel-preset-react-native": "^1.9.0", + "cpx": "^1.5.0", "debug": "^2.2.0", "enzyme": "^2.4.1", + "eslint": "^3.8.1", + "eslint-config-airbnb": "^12.0.0", + "eslint-plugin-flowtype": "^2.20.0", + "eslint-plugin-import": "^2.0.1", + "eslint-plugin-jsx-a11y": "^2.2.3", + "eslint-plugin-react": "^6.4.1", + "flow-bin": "^0.35.0", "jest": "^14.1.0", "jest-react-native": "^14.1.3", "mocha": "^3.0.2",