TestsTested | ✗ |
LangLanguage | Obj-CObjective C |
License | MIT |
ReleasedLast Release | Dec 2014 |
Maintained by Gaspard Viot, Jérôme Morissard, Paul-Anatole CLAUDOT, Philippe Bernery.
BkTask is a library inspired by the Proactor and Reactor design patterns. It is designed with two concepts in mind, asynchronous and modular.
Doing work asynchronously allows to keep your application responsive. But parallel programming is hard. On iOS, tools like GCD and NSOperation makes it much simpler but, for complex workflows it still takes time and resources to achieve. BkTask allows you to simply run workflows in background and be notified once it is done.
What about modularity then ? One way to simply a complex task is to breaking it into multiple simple steps. That is how you build a task with BkTask, by adding the steps you need to complete your workflow. If your workflow changes, add or remove steps to update it. You can even create your own steps and reuse them in different workflows. The modular aspect of BkTask helps you to stay agile.
On the technical side, it is built using technologies like NSOperation and key-value observing.
This library is already used in AppStore applications used by millions of people.
v 0.9.1
v 0.9
v 0.8
Just add the following line in your podfile
pod 'BkTask'
The simplest way to use BkTask with your project is with Xcode 4 workspaces. Just drag and drop BkTask.xcodeproj
below your other projects in the workspace.
Then, select your project's target and click on the Build Phases
tab. Unfold the Link Binary With Libraries
section and click on the + button to add a new library.
Select libBkTask.a
and click Add
.
As it is a static library, you will need to indicate the path to BkTask headers. To do so, select your project's target and click on the Build Settings
tab. Find the Header Search Paths
and set the path to the Classes
directory of BkTask.
For instance, let's assume that you have a Libraries
directory at the same level of your .xcodeproj
where you store all your external libraries. In that case, the header to indicate would be $(SRCROOT)/Librairies/BkTask/Classes
.
A task is made of multiple steps sharing a content between them. A task monitor step execution and ensure that content is passed through steps. When all the steps are finished, the task notify every observer registered by calling a success block. If one step fails to execute, the task notify observers by calling a failure block. Note that observers for success and failure may be different.
A step takes data from its input, process them and return the result through its output. The content is processed by each step and is returned when each step has completed.
A few things to know :
First, add BkTask import
#import "BkTask.h"
Now we can create a simple task to load a file from disk
//Create a task
BKTTask *aTask = [[BKTTask alloc] init];
//Create a file load step
BKTFileLoadingOperation *fileLoadOperation = [BKTFileLoadingOperation loadOperationWithFile:FILE_TO_OPEN_URL];
//Add the step to the task
[aTask addStep:fileLoadOperation];
//Set completion and failure blocks
[aTask addTarget:self completion:^(BKTTask *task, id output) {
NSLog(@"Task completed with success");
}];
[aTask addTarget:self failure:^(BKTTask *task, NSError *error) {
NSLog(@"Task failed");
}];
//Start the task
self.myTask = aTask; //To retain the task
[self.myTask start];
Of course, a task with one step is not very usefull and GCD is better at that. However, when you have a workflow with multiple steps, it becomes a mess of nested blocks with GCD while BkTask takes care of chaining steps and passing data through them.
For frequent use cases, there are helper methods to build preconfigured tasks.
// To create a task with a download URL step
+ (id) taskWithRequest:(NSURLRequest *)aRequest;
// To create a task with a download URL step and a JSON parsing step
+ (id) taskWithJSONRequest:(NSURLRequest *)aRequest;
Sometimes, you may have application specific work to execute in background. Or you may want to replace the default JSON parsing step with a step using your favorite JSON parsing library. Or simply, there is no existing step to do what you want.
BkTask offers multiple ways for creating custom steps. If your step is quite simple and meant to be use only once, you can use BkBlockStepOperation
with a block describing how to process the data in your step.
+ (id) blockOperationWithInputKey:(NSString *)inKey outputKey:(NSString *)outKey block:(BKTBlockStepOperationBlock)workBlock;
+ (id) blockOperationWithQueue:(NSOperationQueue *)queue block:(BKTBlockStepOperationBlock)workBlock;
+ (id) blockOperationWithBlock:(BKTBlockStepOperationBlock)workBlock;
The inKey
and outKey
parameters can take two values :
BkTaskContentBodyData
means your input (or output respectively) is binary data in a NSData objectBkTaskContentBodyObject
means your input (or output respectively) is NSObject subclassThe ouput type of one step have to be identical to the input of the next step. Otherwise, the task would fail to complete.
To deeply customize a step, or if you need a reusable step, you can subclass BKTBasicStepOperation
. A step is an NSOperation
subclass implementing the BKTTaskStep
protocol. BKTBasicStepOperation
is an abstract class implementing the boilerplate parts to provide a much simpler API to create a class. All you need to do is to override the following methods :
- (NSString *) inputKey;
- (NSString *) outputKey;
- (id) processInput:(id)theInput error:(NSError **)error;
The implementation of inputKey
and outputKey
should return BkTaskContentBodyData
or BkTaskContentBodyObject
depending on what kind of input your step will process. Then, the processInput: error:
method is where the input processing is done, returning the output. Of course, subclassing BKTBasicStepOperation
allows you to add any property or method you need to configure and process the input.
If you create a generic step and think it could be useful to other people, feel free to send a pull request. We would be glad to consider and add it to BkTask.
The sample project is a little weather application. You can search for a city and see related forecasts. It demonstrates how to create a task that will download and parse JSON. It also shows how to add a simple custom step to a task using the block method. To see a sample of BKTBasicStepOperation
subclassing, you can read the source of BKTJSONParsingOperation
.
Is is intented to run on iOS 6.1 an above. We recommend using Xcode 6.0 or above to build it.
BkTask requires iOS 5.0 and above. If you need iOS 4.3 compatibility, you can exclude the BKTJSONParsingOperation
class of the BkTask target and it should be okay.
However this tip is not guaranteed to stay true in future releases.
Xcode 5.0 is required to build the BkTask project. The BKTURLSessionLoadingOperation
cannot be used before iOS 7.0.
BkTask uses ARC.
If you are using BkTask in your non-arc project, you will need to set a -fobjc-arc
compiler flag on all of the BkTask source files.
To set a compiler flag in Xcode, go to your active target and select the "Build Phases" tab. Now select all BkTask source files, press Enter, insert -fobjc-arc and then "Done" to enable ARC for BkTask.
If you use Xcode 4 workspaces with BkTask included as a project into your workspace then you don't need any specific configuration related to ARC.
BkTask is available under the MIT license. See the LICENSE file for more info.