diff --git a/README.md b/README.md
index 8f085f618..d458574f8 100644
--- a/README.md
+++ b/README.md
@@ -53,13 +53,13 @@ module.exports.pipe = function(cont, context, action) {
action.logger.log("debug", "Constructing Custom Pipeline");
return pipeline()
- .before(adjustContent)
- .once(cont) // required: execute the continuation function
- .after(cleanupContent)
+ .use(adjustContent)
+ .use(cont) // execute the continuation function
+ .use(cleanupContent)
}
```
-In a typical pipeline, you will add additional processing steps as `.before(require('some-module'))` or as `.after(require('some-module'))`.
+In a typical pipeline, you will add additional processing steps as `.use(require('some-module'))`.
### The Main Function
@@ -138,11 +138,11 @@ Example:
```js
new pipeline()
- .before(doSomething)
- .once(render)
- .after(cleanup)
+ .use(doSomething)
+ .use(render)
+ .use(cleanup)
.error(handleError)
- .after(done);
+ .use(done);
```
If in the above example, the `doSomething` causes an error, subsequently, `render` and `cleanup` will not be invoked. but `handleError` will. If `handleError` clears the error state (i.e. sets `context.error = null`), the `done` function will be invoked again.
@@ -151,16 +151,16 @@ If in the above example, none of the functions causes an error, the `handleError
### Extension Points
-In addition to the (optional) wrapper function which can be invoked prior to the `once` function, pipeline creators can expose named extension points. These extension points allow users of a pipeline to inject additional functions that will be called right before or right after an extension point. To keep the extension points independent from the implementation (i.e. the name of the function), pipeline authors should use the `expose(name)` function to expose a particular extension point.
+In addition to the (optional) wrapper function which can be invoked prior to the `once` function, pipeline creators can expose named extension points. These extension points allow users of a pipeline to inject additional functions that will be called right before, right after or instead of an extension point. To keep the extension points independent from the implementation (i.e. the name of the function), pipeline authors should use the `expose(name)` function to expose a particular extension point.
Example:
```js
new pipeline()
- .before(doSomething).expose('init')
- .once(render)
- .after(cleanup).expose('cleanup')
- .after(done);
+ .use(doSomething).expose('init')
+ .use(render)
+ .use(cleanup).expose('cleanup')
+ .use(done);
```
In this example, two extension points, `init` and `cleanup` have been defined. Note how the name of the extension point can be the same as the name of the function (i.e. `cleanup`), but does not have to be the same (i.e. `init` vs. `doSomething`).
@@ -181,6 +181,7 @@ The easiest way to use extension points is by expanding on the [Wrapper Function
- a `before` object
- an `after` object
+- a `replace` object
Each of these objects can have keys that correspond to the named extension points defined for the pipeline.
@@ -198,6 +199,12 @@ module.exports.after = {
// will get called after the "fetch" pipeline step
}
}
+
+module.exports.replace = {
+ meta: (context, action) => {
+ // will get called instead of the "meta" pipeline step
+ }
+}
```
All functions that are using the `before` and `after` extension points need to follow the same interface that all other pipeline functions follow, i.e. they have access to `context` and `action` and they should return a modified `context` object.
diff --git a/src/defaults/default.js b/src/defaults/default.js
index 48a5784cb..d41d39fcf 100644
--- a/src/defaults/default.js
+++ b/src/defaults/default.js
@@ -24,7 +24,7 @@ const Pipeline = require('../pipeline.js');
*/
function pipe(next, context, action) {
const mypipeline = new Pipeline(action);
- mypipeline.once(next);
+ mypipeline.use(next);
return mypipeline.run(context);
}
diff --git a/src/defaults/html.pipe.js b/src/defaults/html.pipe.js
index 737de0c84..6bd4a4ad3 100644
--- a/src/defaults/html.pipe.js
+++ b/src/defaults/html.pipe.js
@@ -59,31 +59,31 @@ const htmlpipe = (cont, context, action) => {
.every(dump.record)
.every(validate).when(() => !production())
.every(timer.update)
- .before(resolveRef).expose('resolve').when(hascontent)
- .before(fetch).expose('fetch').when(hascontent)
- .before(parse).expose('parse')
- .before(parseFrontmatter)
- .before(embeds)
- .before(smartypants)
- .before(sections)
- .before(meta).expose('meta')
- .before(unwrapSoleImages)
- .before(selectstrain)
- .before(selecttest)
- .before(html).expose('html')
- .before(sanitize).when(paranoid)
- .once(cont)
- .after(type('text/html'))
- .after(cache).when(uncached)
- .after(key)
- .after(tovdom).expose('post') // start HTML post-processing
- .after(removeHlxProps).when(() => production())
- .after(rewriteLinks).when(production)
- .after(addHeaders)
- .after(tohtml) // end HTML post-processing
- .after(flag).expose('esi').when(esi) // flag ESI when there is ESI in the response
- .after(debug)
- .after(timer.report)
+ .use(resolveRef).expose('resolve').when(hascontent)
+ .use(fetch).expose('fetch').when(hascontent)
+ .use(parse).expose('parse')
+ .use(parseFrontmatter)
+ .use(embeds)
+ .use(smartypants)
+ .use(sections)
+ .use(meta).expose('meta')
+ .use(unwrapSoleImages)
+ .use(selectstrain)
+ .use(selecttest)
+ .use(html).expose('html')
+ .use(sanitize).when(paranoid)
+ .use(cont)
+ .use(type('text/html'))
+ .use(cache).when(uncached)
+ .use(key)
+ .use(tovdom).expose('post') // start HTML post-processing
+ .use(removeHlxProps).when(() => production())
+ .use(rewriteLinks).when(production)
+ .use(addHeaders)
+ .use(tohtml) // end HTML post-processing
+ .use(flag).expose('esi').when(esi) // flag ESI when there is ESI in the response
+ .use(debug)
+ .use(timer.report)
.error(dump.report)
.error(selectStatus);
diff --git a/src/defaults/json.pipe.js b/src/defaults/json.pipe.js
index 5d0a9a55f..18df90503 100644
--- a/src/defaults/json.pipe.js
+++ b/src/defaults/json.pipe.js
@@ -37,16 +37,16 @@ const jsonpipe = (cont, context, action) => {
.every(dump.record)
.every(validate).when(() => !production())
.every(timer.update)
- .before(fetch).expose('fetch')
- .before(parse).expose('parse')
- .before(parseFrontmatter)
- .before(smartypants)
- .before(sections)
- .before(meta).expose('meta')
- .once(cont)
- .after(emit).expose('json')
- .after(type('application/json'))
- .after(timer.report)
+ .use(fetch).expose('fetch')
+ .use(parse).expose('parse')
+ .use(parseFrontmatter)
+ .use(smartypants)
+ .use(sections)
+ .use(meta).expose('meta')
+ .use(cont)
+ .use(emit).expose('json')
+ .use(type('application/json'))
+ .use(timer.report)
.error(dump.report)
.error(selectStatus(production()));
diff --git a/src/defaults/xml.pipe.js b/src/defaults/xml.pipe.js
index 73bb074cc..0b332412f 100644
--- a/src/defaults/xml.pipe.js
+++ b/src/defaults/xml.pipe.js
@@ -41,21 +41,20 @@ const xmlpipe = (cont, context, action) => {
.every(dump.record)
.every(validate).when(() => !production())
.every(timer.update)
- .before(fetch).expose('fetch')
- .before(parse).expose('parse')
- .before(parseFrontmatter)
- .before(smartypants)
- .before(sections)
- .before(meta).expose('meta')
- .once(cont)
- .after(emit).expose('xml')
- .after(type('application/xml'))
- .after(check)
- .after(cache)
- .when(uncached)
- .after(key)
- .after(flag).expose('esi').when(esi) // flag ESI when there is ESI in the response
- .after(timer.report)
+ .use(fetch).expose('fetch')
+ .use(parse).expose('parse')
+ .use(parseFrontmatter)
+ .use(smartypants)
+ .use(sections)
+ .use(meta).expose('meta')
+ .use(cont)
+ .use(emit).expose('xml')
+ .use(type('application/xml'))
+ .use(check)
+ .use(cache).when(uncached)
+ .use(key)
+ .use(flag).expose('esi').when(esi) // flag ESI when there is ESI in the response
+ .use(timer.report)
.error(dump.report)
.error(selectStatus);
diff --git a/src/pipeline.js b/src/pipeline.js
index a72cdf2b3..3e606ef0b 100644
--- a/src/pipeline.js
+++ b/src/pipeline.js
@@ -69,10 +69,8 @@ function errorWrapper(fn) {
*/
/**
- * Pipeline that allows to execute a list of functions in order. The pipeline consists of 3
- * major function lists: `pre`, `once` and, `post`. the functions added to the `pre` list are
- * processed first, then the `once` function and finally the `post` functions.
- * Using `when` and `unless` allows to conditionally execute the previously define function.
+ * Pipeline that allows to execute a list of functions in the order they have been added.
+ * Using `when` and `unless` allows to conditionally execute the previously defined function.
* @class
*/
class Pipeline {
@@ -90,53 +88,39 @@ class Pipeline {
// function chain that was defined last. used for `when` and `unless`
this._last = null;
- // functions that are executed first
- this._pres = [];
- // function that is executed once
- this._oncef = null;
- // functions that are executed after
- this._posts = [];
+ // object with the custom functions to attach to the pipeline
+ this._attachments = null;
+ // step functions to execute
+ this._steps = [];
// functions that are executed before each step
this._taps = [];
- this.attach = (ext) => {
- if (this.sealed) {
- return;
- }
- if (ext && ext.before && typeof ext.before === 'object') {
- Object.keys(ext.before).map((key) => this.attach.before(key, ext.before[key]));
- }
- if (ext && ext.after && typeof ext.after === 'object') {
- Object.keys(ext.after).map((key) => this.attach.after(key, ext.after[key]));
- }
- this.sealed = true;
- };
-
/**
* Registers an extension to the pipeline.
* @param {String} name - name of the extension point (typically the function name).
* @param {pipelineFunction} f - a new pipeline step that will be injected relative to `name`.
- * @param {integer} before - where to insert the new function (true = before, false = after)
+ * @param {integer} offset - where to insert the new function (-1: before, 0: replace, 1: after)
*/
- const attachGeneric = (name, f, before) => {
- const offset = before ? 0 : 1;
+ const attachGeneric = (name, f, offset) => {
// find the index of the function where the resolved ext name
- // matches the provided name by searching the list of pre and
- // post functions
- const foundpres = this._pres
- .findIndex((pre) => pre && pre.ext && pre.ext === name);
- const foundposts = this._posts
- .findIndex((post) => post && post.ext && post.ext === name);
-
- // if something has been found in either lists, insert the
+ // matches the provided name by searching the list of step functions
+ const foundstep = this._steps
+ .findIndex((step) => step && step.ext && step.ext === name);
+
+ // if something has been found in the list, insert the
// new function into the list, with the correct offset
- if (foundpres !== -1) {
- this._pres.splice(foundpres + offset, 0, f);
- }
- if (foundposts !== -1) {
- this._posts.splice(foundposts + offset, 0, f);
- }
- if (foundpres === -1 && foundposts === -1) {
+ if (foundstep !== -1) {
+ if (offset === 0) {
+ // replace
+ this._steps.splice(foundstep, 1, f);
+ } else if (offset > 0) {
+ // insert after
+ this._steps.splice(foundstep + 1, 0, f);
+ } else {
+ // insert before (default)
+ this._steps.splice(foundstep, 0, f);
+ }
+ } else {
this._action.logger.warn(`Unknown extension point ${name}`);
}
};
@@ -147,7 +131,7 @@ class Pipeline {
* @param {String} name - name of the extension point (typically the function name).
* @param {pipelineFunction} f - a new pipeline step that will be injected relative to `name`.
*/
- this.attach.before = (name, f) => attachGeneric.bind(this)(name, f, true);
+ this.attach.before = (name, f) => attachGeneric.bind(this)(name, f, -1);
/**
* Registers an extension to the pipeline. The function `f` will be run in
* the pipeline after the function called `name` will be executed. If `name`
@@ -155,30 +139,33 @@ class Pipeline {
* @param {String} name - name of the extension point (typically the function name).
* @param {pipelineFunction} f - a new pipeline step that will be injected relative to `name`.
*/
- this.attach.after = (name, f) => attachGeneric.bind(this)(name, f, false);
- }
-
- /**
- * Adds a processing function to the `pre` list to this pipeline.
- * @param {pipelineFunction} f function to add to the `post` list
- * @returns {Pipeline} this
- */
- before(f) {
- this.describe(f);
- this._pres.push(f);
- this._last = this._pres;
- return this;
+ this.attach.after = (name, f) => attachGeneric.bind(this)(name, f, 1);
+ /**
+ * Registers an extension to the pipeline. The function `f` will be executed in
+ * the pipeline instead of the function called `name`. If `name` does not exist,
+ * `f` will never be executed.
+ * @param {String} name - name of the extension point (typically the function name).
+ * @param {pipelineFunction} f - a new pipeline step that will replace `name`.
+ */
+ this.attach.replace = (name, f) => attachGeneric.bind(this)(name, f, 0);
}
/**
- * Adds a processing function to the `pre` list to this pipeline.
- * @param {pipelineFunction} f function to add to the `post` list
+ * Adds a processing function to the `step` list of this pipeline.
+ * @param {pipelineFunction} f function to add to the `step` list
* @returns {Pipeline} this
*/
- after(f) {
+ use(f) {
this.describe(f);
- this._posts.push(f);
- this._last = this._posts;
+ this._steps.push(f);
+ this._last = this._steps;
+ // check for extensions
+ if (f && (f.before || f.replace || f.after)) {
+ if (typeof this._attachments === 'function') {
+ throw new Error(`Step '${this._attachments.alias}' already registered extensions for this pipeline, refusing to add more with '${f.alias}'.`);
+ }
+ this._attachments = f;
+ }
return this;
}
@@ -204,7 +191,7 @@ class Pipeline {
}
/**
- * Adds a condition to the previously defined `pre` or `post` function. The previously defined
+ * Adds a condition to the previously defined `step` function. The previously defined
* function will only be executed if the predicate evaluates to something truthy or returns a
* Promise that resolves to something truthy.
* @param {function(context)} predicate Predicate function.
@@ -241,7 +228,7 @@ class Pipeline {
}
/**
- * Adds a condition to the previously defined `pre` or `post` function. The previously defined
+ * Adds a condition to the previously defined `step` function. The previously defined
* function will only be executed if the predicate evaluates to something not-truthy or returns a
* Promise that resolves to something not-truthy.
* @param {function(context)} predicate Predicate function.
@@ -254,15 +241,45 @@ class Pipeline {
}
/**
- * Sets the `once` processing function.
- * @param {pipelineFunction} f the `once` function to set
- * @returns {Pipeline} this
+ * Attaches custom extensions to the pipeline. The expected argument is an object
+ * with a before
object, an after
object, and/or
+ * a replace
object as its properties.
+ * Each of these objects can have keys that correspond to the named extension points
+ * defined for the pipeline, with the function to execute as values. For example:
+ *
+ * { + * before: { + * fetch: (context, action) => { + * // do something before the fetch step + * return context; + * } + * } + * replace: { + * html: (context, action) => { + * // do this instead of the default html step + * return context; + * } + * } + * after: { + * meta: (context, action) => { + * // do something after the meta step + * return context; + * } + * } + * } + *+ * @param {Object} att The object containing the attachments */ - once(f) { - this.describe(f); - this._oncef = f; - this._last = this._posts; - return this; + attach(att) { + if (att && att.before && typeof att.before === 'object') { + Object.keys(att.before).map((key) => this.attach.before(key, att.before[key])); + } + if (att && att.after && typeof att.after === 'object') { + Object.keys(att.after).map((key) => this.attach.after(key, att.after[key])); + } + if (att && att.replace && typeof att.replace === 'object') { + Object.keys(att.replace).map((key) => this.attach.replace(key, att.replace[key])); + } } /** @@ -305,7 +322,7 @@ class Pipeline { } /** - * Runs the pipline processor be executing the `pre`, `once`, and `post` functions in order. + * Runs the pipline processor be executing the `step` functions in order. * @param {Context} context Pipeline context * @returns {Promise