jeffcarbs / react-native-background-job

Schedule background jobs in React Native that run your JavaScript when your app is in the background/killed.

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

react-native-background-job npm version

Schedule background jobs that run your JavaScript when your app is in the background.

The jobs will run even if the app has been closed and, by default, also persists over restarts.

This library relies on React Native's HeadlessJS which is currently only supported on Android.

On the native side it uses JobScheduler which means that the jobs can't be scheduled exactly and for Android 23+ they fire at most once per 15 minutes +-5 minutes. JobSceduler was used since it seemed to be the most battery efficient way of scheduling background tasks. I'm open to pull requests that implement more exact scheduling.

Requirements

  • RN 0.36+
  • Android API 21+

Supported platforms

  • Android

Want iOS? Go in and vote for Headless JS to be implemented for iOS: Product pains

Getting started

$ yarn add react-native-background-job

or

$ npm install react-native-background-job --save

Mostly automatic installation

$ react-native link react-native-background-job

Manual installation

Android

  1. Open up android/app/src/main/java/[...]/MainActivity.java

    • Add import com.pilloxa.backgroundjob.BackgroundJobPackage; to the imports at the top of the file
    • Add new BackgroundJobPackage() to the list returned by the getPackages() method in MainApplication.java
  2. Append the following lines to android/settings.gradle:

        include ':react-native-background-job'
        project(':react-native-background-job').projectDir = new File(rootProject.projectDir, 	'../node_modules/react-native-background-job/android')
    
  3. Insert the following lines inside the dependencies block in android/app/build.gradle and bump the minSdkVersion to 21:

          compile project(':react-native-background-job')
    

Usage

The jobs have to be registered each time React Native starts, this is done using the register function. Since HeadlessJS does not mount any components the register function must be run outside of any class definitions (see example/index.android.js)

Registering the job does not mean that the job is scheduled, it just informs React Native that this job function should be tied to this jobKey. The job is then scheduled using the schedule function. The job will not fire while the app is in the foreground. This is since the job is run on the only JavaScript thread and if running the job when app is in the foreground it would freeze the app.

For a full example check out example/index.android.js

API

register

Registers jobs and the functions they should run.

This has to run on each initialization of React Native. Only doing this will not start running the job. It has to be scheduled by schedule to start running.

Parameters

  • obj Object
    • obj.jobKey string A unique key for the job
    • obj.job function The JS-function that will be run

Examples

import BackgroundJob from 'react-native-background-job';

const backgroundJob = {
 jobKey: "myJob",
 job: () => console.log("Running in background")
};

BackgroundJob.register(backgroundJob);

schedule

Schedules a new job.

This only has to be run once while register has to be run on each initialization of React Native.

Parameters

  • obj Object
    • obj.jobKey string A unique key for the job
    • obj.timeout number How long the JS job may run before being terminated by Android (in ms).
    • obj.period number? The frequency to run the job with (in ms). This number is not exact, Android may modify it to save batteries. Note: For Android > N, the minimum is 900 0000 (15 min). (optional, default 900000)
    • obj.persist boolean? If the job should persist over a device restart. (optional, default true)
    • obj.warn boolean? If a warning should be raised if overwriting a job that was already scheduled. (optional, default true)
    • obj.networkType number? Only run for specific network requirements, (not respected by pre Android N devices) docs (optional, default BackgroundJob.NETWORK_TYPE_NONE)
    • obj.requiresCharging boolean? Only run job when device is charging, (not respected by pre Android N devices) docs (optional, default false)
    • obj.requiresDeviceIdle boolean? Only run job when the device is idle, (not respected by pre Android N devices) docs (optional, default false)

Examples

import BackgroundJob from 'react-native-background-job';

const backgroundJob = {
 jobKey: "myJob",
 job: () => console.log("Running in background")
};

BackgroundJob.register(backgroundJob);

var backgroundSchedule = {
 jobKey: "myJob",
 timeout: 5000
}

BackgroundJob.schedule(backgroundSchedule);

getAll

Fetches all the currently scheduled jobs

Parameters

  • obj Object
    • obj.callback function (Array) A list of all the scheduled jobs will be passed to the callback

Examples

import BackgroundJob from 'react-native-background-job';

BackgroundJob.getAll({callback: (jobs) => console.log("Jobs:",jobs)});

cancel

Cancel a specific job

Parameters

  • obj Object
    • obj.jobKey string The unique key for the job
    • obj.warn boolean? If one tries to cancel a job that has not been scheduled it will warn (optional, default true)

Examples

import BackgroundJob from 'react-native-background-job';

BackgroundJob.cancel({jobKey: 'myJob'});

cancelAll

Cancels all the scheduled jobs

Examples

import BackgroundJob from 'react-native-background-job';

BackgroundJob.cancelAll();

setGlobalWarnings

Sets the global warning level

Parameters

Examples

import BackgroundJob from 'react-native-background-job';

BackgroundJob.setGlobalWarnings(false);

Debugging

If you are using Android API +25 you can manually trigger the jobs by using the following command in a terminal:

$ adb shell cmd jobscheduler run -f your.package jobIntId

jobIntId: is the hashed jobKey. Get that value by going to Java REPL and input:

"yourJobKey".hashCode();
// 1298333557

Troubleshooting

No task registered for key myJob

Make sure you call the register function at the global scope (i.e. not in any component life cycle methods (render, iDidMount etc)). Since the components are not rendered in Headless mode if you run the register function there it will not run in the background and hence the library will not find which function to run.

See example project

AppState.currentState is "active" when I'm running my Headless task in the background

This is a React Native issue, you can get around it by calling NativeModules.AppState.getCurrentAppState directly instead.

My job always runs in the background even if i specified requiresCharging, requiresDeviceIdle or a specific networkType

This is an Android issue, it seems that you can not have these restrictions at the same time as you have a periodic interval for pre Android N devices.

Sponsored by

pilloxa

About

Schedule background jobs in React Native that run your JavaScript when your app is in the background/killed.


Languages

Language:JavaScript 43.8%Language:Java 35.7%Language:Objective-C 15.1%Language:Python 5.3%