Skip to content

Commit 0082a4e

Browse files
authored
Fix/243 bg task processing (fluttercommunity#279)
* Activate iOS BG Task Processing
1 parent 93053da commit 0082a4e

21 files changed

+475
-90
lines changed
Loading

ANDROID_SETUP.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ If for some reason you can't upgrade yet we still support the [older way of embe
1717
Debugging a background task can be difficult, Android decides when is the best time to run.
1818
There is no guaranteed way to enforce a run of a job even in debug mode.
1919

20-
However to facilitate debugging, the plugin provides an `isInDebugMode` flag when initializing the plugin: `Workmanager.initialize(callbackDispatcher, isInDebugMode: true)`
20+
However to facilitate debugging, the plugin provides an `isInDebugMode` flag when initializing the plugin: `Workmanager().initialize(callbackDispatcher, isInDebugMode: true)`
2121

2222
Once this flag is enabled you will receive a notification whenever a background task was triggered.
2323
This way you can keep track whether that task ran successfully or not.

CHANGELOG.md

+9
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,12 @@
1+
# 0.5.0-dev.3
2+
3+
* Documentation for BGTaskScheduler added
4+
* Throw standard errors when scheduling a task on iOS failed
5+
6+
# 0.5.0-dev.2
7+
8+
* iOS: Modern-style task processing using BGTaskScheduler is now supported. Please see the updated instructions in IOS_SETUP.md.
9+
110
# 0.5.0-dev.1
211

312
* Android: Load Flutter environment asynchronously in the Worker task

IOS_SETUP.md

+51-3
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,57 @@
44

55
This plugin is compatible with **Swift 4.2** and up. Make sure you are using **Xcode 10.3** or higher and have set your minimum deployment target to **iOS 10** or higher by defining a platform version in your podfile: `platform :ios, '10.0'`
66

7+
8+
## Enable BGTaskScheduler
9+
10+
> ⚠️ BGTaskScheduler is similar to Background Fetch described below and brings a similar set of constraints. Most notably, there are no guarantees when the background task will be run. Excerpt from the documentation:
11+
>
12+
> Schedule a processing task request to ask that the system launch your app when conditions are favorable for battery life to handle deferrable, longer-running processing, such as syncing, database maintenance, or similar tasks. The system will attempt to fulfill this request to the best of its ability within the next two days as long as the user has used your app within the past week.
13+
14+
![Screenshot of Background Fetch Capabilities tab in Xcode ](.art/ios_background_mode_background_processing.png)
15+
16+
This will add the **UIBackgroundModes** key to your project's `Info.plist`:
17+
18+
``` xml
19+
<key>UIBackgroundModes</key>
20+
<array>
21+
<string>processing</string>
22+
</array>
23+
```
24+
25+
Additionally, you must configure the background task identifiers. The default identifier is `workmanager.background.task` in the host Apps Info.plist:
26+
27+
``` xml
28+
<key>BGTaskSchedulerPermittedIdentifiers</key>
29+
<array>
30+
<string>workmanager.background.task</string>
31+
</array>
32+
</plist>
33+
```
34+
35+
And will set the correct *SystemCapabilities* for your target in the `project.pbxproj` file:
36+
37+
```
38+
SystemCapabilities = {
39+
com.apple.BackgroundModes = {
40+
enabled = 1;
41+
};
42+
};
43+
```
44+
45+
## Testing BGTaskScheduler
46+
47+
Follow the instructions on https://developer.apple.com/documentation/backgroundtasks/starting_and_terminating_tasks_during_development.
48+
49+
The exact command to trigger the WorkManager default BG Task is:
50+
51+
```
52+
e -l objc -- (void)[[BGTaskScheduler sharedScheduler] _simulateLaunchForTaskWithIdentifier:@"workmanager.background.task"]
53+
```
54+
755
## Enabling Background Fetch
856

9-
> ⚠️ Background fetch this is currently the *only* supported way to do background work on iOS with work manager: **One off tasks** or **Periodic tasks** are available on Android only for now! (see #109)
57+
> ⚠️ Background fetch is one supported way to do background work on iOS with work manager: **Periodic tasks** are available on Android only for now! (see #109)
1058
1159
Background fetching is very different compared to Android's Background Jobs.
1260
In order for your app to support Background Fetch, you have to add the *Background Modes* capability in Xcode for your app's Target and check *Background fetch*:
@@ -72,7 +120,7 @@ Here is an example of a Flutter entrypoint called `callbackDispatcher`:
72120

73121
```dart
74122
void callbackDispatcher() {
75-
Workmanager.executeTask((task, inputData) {
123+
Workmanager().executeTask((task, inputData) {
76124
switch (task) {
77125
case Workmanager.iOSBackgroundTask:
78126
stderr.writeln("The iOS background fetch was triggered");
@@ -105,7 +153,7 @@ If you launched your app using the Flutter command line tools or another IDE lik
105153
To make background work more visible when developing, the WorkManager plugin provides an `isInDebugMode` flag when initializing the plugin:
106154

107155
```dart
108-
Workmanager.initialize(callbackDispatcher, isInDebugMode: true)
156+
Workmanager().initialize(callbackDispatcher, isInDebugMode: true)
109157
```
110158

111159
If `isInDebugMode` is `true`, a local notification will be displayed whenever a background fetch was triggered by iOS. In the example gif below, two background fetches were *simulated* in quick succession. Both completing succesfully after a few seconds in this case:

README.md

+26-2
Original file line numberDiff line numberDiff line change
@@ -39,21 +39,45 @@ void main() {
3939

4040
> The `callbackDispatcher` needs to be either a static function or a top level function to be accessible as a Flutter entry point.
4141
42+
Android tasks are identified using their `taskName`, whereas two default constants are provided for iOS background operations, depending on whether background fetch or BGTaskScheduler is used: `Workmanager.iOSBackgroundTask` & `Workmanager.iOSBackgroundProcessingTask`.
43+
4244
---
4345

4446
# Work Result
4547

4648
The `Workmanager().executeTask(...` block supports 3 possible outcomes:
4749

4850
1. `Future.value(true)`: The task is successful.
49-
2. `Future.value(false)`: The task did not complete successfully and needs to be retried.
51+
2. `Future.value(false)`: The task did not complete successfully and needs to be retried. On Android, the retry is done automatically. On iOS (when using BGTaskScheduler), the retry needs to be scheduled manually.
5052
3. `Future.error(...)`: The task failed.
5153

5254
On Android, the `BackoffPolicy` will configure how `WorkManager` is going to retry the task.
5355

5456
Refer to the example app for a successful, retrying and a failed task.
5557

56-
# Customisation (Android only!)
58+
# Customisation (iOS - BGTaskScheduler only)
59+
iOS supports **One off tasks** with a few basic constraints:
60+
61+
```dart
62+
Workmanager().registerOneOffTask(
63+
"1", // Ignored on iOS
64+
simpleTaskKey, // Ignored on iOS
65+
initialDelay: Duration(minutes: 30),
66+
constraints: Constraints(
67+
// connected or metered mark the task as requiring internet
68+
networkType: NetworkType.connected,
69+
// require external power
70+
requiresCharging: true,
71+
),
72+
inputData: ... // fully supported
73+
);
74+
```
75+
76+
Tasks registered this way will appear in the callback dispatcher using as `Workmanager.iOSBackgroundProcessingTask`.
77+
78+
For more information see the [BGTaskScheduler documentation](https://developer.apple.com/documentation/backgroundtasks).
79+
80+
# Customisation (Android)
5781
Not every `Android WorkManager` feature is ported.
5882

5983
Two kinds of background tasks can be registered :

android/src/main/kotlin/be/tramckrijte/workmanager/Extractor.kt

+4-4
Original file line numberDiff line numberDiff line change
@@ -195,14 +195,14 @@ object Extractor {
195195

196196
private fun extractExistingWorkPolicyFromCall(call: MethodCall): ExistingWorkPolicy =
197197
try {
198-
ExistingWorkPolicy.valueOf(call.argument<String>(REGISTER_TASK_EXISTING_WORK_POLICY_KEY)!!.toUpperCase())
198+
ExistingWorkPolicy.valueOf(call.argument<String>(REGISTER_TASK_EXISTING_WORK_POLICY_KEY)!!.uppercase())
199199
} catch (ignored: Exception) {
200200
defaultOneOffExistingWorkPolicy
201201
}
202202

203203
private fun extractExistingPeriodicWorkPolicyFromCall(call: MethodCall): ExistingPeriodicWorkPolicy =
204204
try {
205-
ExistingPeriodicWorkPolicy.valueOf(call.argument<String>(REGISTER_TASK_EXISTING_WORK_POLICY_KEY)!!.toUpperCase())
205+
ExistingPeriodicWorkPolicy.valueOf(call.argument<String>(REGISTER_TASK_EXISTING_WORK_POLICY_KEY)!!.uppercase())
206206
} catch (ignored: Exception) {
207207
defaultPeriodExistingWorkPolicy
208208
}
@@ -221,7 +221,7 @@ object Extractor {
221221
}
222222

223223
val backoffPolicy = try {
224-
BackoffPolicy.valueOf(call.argument<String>(REGISTER_TASK_BACK_OFF_POLICY_TYPE_KEY)!!.toUpperCase())
224+
BackoffPolicy.valueOf(call.argument<String>(REGISTER_TASK_BACK_OFF_POLICY_TYPE_KEY)!!.uppercase())
225225
} catch (ignored: Exception) {
226226
defaultBackOffPolicy
227227
}
@@ -240,7 +240,7 @@ object Extractor {
240240
private fun extractConstraintConfigFromCall(call: MethodCall): Constraints {
241241
fun extractNetworkTypeFromCall(call: MethodCall) =
242242
try {
243-
NetworkType.valueOf(call.argument<String>(REGISTER_TASK_CONSTRAINTS_NETWORK_TYPE_KEY)!!.toUpperCase())
243+
NetworkType.valueOf(call.argument<String>(REGISTER_TASK_CONSTRAINTS_NETWORK_TYPE_KEY)!!.uppercase())
244244
} catch (ignored: Exception) {
245245
defaultNetworkType
246246
}

android/src/main/kotlin/be/tramckrijte/workmanager/WorkmanagerCallHandler.kt

+1-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ private object RegisterTaskHandler : CallHandler<WorkManagerCall.RegisterTask> {
4343
"You should ensure you have called the 'initialize' function first! " +
4444
"Example: \n" +
4545
"\n" +
46-
"`Workmanager.initialize(\n" +
46+
"`Workmanager().initialize(\n" +
4747
" callbackDispatcher,\n" +
4848
" )`" +
4949
"\n" +

example/ios/Runner.xcodeproj/project.pbxproj

+2-2
Original file line numberDiff line numberDiff line change
@@ -168,7 +168,7 @@
168168
TargetAttributes = {
169169
97C146ED1CF9000F007C117D = {
170170
CreatedOnToolsVersion = 7.3.1;
171-
DevelopmentTeam = 79BMQESM94;
171+
DevelopmentTeam = GPGRWN6G4J;
172172
LastSwiftMigration = 1110;
173173
};
174174
};
@@ -506,7 +506,7 @@
506506
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
507507
CLANG_ENABLE_MODULES = YES;
508508
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
509-
DEVELOPMENT_TEAM = 79BMQESM94;
509+
DEVELOPMENT_TEAM = GPGRWN6G4J;
510510
ENABLE_BITCODE = NO;
511511
FRAMEWORK_SEARCH_PATHS = (
512512
"$(inherited)",

example/ios/Runner/Info.plist

+7-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@
22
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
33
<plist version="1.0">
44
<dict>
5-
<key>UIBackgroundModes</key>
5+
<key>BGTaskSchedulerPermittedIdentifiers</key>
66
<array>
7-
<string>fetch</string>
7+
<string>workmanager.background.task</string>
88
</array>
99
<key>CFBundleDevelopmentRegion</key>
1010
<string>$(DEVELOPMENT_LANGUAGE)</string>
@@ -26,6 +26,11 @@
2626
<string>1.0</string>
2727
<key>LSRequiresIPhoneOS</key>
2828
<true/>
29+
<key>UIBackgroundModes</key>
30+
<array>
31+
<string>fetch</string>
32+
<string>processing</string>
33+
</array>
2934
<key>UILaunchStoryboardName</key>
3035
<string>LaunchScreen</string>
3136
<key>UIMainStoryboardFile</key>

example/lib/main.dart

+26
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,10 @@ void callbackDispatcher() {
5555
print(
5656
"You can access other plugins in the background, for example Directory.getTemporaryDirectory(): $tempPath");
5757
break;
58+
case Workmanager.iOSBackgroundProcessingTask:
59+
print("The iOS Background processing task was called");
60+
await Future.delayed(Duration(seconds: 2));
61+
break;
5862
}
5963

6064
return Future.value(true);
@@ -107,6 +111,28 @@ class _MyAppState extends State<MyApp> {
107111
);
108112
}),
109113
SizedBox(height: 16),
114+
Text("BG Processing Tasks (iOS only)",
115+
style: Theme.of(context).textTheme.headline),
116+
//This task runs once.
117+
//Most likely this will trigger immediately
118+
PlatformEnabledButton(
119+
platform: _Platform.ios,
120+
child: Text("Perform a BG Task"),
121+
onPressed: () {
122+
Workmanager().registerOneOffTask(
123+
"1",
124+
simpleTaskKey,
125+
inputData: <String, dynamic>{
126+
'int': 1,
127+
'bool': true,
128+
'double': 1.0,
129+
'string': 'string',
130+
'array': [1, 2, 3],
131+
},
132+
);
133+
},
134+
),
135+
110136
Text("One Off Tasks (Android only)",
111137
style: Theme.of(context).textTheme.headline),
112138
//This task runs once.
+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
//
2+
// BackgroundTaskOperation.swift
3+
// workmanager
4+
//
5+
// Created by Sebastian Roth on 10/06/2021.
6+
//
7+
8+
import Foundation
9+
10+
class BackgroundTaskOperation : Operation {
11+
12+
private let identifier: String
13+
private let flutterPluginRegistrantCallback: FlutterPluginRegistrantCallback?
14+
15+
init(_ identifier: String, flutterPluginRegistrantCallback: FlutterPluginRegistrantCallback?) {
16+
self.identifier = identifier
17+
self.flutterPluginRegistrantCallback = flutterPluginRegistrantCallback
18+
}
19+
20+
override func main() {
21+
let semaphore = DispatchSemaphore(value: 0)
22+
23+
DispatchQueue.main.async {
24+
let worker = BackgroundWorker(mode: .backgroundTask(identifier: self.identifier),
25+
flutterPluginRegistrantCallback: self.flutterPluginRegistrantCallback)
26+
27+
worker.performBackgroundRequest { _ in
28+
semaphore.signal()
29+
}
30+
}
31+
32+
semaphore.wait()
33+
}
34+
}

0 commit comments

Comments
 (0)