UNPKG

bpm-engine

Version:

Business Process Management Engine for JavaScript

387 lines (300 loc) 36.1 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.Plugins = undefined; var _assign = require('babel-runtime/core-js/object/assign'); var _assign2 = _interopRequireDefault(_assign); var _asyncToGenerator2 = require('babel-runtime/helpers/asyncToGenerator'); var _asyncToGenerator3 = _interopRequireDefault(_asyncToGenerator2); var _Plugins = require('./lib/Plugins'); Object.defineProperty(exports, 'Plugins', { enumerable: true, get: function get() { return _interopRequireDefault(_Plugins).default; } }); require('source-map-support/register'); var _defaults = require('./lib/defaults'); var _defaults2 = _interopRequireDefault(_defaults); var _loadPlugins = require('./lib/loadPlugins'); var _loadPlugins2 = _interopRequireDefault(_loadPlugins); var _debug = require('./lib/debug'); var _debug2 = _interopRequireDefault(_debug); var _TokenInstance = require('./lib/TokenInstance'); var _TokenInstance2 = _interopRequireDefault(_TokenInstance); var _Clock = require('./lib/Clock'); var _Clock2 = _interopRequireDefault(_Clock); var _iso8601RepeatingInterval = require('iso8601-repeating-interval'); var _iso8601RepeatingInterval2 = _interopRequireDefault(_iso8601RepeatingInterval); var _constants = require('./lib/constants'); var constants = _interopRequireWildcard(_constants); var _getNextFlowObjects = require('./lib/getNextFlowObjects'); var _getNextFlowObjects2 = _interopRequireDefault(_getNextFlowObjects); function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } } function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } const log = (0, _debug2.default)('engine'); const isTimerStartEvent = flowObject => { if (flowObject.$type === 'bpmn:StartEvent') { const eventDefinitions = flowObject.eventDefinitions; if (eventDefinitions) { const firstEventDefinition = eventDefinitions[0]; // if this start event has timereventdefinitions if (firstEventDefinition.$type === 'bpmn:TimerEventDefinition') { // if it's timer is of type cycle (repetition) if (firstEventDefinition.timeCycle) { return firstEventDefinition; } } } } return false; }; /** * */ class BPMEngine { constructor({ generateId = _defaults2.default.generateId, evalCondition = _defaults2.default.evalCondition, persist = new _defaults2.default.MemoryPersist(), enableClock = true, plugins = [] } = {}) { var _this = this; this.onTick = (0, _asyncToGenerator3.default)(function* () { log('onTick'); const currentTimestamp = new Date().getTime(); const timerEvent = yield _this.persist.timer.getNext(currentTimestamp); if (timerEvent) { // set this timer to done, so it won't be returned by above `getNext` again yield _this.persist.timer.update({ timerId: timerEvent.timerId }, { status: 'done' }); yield _this.handleTimerEvent(timerEvent); // get the next timer after the current one const interval = (0, _iso8601RepeatingInterval2.default)(timerEvent.interval); // the + 1 is to not get the same timer again const nextTimerEvent = interval.firstAfter(timerEvent.time + 1); // if there is no next, complete this tick if (!nextTimerEvent) { return; } // if the nextTimerEvent's index is lower than the maximum amount of repetitions // create a next timer event const hasPendingRepetitions = nextTimerEvent.index < interval._repeatCount; if (hasPendingRepetitions && timerEvent.intent !== constants.CONTINUE_TOKEN_INSTANCE_INTENT) { yield _this.persist.timer.create({ timerId: _this.generateId(), index: nextTimerEvent.index, interval: timerEvent.interval, time: nextTimerEvent.date / 1, previousTimerId: timerEvent.timerId, intent: timerEvent.intent, workflowDefinitionId: timerEvent.workflowDefinitionId }); } } }); (0, _assign2.default)(this, { generateId, evalCondition, persist, plugins: (0, _loadPlugins2.default)(plugins) }); if (enableClock) { this.clock = new _Clock2.default({ onTick: this.onTick, interval: 100 }); } log('Initiated'); } handleTimerEvent(timerEvent) { var _this2 = this; return (0, _asyncToGenerator3.default)(function* () { let token; const intent = timerEvent.intent; // if the intention of the timer is to start a process instance // create a new process instance and get its token if (intent === constants.START_PROCESS_INSTANCE_INTENT) { const workflowDefinitionId = timerEvent.workflowDefinitionId; token = yield _this2.createProcessInstance({ workflowDefinitionId }); } // if the intention of the timer is to continue a token instance // get the token instance if (intent === constants.CONTINUE_TOKEN_INSTANCE_INTENT) { const tokenId = timerEvent.tokenId; token = yield _this2.continueTokenInstance({ tokenId }); } // if a token was created/fetched, continue its execution if (token) { token.execute(); } })(); } createProcessInstance({ workflowDefinition, workflowDefinitionId, payload = {} } = {}) { var _this3 = this; return (0, _asyncToGenerator3.default)(function* () { let workflowDefinitionXML; if (workflowDefinition) { workflowDefinitionXML = workflowDefinition; } else if (workflowDefinitionId) { const deployedWorkflowDefinition = yield _this3.findWorkflowDefinition(workflowDefinitionId); if (deployedWorkflowDefinition) { workflowDefinitionXML = deployedWorkflowDefinition.xml; } } if (!workflowDefinitionXML) { throw new Error('Invalid WorkflowDefinition'); } const processId = _this3.generateId(); log(`Creating processInstance ${processId}`); yield _this3.persist.processInstance.create({ processId, workflowDefinition: workflowDefinitionXML, payload }); log(`processInstance ${processId} created`); const tokenInstance = yield _this3.createTokenInstance({ processId, payload, workflowDefinition: workflowDefinitionXML }); return tokenInstance; })(); } createTokenInstance({ processId, payload, workflowDefinition, tokenId = this.generateId(), status, parent, isSubProcess, currentActivity, flowObjects }) { var _this4 = this; return (0, _asyncToGenerator3.default)(function* () { log(`Creating tokenInstance ${tokenId}`); const tokenInstance = new _TokenInstance2.default({ processId, tokenId, payload, workflowDefinition, status, parent, isSubProcess, currentActivity, engine: _this4 }); if (!flowObjects) { yield tokenInstance.initialize(); } else { tokenInstance.flowObjects = flowObjects; } log(`tokenInstance ${tokenId} created`); return tokenInstance; })(); } continueTokenInstance({ tokenId, payload = {} }) { var _this5 = this; return (0, _asyncToGenerator3.default)(function* () { log(`Continuing tokenInstance ${tokenId}`); const persistedTokenInstance = yield _this5.findTokenInstance(tokenId); const persistedProcessInstance = yield _this5.findProcessInstance(persistedTokenInstance.processId); const workflowDefinition = persistedProcessInstance.workflowDefinition; const mergedPayload = (0, _assign2.default)(persistedProcessInstance.payload, persistedTokenInstance.payload, payload); const tokenInstance = yield _this5.createTokenInstance({ processId: persistedProcessInstance.processId, payload: mergedPayload, workflowDefinition, tokenId, status: 'paused', currentActivity: persistedTokenInstance.next, isSubProcess: persistedTokenInstance.isSubProcess, parent: persistedTokenInstance.parent }); tokenInstance.flowObjects = (0, _getNextFlowObjects2.default)(tokenInstance.flowObjects, persistedTokenInstance.currentActivity); tokenInstance.next = tokenInstance.flowObjects.find(function (el) { return el.id === persistedTokenInstance.currentActivity; }); return tokenInstance; })(); } findTokenInstance(tokenId) { return this.persist.tokenInstance.find({ tokenId }); } findProcessInstance(processId) { return this.persist.processInstance.find({ processId }); } findWorkflowDefinition(workflowDefinitionId) { return this.persist.workflowDefinition.find({ workflowDefinitionId }); } // create listeners for timer and message start events createStartEvents(dummyTokenInstance, workflowDefinitionId) { var _this6 = this; return (0, _asyncToGenerator3.default)(function* () { const flowObjects = dummyTokenInstance.flowObjects; flowObjects.forEach((() => { var _ref2 = (0, _asyncToGenerator3.default)(function* (flowObject) { const eventDefinition = isTimerStartEvent(flowObject); if (eventDefinition) { // evaluate the interval const intervalExpression = `return \`${eventDefinition.timeCycle.body}\`;`; const intervalExpressionFunction = new Function('timestamp', intervalExpression); const intervalString = intervalExpressionFunction(new Date().toISOString()); // create the interval and get the first iteration const interval = (0, _iso8601RepeatingInterval2.default)(intervalString); const firstAfter = interval.firstAfter(new Date() - 1000); // if the first iteration is not bigger than the max amount of repetitions const hasPendingRepetitions = firstAfter.index <= interval._repeatCount; if (hasPendingRepetitions) { // create a timer event yield _this6.persist.timer.create({ timerId: _this6.generateId(), index: firstAfter.index, time: firstAfter.date / 1, interval: intervalString, intent: constants.START_PROCESS_INSTANCE_INTENT, workflowDefinitionId }); } } }); return function (_x) { return _ref2.apply(this, arguments); }; })()); })(); } deployWorkflowDefinition({ xml, workflowDefinitionId = this.generateId() }) { var _this7 = this; return (0, _asyncToGenerator3.default)(function* () { const dummyTokenInstance = new _TokenInstance2.default({ workflowDefinition: xml }); // initialize the tokenInstance so flowObjects get parsed, // we use them to get all the StartEvents yield dummyTokenInstance.initialize(); const alreadyExistingWorkflowDefinition = yield _this7.persist.workflowDefinition.find({ workflowDefinitionId }); let workflowDefinition; if (alreadyExistingWorkflowDefinition) { workflowDefinition = yield _this7.persist.workflowDefinition.update({ workflowDefinitionId }, { xml }); } else { workflowDefinition = yield _this7.persist.workflowDefinition.create({ xml, workflowDefinitionId }); } // remove already existing start events for this workflowDefinition yield _this7.persist.timer.update({ workflowDefinitionId }, { status: 'done' }, { multi: true }); // we need to create the start events after the creation of the workflowDefinition // so that a timer can not happen before the workflowDefinition is deployed. yield _this7.createStartEvents(dummyTokenInstance, workflowDefinition.workflowDefinitionId); return workflowDefinition; })(); } } exports.default = BPMEngine; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/index.js"],"names":["default","constants","log","isTimerStartEvent","flowObject","$type","eventDefinitions","firstEventDefinition","timeCycle","BPMEngine","constructor","generateId","defaults","evalCondition","persist","MemoryPersist","enableClock","plugins","onTick","currentTimestamp","Date","getTime","timerEvent","timer","getNext","update","timerId","status","handleTimerEvent","interval","nextTimerEvent","firstAfter","time","hasPendingRepetitions","index","_repeatCount","intent","CONTINUE_TOKEN_INSTANCE_INTENT","create","date","previousTimerId","workflowDefinitionId","clock","Clock","token","START_PROCESS_INSTANCE_INTENT","createProcessInstance","tokenId","continueTokenInstance","execute","workflowDefinition","payload","workflowDefinitionXML","deployedWorkflowDefinition","findWorkflowDefinition","xml","Error","processId","processInstance","tokenInstance","createTokenInstance","parent","isSubProcess","currentActivity","flowObjects","TokenInstance","engine","initialize","persistedTokenInstance","findTokenInstance","persistedProcessInstance","findProcessInstance","mergedPayload","next","find","el","id","createStartEvents","dummyTokenInstance","forEach","eventDefinition","intervalExpression","body","intervalExpressionFunction","Function","intervalString","toISOString","deployWorkflowDefinition","alreadyExistingWorkflowDefinition","multi"],"mappings":"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","file":"index.js","sourcesContent":["import 'source-map-support/register';\n\nimport defaults from 'lib/defaults';\nimport loadPlugins from 'lib/loadPlugins';\nimport debug from 'lib/debug';\nimport TokenInstance from 'lib/TokenInstance';\nimport Clock from 'lib/Clock';\nimport makeInterval from 'iso8601-repeating-interval';\n\nimport * as constants from 'lib/constants';\nimport getNextFlowObjects from 'lib/getNextFlowObjects';\n\nconst log = debug('engine');\n\nconst isTimerStartEvent = (flowObject) => {\n  if (flowObject.$type === 'bpmn:StartEvent') {\n    const { eventDefinitions } = flowObject;\n    if (eventDefinitions) {\n      const firstEventDefinition = eventDefinitions[0];\n      // if this start event has timereventdefinitions\n      if (firstEventDefinition.$type === 'bpmn:TimerEventDefinition') {\n        // if it's timer is of type cycle (repetition)\n        if (firstEventDefinition.timeCycle) {\n          return firstEventDefinition;\n        }\n      }\n    }\n  }\n  return false;\n};\n\n/**\n *\n */\nclass BPMEngine {\n  constructor({\n    generateId = defaults.generateId,\n    evalCondition = defaults.evalCondition,\n    persist = new defaults.MemoryPersist(),\n    enableClock = true,\n    plugins = [],\n  } = {}) {\n    Object.assign(this, {\n      generateId,\n      evalCondition,\n      persist,\n      plugins: loadPlugins(plugins),\n    });\n\n    if (enableClock) {\n      this.clock = new Clock({ onTick: this.onTick, interval: 100 });\n    }\n\n    log('Initiated');\n  }\n\n  onTick = async () => {\n    log('onTick');\n    const currentTimestamp = new Date().getTime();\n    const timerEvent = await this.persist.timer.getNext(currentTimestamp);\n\n    if (timerEvent) {\n      // set this timer to done, so it won't be returned by above `getNext` again\n      await this.persist.timer.update({ timerId: timerEvent.timerId }, { status: 'done' });\n\n      await this.handleTimerEvent(timerEvent);\n\n      // get the next timer after the current one\n      const interval = makeInterval(timerEvent.interval);\n\n      // the + 1 is to not get the same timer again\n      const nextTimerEvent = interval.firstAfter(timerEvent.time + 1);\n\n      // if there is no next, complete this tick\n      if (!nextTimerEvent) {\n        return;\n      }\n\n      // if the nextTimerEvent's index is lower than the maximum amount of repetitions\n      // create a next timer event\n      const hasPendingRepetitions = nextTimerEvent.index < interval._repeatCount;\n      if (hasPendingRepetitions && timerEvent.intent !== constants.CONTINUE_TOKEN_INSTANCE_INTENT) {\n        await this.persist.timer.create({\n          timerId: this.generateId(),\n          index: nextTimerEvent.index,\n          interval: timerEvent.interval,\n          time: nextTimerEvent.date / 1,\n          previousTimerId: timerEvent.timerId,\n          intent: timerEvent.intent,\n          workflowDefinitionId: timerEvent.workflowDefinitionId,\n        });\n      }\n    }\n  };\n\n  async handleTimerEvent(timerEvent) {\n    let token;\n    const { intent } = timerEvent;\n\n    // if the intention of the timer is to start a process instance\n    // create a new process instance and get its token\n    if (intent === constants.START_PROCESS_INSTANCE_INTENT) {\n      const { workflowDefinitionId } = timerEvent;\n      token = await this.createProcessInstance({ workflowDefinitionId });\n    }\n\n    // if the intention of the timer is to continue a token instance\n    // get the token instance\n    if (intent === constants.CONTINUE_TOKEN_INSTANCE_INTENT) {\n      const { tokenId } = timerEvent;\n      token = await this.continueTokenInstance({ tokenId });\n    }\n\n    // if a token was created/fetched, continue its execution\n    if (token) {\n      token.execute();\n    }\n  }\n\n  async createProcessInstance({ workflowDefinition, workflowDefinitionId, payload = {} } = {}) {\n    let workflowDefinitionXML;\n\n    if (workflowDefinition) {\n      workflowDefinitionXML = workflowDefinition;\n    }\n    else if (workflowDefinitionId) {\n      const deployedWorkflowDefinition = await this.findWorkflowDefinition(workflowDefinitionId);\n      if (deployedWorkflowDefinition) {\n        workflowDefinitionXML = deployedWorkflowDefinition.xml;\n      }\n    }\n\n    if (!workflowDefinitionXML) {\n      throw new Error('Invalid WorkflowDefinition');\n    }\n\n    const processId = this.generateId();\n    log(`Creating processInstance ${processId}`);\n    await this.persist.processInstance.create({\n      processId,\n      workflowDefinition: workflowDefinitionXML,\n      payload,\n    });\n    log(`processInstance ${processId} created`);\n\n    const tokenInstance = await this.createTokenInstance({\n      processId,\n      payload,\n      workflowDefinition: workflowDefinitionXML,\n    });\n\n    return tokenInstance;\n  }\n\n  async createTokenInstance({\n    processId,\n    payload,\n    workflowDefinition,\n    tokenId = this.generateId(),\n    status,\n    parent,\n    isSubProcess,\n    currentActivity,\n    flowObjects,\n  }) {\n    log(`Creating tokenInstance ${tokenId}`);\n    const tokenInstance = new TokenInstance({\n      processId,\n      tokenId,\n      payload,\n      workflowDefinition,\n      status,\n      parent,\n      isSubProcess,\n      currentActivity,\n      engine: this,\n    });\n    if (!flowObjects) {\n      await tokenInstance.initialize();\n    }\n    else {\n      tokenInstance.flowObjects = flowObjects;\n    }\n    log(`tokenInstance ${tokenId} created`);\n\n    return tokenInstance;\n  }\n\n  async continueTokenInstance({ tokenId, payload = {} }) {\n    log(`Continuing tokenInstance ${tokenId}`);\n    const persistedTokenInstance = await this.findTokenInstance(tokenId);\n    const persistedProcessInstance = await this.findProcessInstance(persistedTokenInstance.processId);\n\n    const { workflowDefinition } = persistedProcessInstance;\n    const mergedPayload = Object.assign(\n      persistedProcessInstance.payload,\n      persistedTokenInstance.payload,\n      payload,\n    );\n\n    const tokenInstance = await this.createTokenInstance({\n      processId: persistedProcessInstance.processId,\n      payload: mergedPayload,\n      workflowDefinition,\n      tokenId,\n      status: 'paused',\n      currentActivity: persistedTokenInstance.next,\n      isSubProcess: persistedTokenInstance.isSubProcess,\n      parent: persistedTokenInstance.parent,\n    });\n\n    tokenInstance.flowObjects = getNextFlowObjects(\n      tokenInstance.flowObjects,\n      persistedTokenInstance.currentActivity,\n    );\n\n    tokenInstance.next = tokenInstance.flowObjects.find(el => el.id === persistedTokenInstance.currentActivity);\n\n    return tokenInstance;\n  }\n\n  findTokenInstance(tokenId) {\n    return this.persist.tokenInstance.find({ tokenId });\n  }\n\n  findProcessInstance(processId) {\n    return this.persist.processInstance.find({ processId });\n  }\n\n  findWorkflowDefinition(workflowDefinitionId) {\n    return this.persist.workflowDefinition.find({ workflowDefinitionId });\n  }\n\n  // create listeners for timer and message start events\n  async createStartEvents(dummyTokenInstance, workflowDefinitionId) {\n    const { flowObjects } = dummyTokenInstance;\n\n    flowObjects.forEach(async (flowObject) => {\n      const eventDefinition = isTimerStartEvent(flowObject);\n      if (eventDefinition) {\n        // evaluate the interval\n        const intervalExpression = `return \\`${eventDefinition.timeCycle.body}\\`;`;\n        const intervalExpressionFunction = new Function('timestamp', intervalExpression);\n        const intervalString = intervalExpressionFunction(new Date().toISOString());\n\n        // create the interval and get the first iteration\n        const interval = makeInterval(intervalString);\n        const firstAfter = interval.firstAfter(new Date() - 1000);\n\n        // if the first iteration is not bigger than the max amount of repetitions\n        const hasPendingRepetitions = firstAfter.index <= interval._repeatCount;\n\n        if (hasPendingRepetitions) {\n          // create a timer event\n          await this.persist.timer.create({\n            timerId: this.generateId(),\n            index: firstAfter.index,\n            time: firstAfter.date / 1,\n            interval: intervalString,\n            intent: constants.START_PROCESS_INSTANCE_INTENT,\n            workflowDefinitionId,\n          });\n        }\n      }\n    });\n  }\n\n  async deployWorkflowDefinition({ xml, workflowDefinitionId = this.generateId() }) {\n    const dummyTokenInstance = new TokenInstance({\n      workflowDefinition: xml,\n    });\n\n    // initialize the tokenInstance so flowObjects get parsed,\n    // we use them to get all the StartEvents\n    await dummyTokenInstance.initialize();\n\n    const alreadyExistingWorkflowDefinition = await this.persist.workflowDefinition.find({\n      workflowDefinitionId,\n    });\n\n    let workflowDefinition;\n\n    if (alreadyExistingWorkflowDefinition) {\n      workflowDefinition = await this.persist.workflowDefinition.update(\n        { workflowDefinitionId },\n        {\n          xml,\n        },\n      );\n    }\n    else {\n      workflowDefinition = await this.persist.workflowDefinition.create({\n        xml,\n        workflowDefinitionId,\n      });\n    }\n\n    // remove already existing start events for this workflowDefinition\n    await this.persist.timer.update({ workflowDefinitionId }, { status: 'done' }, { multi: true });\n\n    // we need to create the start events after the creation of the workflowDefinition\n    // so that a timer can not happen before the workflowDefinition is deployed.\n    await this.createStartEvents(dummyTokenInstance, workflowDefinition.workflowDefinitionId);\n\n    return workflowDefinition;\n  }\n}\n\nexport default BPMEngine;\nexport { default as Plugins } from 'lib/Plugins';\n"]}