bob.js

  1. "use strict";
  2. import async from 'async';
  3. import config from './config.js';
  4. import jazz from 'jazz';
  5. import p from 'path';
  6. import runner from './runner.js';
  7. import task from './task.js';
  8. import util from 'util';
  9. class Bob {
  10. /**
  11. * Constructor for initialising Bob.
  12. *
  13. * @param {Object} opts: optional
  14. * - appDir: directory where the userland application calls Bob from
  15. * - bobDir: directory where Bob's installation is located
  16. * - appName: name of the userland application (retrieved from its package.json)
  17. * - appVersion: version value of the userland application (retrieved from its package.json)
  18. * - bobMode: either `human` or `robot`, when you are a robot - you have to declare yourself :-]
  19. * - quiet: when true, only display task name and command, but without the command output
  20. */
  21. constructor(opts) {
  22. this.opts = opts;
  23. }
  24. /**
  25. * Bob is goin' to work!
  26. * Execute specified tasks
  27. *
  28. * @param {Array} taskNames: an array of task names
  29. * @param {Function} cb: standard cb(err, result) callback
  30. */
  31. build(taskNames, cb) {
  32. const EXEC_OPTS = {
  33. cwd: this.opts.appDir,
  34. quiet: this.opts.quiet,
  35. maxBuffer: this.opts.maxBufferInKb * 1024
  36. };
  37. const self = this;
  38. this._init(taskNames, (err, settings) => {
  39. if (err) { cb(err); } else {
  40. self._commands(taskNames, settings, (err, commands) => {
  41. if (err) { cb(err); } else {
  42. runner.execSeries(commands, EXEC_OPTS, cb);
  43. }
  44. });
  45. }
  46. });
  47. }
  48. /**
  49. * Load Bob task files and application config file, and install optional dependencies, all in parallel.
  50. *
  51. * @param {Array} taskNames: an array of task names
  52. * @param {Function} cb: standard cb(err, result) callback
  53. */
  54. _init(taskNames, cb) {
  55. const self = this;
  56. function loadTask(cb) {
  57. task.load(taskNames, p.join(self.opts.bobDir, 'conf', 'tasks'), cb);
  58. }
  59. function loadConfig(cb) {
  60. config.load(taskNames, self.opts.appDir, cb);
  61. }
  62. async.parallel({ bobTasks: loadTask, appConfig: loadConfig }, cb);
  63. }
  64. /**
  65. * Prepare commands to execute.
  66. *
  67. * @param {Array} taskNames: an array of task names
  68. * @param {Object} settings: tasks and application configs
  69. * @param {Function} cb: standard cb(err, result) callback
  70. */
  71. _commands(taskNames, settings, cb) {
  72. const TPL_PARAMS = {
  73. app: this.opts.appDir,
  74. bob: this.opts.bobDir,
  75. name: this.opts.appName,
  76. tmp: this.opts.tmpDir,
  77. version: this.opts.appVersion
  78. };
  79. const bobTasks = settings.bobTasks,
  80. bobMode = this.opts.bobMode,
  81. self = this;
  82. const commands = [];
  83. taskNames.forEach((taskName) => {
  84. self._taskTypeNames(taskName, settings).forEach((taskTypeName) => {
  85. const taskType = bobTasks[taskName].types[taskTypeName];
  86. Object.keys(taskType).forEach((key) => {
  87. if (typeof taskType[key] === 'object') {
  88. taskType[key] = taskType[key][bobMode];
  89. }
  90. });
  91. taskType.preOpts = taskType.preOpts || taskType.opts;
  92. // strip undefined and null from command elements
  93. const elems = [taskType.bin, taskType.preOpts, taskType.args, taskType.postOpts].filter(elem => elem);
  94. commands.push({
  95. format: util.format(elems.join(' ')),
  96. meta: {
  97. task: taskName,
  98. type: taskTypeName
  99. }
  100. });
  101. });
  102. });
  103. const jobs = [];
  104. commands.forEach((command) => {
  105. jobs.push((cb) => {
  106. jazz.compile(command.format).process(TPL_PARAMS, (result) => {
  107. command.exec = result;
  108. delete command.format;
  109. cb(null, command);
  110. });
  111. });
  112. });
  113. async.parallel(jobs, cb);
  114. }
  115. _taskTypeNames(taskName, settings) {
  116. // NOTE: allow multiple types in application config
  117. let taskTypeNames =
  118. (settings.appConfig[taskName] && settings.appConfig[taskName].type) ?
  119. settings.appConfig[taskName].type :
  120. settings.bobTasks[taskName].default;
  121. if (!Array.isArray(taskTypeNames)) {
  122. taskTypeNames = [taskTypeNames];
  123. }
  124. return taskTypeNames;
  125. }
  126. }
  127. export {
  128. Bob as default
  129. };