#Workflo.js
Workflo.js is a schema-based framework, based on the AWS SWF SDK, for easily defining, registering, and executing your workflows
###What is the AWS Simple Workflow Service?
(I am already familiar with this stuff...)
The AWS Simple Workflow Service, SWF, is a webservice that allows you to define Workflows, which are comprised of Tasks, that represent processes within your business/organization/stack.
A processes may be registering a user with your system, charging a payment account, generating the thumbnails for an image or set of images and publishing them to a CDN, transcoding a video, aggregating log files...whatever. If the process can be broken into individual Tasks, and would benefit from each task being executed in a reliable manner, then a Workflow may be the right way to package it.
Your workflows are invoked via the SWF API, but the execution is handled on your servers by Workers and Deciders. A Worker or Decider is simply a Node.js process that you define, which connects to SWF via HTTP, and, using a standard long-poll method, listens for the next action to process, or decision to make.
This design allows your Workers and Deciders to be very flexible; there is no limit to how many you can have, where they are located, what sort of hardware they run on; this can simplify scaling/adding more capacity, often reduced to just adding more Workers.
You can also define which Workers handle which Workflows/Tasks if needed. This allows you to easily isolate the processing of certain workflows or tasks to one part of your infrastructure, maybe based on specific CPU/Memory/Compliance needs, while running the rest in another part of your infrastructure.
Using SWF, you can avoid extensive boilerplate development for configuring and maintaining message queues, defining and orchestrating task execution across multiple servers with failover and retry mechanisms...a lot of code.
###What does Workflo.js do?
Workflo.js simplifies the process of using the AWS Simple Workflow Service, with Node.js by providing a framework for defining your Workflows and their Tasks using a JSON schema. Workflo provides a facade around the AWS-SDK, adding a promise implementation to the base library using Q, with added functionality including:
- defining Workflows using a JSON schema
- automatic registration and validation of Workflows/Tasks with the API
- helpers for running Workflows and executing Tasks
- chainable methods using promises
- polling for decisions using a DecisionContext, which supports:
- automatically, recursively, loading a Workflow's full event history
- helpers to access specific events in the Workflow's event history, ex last completed task, last failed task, next scheduled task, etc
- an EventHistory collection, which provides functional helpers and methods for accessing and isolating events in the Workflow's event history
- helper methods for driving workflows (failing, completing, cancelling, scheduling tasks)
- polling for tasks using an ActivityContext, which supports:
- JSON data serialization/deserialization for persisting task/workflow state in the Workflow's meta data between Task Execution
- Task level helper methods for driving the Workflows and Tasks (completing, failing, cancelling Tasks, completing and cancelling the Workflow, starting child Workflows, etc)
Workflo also provides a base implementation for your Workers and Deciders which handles the core orchestration logic for making decisions, running tasks, and handling failures. This allows you to focus development on your Workflows and Tasks, not the plumbing, from the start.
###Enough words, show me code
Below is an example workflow that handles an order for a website; it defines the Workflow's minimal configuration, the Tasks (in order) with their minimal configuration, and the handlers for each Task.
Workflo.define('handle-website-order', {
//** defines the workflows meta-data that will be registered with the SWF API
swf: {
description: 'after we charge the users card, we send them a receipt and add them to our mailing list'
},
//** defines the tasks meta-data that will be registered with the SWF API
tasks: [
{
name: 'generate-pdf',
description: 'generates a pdf of the order receipt, returns url'
},
{
name: 'send-receipt',
description: 'sends the user a receipt with a link to the pdf'
},
{
name: 'add-to-mailing-list',
description: 'adds the user to our mailing list'
}
],
//** defines the tasks handlers, by task name
handlers: {
'generate-pdf': function(task) {
var orderId = task.data.orderId;
someLibrary.generatePdf(orderId)
.then(function(url) {
//** pass the url of the generated pdf to the next task
task.complete({ pdfUrl: url });
})
.catch(task.fail);
//** this task is asynchronous
return task.async();
},
'send-receipt': function(task) {
var email = task.data.email,
pdfUrl = task.data.pdfUrl;
if(!someLibrary.sendEmail(email, pdfUrl))
return task.fail({ reason: 'failed to send the email' });
task.complete();
},
'add-to-mailing-list': function(task) {
var email = task.data.email;
//** complete the workflow when this promise resolves, otherwise fail the task and allow it to be retried
anotherLibrary.addEmailToMailList(email)
.then(task.completeWorkflow)
.catch(function(err) {
return task.fail(err);
});
return task.async();
}
}
});
...
//** run the workflow with a data context
Workflow.run('handle-website-order', {
orderId: 'foobar123',
email: 'user@foobar.com'
});
###Ok, What's Next?
- Install Workflo.js
npm install workflo
- If you haven't used the Simple Workflow Service before, you need to register a domain
- you may use more than one domain if you're implementation calls for it, but you need at least one.
- Create an IAM User to use specifically for SWF
- Attach the policy "SimpleWorkflowFullAccess"
- If those permissions don't work for you, read this guide for configuring an IAM policy for SWF.
- Get the Access Key and Secret for the SWF user
- Attach the policy "SimpleWorkflowFullAccess"
- View the Documentation
- Check out the Basic Example
- Browse some additional examples
- Using EC2? Download an AMI that has this all already configured here