| /*! | 
|  * use <https://github.com/jonschlinkert/use> | 
|  * | 
|  * Copyright (c) 2015-2017, Jon Schlinkert. | 
|  * Released under the MIT License. | 
|  */ | 
|   | 
| 'use strict'; | 
|   | 
| module.exports = function base(app, options) { | 
|   if (!isObject(app) && typeof app !== 'function') { | 
|     throw new TypeError('expected an object or function'); | 
|   } | 
|   | 
|   var opts = isObject(options) ? options : {}; | 
|   var prop = typeof opts.prop === 'string' ? opts.prop : 'fns'; | 
|   if (!Array.isArray(app[prop])) { | 
|     define(app, prop, []); | 
|   } | 
|   | 
|   /** | 
|    * Define a plugin function to be passed to use. The only | 
|    * parameter exposed to the plugin is `app`, the object or function. | 
|    * passed to `use(app)`. `app` is also exposed as `this` in plugins. | 
|    * | 
|    * Additionally, **if a plugin returns a function, the function will | 
|    * be pushed onto the `fns` array**, allowing the plugin to be | 
|    * called at a later point by the `run` method. | 
|    * | 
|    * ```js | 
|    * var use = require('use'); | 
|    * | 
|    * // define a plugin | 
|    * function foo(app) { | 
|    *   // do stuff | 
|    * } | 
|    * | 
|    * var app = function(){}; | 
|    * use(app); | 
|    * | 
|    * // register plugins | 
|    * app.use(foo); | 
|    * app.use(bar); | 
|    * app.use(baz); | 
|    * ``` | 
|    * @name .use | 
|    * @param {Function} `fn` plugin function to call | 
|    * @api public | 
|    */ | 
|   | 
|   define(app, 'use', use); | 
|   | 
|   /** | 
|    * Run all plugins on `fns`. Any plugin that returns a function | 
|    * when called by `use` is pushed onto the `fns` array. | 
|    * | 
|    * ```js | 
|    * var config = {}; | 
|    * app.run(config); | 
|    * ``` | 
|    * @name .run | 
|    * @param {Object} `value` Object to be modified by plugins. | 
|    * @return {Object} Returns the object passed to `run` | 
|    * @api public | 
|    */ | 
|   | 
|   define(app, 'run', function(val) { | 
|     if (!isObject(val)) return; | 
|   | 
|     if (!val.use || !val.run) { | 
|       define(val, prop, val[prop] || []); | 
|       define(val, 'use', use); | 
|     } | 
|   | 
|     if (!val[prop] || val[prop].indexOf(base) === -1) { | 
|       val.use(base); | 
|     } | 
|   | 
|     var self = this || app; | 
|     var fns = self[prop]; | 
|     var len = fns.length; | 
|     var idx = -1; | 
|   | 
|     while (++idx < len) { | 
|       val.use(fns[idx]); | 
|     } | 
|     return val; | 
|   }); | 
|   | 
|   /** | 
|    * Call plugin `fn`. If a function is returned push it into the | 
|    * `fns` array to be called by the `run` method. | 
|    */ | 
|   | 
|   function use(type, fn, options) { | 
|     var offset = 1; | 
|   | 
|     if (typeof type === 'string' || Array.isArray(type)) { | 
|       fn = wrap(type, fn); | 
|       offset++; | 
|     } else { | 
|       options = fn; | 
|       fn = type; | 
|     } | 
|   | 
|     if (typeof fn !== 'function') { | 
|       throw new TypeError('expected a function'); | 
|     } | 
|   | 
|     var self = this || app; | 
|     var fns = self[prop]; | 
|   | 
|     var args = [].slice.call(arguments, offset); | 
|     args.unshift(self); | 
|   | 
|     if (typeof opts.hook === 'function') { | 
|       opts.hook.apply(self, args); | 
|     } | 
|   | 
|     var val = fn.apply(self, args); | 
|     if (typeof val === 'function' && fns.indexOf(val) === -1) { | 
|       fns.push(val); | 
|     } | 
|     return self; | 
|   } | 
|   | 
|   /** | 
|    * Wrap a named plugin function so that it's only called on objects of the | 
|    * given `type` | 
|    * | 
|    * @param {String} `type` | 
|    * @param {Function} `fn` Plugin function | 
|    * @return {Function} | 
|    */ | 
|   | 
|   function wrap(type, fn) { | 
|     return function plugin() { | 
|       return this.type === type ? fn.apply(this, arguments) : plugin; | 
|     }; | 
|   } | 
|   | 
|   return app; | 
| }; | 
|   | 
| function isObject(val) { | 
|   return val && typeof val === 'object' && !Array.isArray(val); | 
| } | 
|   | 
| function define(obj, key, val) { | 
|   Object.defineProperty(obj, key, { | 
|     configurable: true, | 
|     writable: true, | 
|     value: val | 
|   }); | 
| } |