Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

@slack/interactive-messages

Package Overview
Dependencies
Maintainers
11
Versions
21
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

@slack/interactive-messages - npm Package Compare versions

Comparing version 0.1.2 to 0.2.0

30

dist/adapter.js

@@ -239,7 +239,15 @@ 'use strict';

var action = payload.actions && payload.actions[0];
// The following result value represents "no replacement"
var result = { status: 200 };
// when the matcher finds a dialog submission, it will populate this value with a function
var dialogPromiseResolve = void 0;
var respond = function respond(message) {
debug('sending async response');
return _this4.axios.post(payload.response_url, message);
if (payload.response_url) {
debug('sending async response');
return _this4.axios.post(payload.response_url, message);
} else if (dialogPromiseResolve) {
dialogPromiseResolve(message);
}
return true;
};

@@ -282,2 +290,18 @@

// Dialog submissions must be responded to in under 3 seconds
// Setting timeout to 2.5 seconds to account for propagation
if (payload.type === 'dialog_submission') {
var ms = 2500;
if (callbackResult) {
result = { status: 200, content: (0, _util.promiseTimeout)(ms, callbackResult) };
} else {
result = {
status: 200, content: new Promise(function (resolve) {
dialogPromiseResolve = resolve;
})
};
}
return true;
}
if (callbackResult) {

@@ -305,3 +329,3 @@ // Checking for Promise type

exports.default = SlackMessageAdapter;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/adapter.js"],"names":["debug","formatMatchingConstraints","matchingConstraints","ret","callbackId","Object","assign","validateConstraints","TypeError","validateActionConstraints","actionConstraints","type","SlackMessageAdapter","verificationToken","callbacks","axios","create","headers","path","Promise","resolve","then","all","express","bodyParser","app","use","urlencoded","extended","post","expressMiddleware","createServer","port","reject","server","on","listen","close","error","Error","callback","message","registerCallback","optionsConstraints","constraints","push","payload","action","actions","result","status","respond","response_url","some","fn","callbackResult","callback_id","test","unfurl","is_app_unfurl","call","catch","content"],"mappings":";;;;;;;;;;AAAA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;AACA;;;;;;AAEA,IAAMA,QAAQ,qBAAa,qCAAb,CAAd;;AAEA;;;;;;AAMA,SAASC,yBAAT,CAAmCC,mBAAnC,EAAwD;AACtD,MAAIC,MAAM,EAAV;AACA,MAAI,CAAC,sBAAcD,mBAAd,CAAL,EAAyC;AACvCC,QAAIC,UAAJ,GAAiBF,mBAAjB;AACD,GAFD,MAEO;AACLC,UAAME,OAAOC,MAAP,CAAc,EAAd,EAAkBJ,mBAAlB,CAAN;AACD;AACD,SAAOC,GAAP;AACD;;AAED;;;;;;AAMA,SAASI,mBAAT,CAA6BL,mBAA7B,EAAkD;AAChD,MAAIA,oBAAoBE,UAApB,IACA,EAAE,sBAASF,oBAAoBE,UAA7B,KAA4C,sBAASF,oBAAoBE,UAA7B,CAA9C,CADJ,EAC6F;AAC3F,WAAO,IAAII,SAAJ,CAAc,wCAAd,CAAP;AACD;;AAED,SAAO,KAAP;AACD;;AAED;;;;;;AAMA,SAASC,yBAAT,CAAmCC,iBAAnC,EAAsD;AACpD,MAAIA,kBAAkBC,IAAlB,IACA,EAAED,kBAAkBC,IAAlB,KAA2B,QAA3B,IAAuCD,kBAAkBC,IAAlB,KAA2B,QAApE,CADJ,EACmF;AACjF,WAAO,IAAIH,SAAJ,CAAc,uCAAd,CAAP;AACD;;AAED;AACA,SAAO,KAAP;AACD;;IAEoBI,mB;AACnB;;;;;AAKA,+BAAYC,iBAAZ,EAA+B;AAAA;;AAC7B,QAAI,CAAC,sBAASA,iBAAT,CAAL,EAAkC;AAChC,YAAM,IAAIL,SAAJ,CAAc,gDAAd,CAAN;AACD;;AAED,SAAKK,iBAAL,GAAyBA,iBAAzB;AACA,SAAKC,SAAL,GAAiB,EAAjB;AACA,SAAKC,KAAL,GAAa,gBAAMC,MAAN,CAAa;AACxBC,eAAS;AACP,sBAAc;AADP;AADe,KAAb,CAAb;;AAMAjB,UAAM,cAAN;AACD;;AAED;;;;;;;;;;mCAMsC;AAAA;;AAAA,UAAzBkB,IAAyB,uEAAlB,gBAAkB;;AACpC;AACA,aAAOC,QAAQC,OAAR,GAAkBC,IAAlB,CAAuB;AAAA,eAAMF,QAAQG,GAAR,CAAY,0BACvC,SADuC,8BAEvC,aAFuC,IAAZ,CAAN;AAAA,OAAvB,EAIND,IAJM,CAID,gBAA2B;AAAA;AAAA,YAAzBE,OAAyB;AAAA,YAAhBC,UAAgB;;AAC/B,YAAMC,MAAMF,SAAZ;AACAE,YAAIC,GAAJ,CAAQF,WAAWG,UAAX,CAAsB,EAAEC,UAAU,KAAZ,EAAtB,CAAR;AACAH,YAAII,IAAJ,CAASX,IAAT,EAAe,MAAKY,iBAAL,EAAf;;AAEA9B,cAAM,2BAAN,EAAmCkB,IAAnC;;AAEA,eAAO,eAAKa,YAAL,CAAkBN,GAAlB,CAAP;AACD,OAZM,CAAP;AAaD;;;0BAEKO,I,EAAM;AAAA;;AACV,aAAO,KAAKD,YAAL,GACJV,IADI,CACC;AAAA,eAAU,IAAIF,OAAJ,CAAY,UAACC,OAAD,EAAUa,MAAV,EAAqB;AAC/C,iBAAKC,MAAL,GAAcA,MAAd;AACAA,iBAAOC,EAAP,CAAU,OAAV,EAAmBF,MAAnB;AACAC,iBAAOE,MAAP,CAAcJ,IAAd,EAAoB;AAAA,mBAAMZ,QAAQc,MAAR,CAAN;AAAA,WAApB;AACAlC,gBAAM,2BAAN,EAAmCgC,IAAnC;AACD,SALe,CAAV;AAAA,OADD,CAAP;AAOD;;;2BAEM;AAAA;;AACL,aAAO,IAAIb,OAAJ,CAAY,UAACC,OAAD,EAAUa,MAAV,EAAqB;AACtC,YAAI,OAAKC,MAAT,EAAiB;AACf,iBAAKA,MAAL,CAAYG,KAAZ,CAAkB,UAACC,KAAD,EAAW;AAC3B,mBAAO,OAAKJ,MAAZ;AACA,gBAAII,KAAJ,EAAW;AACTL,qBAAOK,KAAP;AACD,aAFD,MAEO;AACLlB;AACD;AACF,WAPD;AAQD,SATD,MASO;AACLa,iBAAO,IAAIM,KAAJ,CAAU,gEAAV,CAAP;AACD;AACF,OAbM,CAAP;AAcD;;;wCAEmB;AAClB,aAAO,gDAAwB,IAAxB,CAAP;AACD;;;2BAEMrC,mB,EAAqBsC,Q,EAAU;AACpC,UAAM9B,oBAAoBT,0BAA0BC,mBAA1B,CAA1B;;AAEA,UAAMoC,QAAQ/B,oBAAoBG,iBAApB,KACZD,0BAA0BC,iBAA1B,CADF;AAEA,UAAI4B,KAAJ,EAAW;AACTtC,cAAM,oCAAN,EAA4CsC,MAAMG,OAAlD;AACA,cAAMH,KAAN;AACD;;AAED,aAAO,KAAKI,gBAAL,CAAsBhC,iBAAtB,EAAyC8B,QAAzC,CAAP;AACD;;;4BAEOtC,mB,EAAqBsC,Q,EAAU;AACrC,UAAMG,qBAAqB1C,0BAA0BC,mBAA1B,CAA3B;;AAEA,UAAMoC,QAAQ/B,oBAAoBoC,kBAApB,CAAd;AACA,UAAIL,KAAJ,EAAW;AACTtC,cAAM,qCAAN,EAA6CsC,MAAMG,OAAnD;AACA,cAAMH,KAAN;AACD;;AAED,aAAO,KAAKI,gBAAL,CAAsBC,kBAAtB,EAA0CH,QAA1C,CAAP;AACD;;AAED;;;;qCAEiBI,W,EAAaJ,Q,EAAU;AACtC;AACA,UAAI,CAAC,sBAAWA,QAAX,CAAL,EAA2B;AACzBxC,cAAM,sDAAN;AACA,cAAM,IAAIQ,SAAJ,CAAc,6BAAd,CAAN;AACD;;AAED,WAAKM,SAAL,CAAe+B,IAAf,CAAoB,CAACD,WAAD,EAAcJ,QAAd,CAApB;;AAEA,aAAO,IAAP;AACD;;;6BAEQM,O,EAAS;AAAA;;AAChB,UAAMC,SAASD,QAAQE,OAAR,IAAmBF,QAAQE,OAAR,CAAgB,CAAhB,CAAlC;AACA;AACA,UAAIC,SAAS,EAAEC,QAAQ,GAAV,EAAb;AACA,UAAMC,UAAU,SAAVA,OAAU,CAACV,OAAD,EAAa;AAC3BzC,cAAM,wBAAN;AACA,eAAO,OAAKe,KAAL,CAAWc,IAAX,CAAgBiB,QAAQM,YAAxB,EAAsCX,OAAtC,CAAP;AACD,OAHD;;AAKA,WAAK3B,SAAL,CAAeuC,IAAf,CAAoB,iBAAuB;AAAA;AAAA,YAArBT,WAAqB;AAAA,YAARU,EAAQ;;AACzC;AACA;AACA;AACA,YAAIC,uBAAJ;;AAEA,YAAIX,YAAYxC,UAAhB,EAA4B;AAC1B,cAAI,sBAASwC,YAAYxC,UAArB,KAAoC0C,QAAQU,WAAR,KAAwBZ,YAAYxC,UAA5E,EAAwF;AACtF,mBAAO,KAAP;AACD;AACD,cAAI,sBAASwC,YAAYxC,UAArB,KAAoC,CAACwC,YAAYxC,UAAZ,CAAuBqD,IAAvB,CAA4BX,QAAQU,WAApC,CAAzC,EAA2F;AACzF,mBAAO,KAAP;AACD;AACF;;AAED,YAAIT,UAAUH,YAAYjC,IAAtB,IAA8BiC,YAAYjC,IAAZ,KAAqBoC,OAAOpC,IAA9D,EAAoE;AAClE,iBAAO,KAAP;AACD;;AAED,YAAI,YAAYiC,WAAZ,KAEIA,YAAYc,MAAZ,IAAsB,CAACZ,QAAQa,aAAhC,IACC,CAACf,YAAYc,MAAb,IAAuBZ,QAAQa,aAHnC,CAAJ,EAKK;AACH,iBAAO,KAAP;AACD;;AAED,YAAI;AACFJ,2BAAiBD,GAAGM,IAAH,SAAcd,OAAd,EAAuBK,OAAvB,CAAjB;AACD,SAFD,CAEE,OAAOb,KAAP,EAAc;AACdtC,gBAAM,oBAAN,EAA4BsC,KAA5B;AACAW,mBAAS,EAAEC,QAAQ,GAAV,EAAT;AACA,iBAAO,IAAP;AACD;;AAED,YAAIK,cAAJ,EAAoB;AAClB;AACA,cAAI,OAAOA,eAAelC,IAAtB,KAA+B,UAAnC,EAA+C;AAC7CkC,2BAAelC,IAAf,CAAoB8B,OAApB,EAA6BU,KAA7B,CAAmC,UAACvB,KAAD,EAAW;AAC5CtC,oBAAM,sDAAN,EACM8C,QAAQU,WADd,EAC2BlB,MAAMG,OADjC;AAED,aAHD;AAIA,mBAAO,IAAP;AACD;AACDQ,mBAAS,EAAEC,QAAQ,GAAV,EAAeY,SAASP,cAAxB,EAAT;AACA,iBAAO,IAAP;AACD;AACD,eAAO,IAAP;AACD,OAjDD;;AAmDA,aAAON,MAAP;AACD;;;;;;kBAhLkBrC,mB","file":"adapter.js","sourcesContent":["import http from 'http';\nimport axios from 'axios';\nimport isString from 'lodash.isstring';\nimport isPlainObject from 'lodash.isplainobject';\nimport isRegExp from 'lodash.isregexp';\nimport isFunction from 'lodash.isfunction';\nimport debugFactory from 'debug';\nimport { createExpressMiddleware } from './express-middleware';\nimport { packageIdentifier } from './util';\n\nconst debug = debugFactory('@slack/interactive-messages:adapter');\n\n/**\n * Transforms various forms of matching constraints to a single standard object shape\n * @param {string|RegExp|Object} matchingConstraints - the various forms of matching constraints\n * accepted\n * @returns {Object} - an object where each matching constraint is a property\n */\nfunction formatMatchingConstraints(matchingConstraints) {\n  let ret = {};\n  if (!isPlainObject(matchingConstraints)) {\n    ret.callbackId = matchingConstraints;\n  } else {\n    ret = Object.assign({}, matchingConstraints);\n  }\n  return ret;\n}\n\n/**\n * Validates general properties of a matching constraints object\n * @param {Object} matchingConstraints - object describing the constraints on a callback\n * @return {Error|false} - a false value represents successful validation, otherwise an error to\n * describe why validation failed.\n */\nfunction validateConstraints(matchingConstraints) {\n  if (matchingConstraints.callbackId &&\n      !(isString(matchingConstraints.callbackId) || isRegExp(matchingConstraints.callbackId))) {\n    return new TypeError('Callback ID must be a string or RegExp');\n  }\n\n  return false;\n}\n\n/**\n * Validates properties of a matching constraints object specific to registering an action\n * @param {Object} matchingConstraints - object describing the constraints on a callback\n * @return {Error|false} - a false value represents successful validation, otherwise an error to\n * describe why validation failed.\n */\nfunction validateActionConstraints(actionConstraints) {\n  if (actionConstraints.type &&\n      !(actionConstraints.type === 'select' || actionConstraints.type === 'button')) {\n    return new TypeError('Type must be \\'select\\' or \\'button\\'');\n  }\n\n  // We don't need to validate unfurl, we'll just cooerce it to a boolean\n  return false;\n}\n\nexport default class SlackMessageAdapter {\n  /**\n   * Create a message adapter.\n   *\n   * @param {string} verificationToken - Slack app verification token used to authenticate request\n   */\n  constructor(verificationToken) {\n    if (!isString(verificationToken)) {\n      throw new TypeError('SlackMessageAdapter needs a verification token');\n    }\n\n    this.verificationToken = verificationToken;\n    this.callbacks = [];\n    this.axios = axios.create({\n      headers: {\n        'User-Agent': packageIdentifier(),\n      },\n    });\n\n    debug('instantiated');\n  }\n\n  /**\n   * Create a server that's ready to serve requests from Slack's interactive messages.\n   *\n   * @param {string} [path=/slack/actions] - The path portion of the URL where the server will\n   * listen for requests from Slack's interactive messages.\n   */\n  createServer(path = '/slack/actions') {\n    // TODO: more options (like https)\n    return Promise.resolve().then(() => Promise.all([\n      import('express'),\n      import('body-parser'),\n    ]))\n    .then(([express, bodyParser]) => {\n      const app = express();\n      app.use(bodyParser.urlencoded({ extended: false }));\n      app.post(path, this.expressMiddleware());\n\n      debug('server created - path: %s', path);\n\n      return http.createServer(app);\n    });\n  }\n\n  start(port) {\n    return this.createServer()\n      .then(server => new Promise((resolve, reject) => {\n        this.server = server;\n        server.on('error', reject);\n        server.listen(port, () => resolve(server));\n        debug('server started - port: %s', port);\n      }));\n  }\n\n  stop() {\n    return new Promise((resolve, reject) => {\n      if (this.server) {\n        this.server.close((error) => {\n          delete this.server;\n          if (error) {\n            reject(error);\n          } else {\n            resolve();\n          }\n        });\n      } else {\n        reject(new Error('SlackMessageAdapter cannot stop when it did not start a server'));\n      }\n    });\n  }\n\n  expressMiddleware() {\n    return createExpressMiddleware(this);\n  }\n\n  action(matchingConstraints, callback) {\n    const actionConstraints = formatMatchingConstraints(matchingConstraints);\n\n    const error = validateConstraints(actionConstraints) ||\n      validateActionConstraints(actionConstraints);\n    if (error) {\n      debug('action could not be registered: %s', error.message);\n      throw error;\n    }\n\n    return this.registerCallback(actionConstraints, callback);\n  }\n\n  options(matchingConstraints, callback) {\n    const optionsConstraints = formatMatchingConstraints(matchingConstraints);\n\n    const error = validateConstraints(optionsConstraints);\n    if (error) {\n      debug('options could not be registered: %s', error.message);\n      throw error;\n    }\n\n    return this.registerCallback(optionsConstraints, callback);\n  }\n\n  /* @private */\n\n  registerCallback(constraints, callback) {\n    // Validation\n    if (!isFunction(callback)) {\n      debug('did not register callback because its not a function');\n      throw new TypeError('callback must be a function');\n    }\n\n    this.callbacks.push([constraints, callback]);\n\n    return this;\n  }\n\n  dispatch(payload) {\n    const action = payload.actions && payload.actions[0];\n    // The following result value represents \"no replacement\"\n    let result = { status: 200 };\n    const respond = (message) => {\n      debug('sending async response');\n      return this.axios.post(payload.response_url, message);\n    };\n\n    this.callbacks.some(([constraints, fn]) => {\n      // Returning false in this function continues the iteration, and returning true ends it.\n      // The pattern is that we assign a value to `result` and then return true. We only desire one\n      // result for the response.\n      let callbackResult;\n\n      if (constraints.callbackId) {\n        if (isString(constraints.callbackId) && payload.callback_id !== constraints.callbackId) {\n          return false;\n        }\n        if (isRegExp(constraints.callbackId) && !constraints.callbackId.test(payload.callback_id)) {\n          return false;\n        }\n      }\n\n      if (action && constraints.type && constraints.type !== action.type) {\n        return false;\n      }\n\n      if ('unfurl' in constraints &&\n           (\n             (constraints.unfurl && !payload.is_app_unfurl) ||\n             (!constraints.unfurl && payload.is_app_unfurl)\n           )\n         ) {\n        return false;\n      }\n\n      try {\n        callbackResult = fn.call(this, payload, respond);\n      } catch (error) {\n        debug('callback error: %o', error);\n        result = { status: 500 };\n        return true;\n      }\n\n      if (callbackResult) {\n        // Checking for Promise type\n        if (typeof callbackResult.then === 'function') {\n          callbackResult.then(respond).catch((error) => {\n            debug('async error for callback. callback_id: %s, error: %s',\n                  payload.callback_id, error.message);\n          });\n          return true;\n        }\n        result = { status: 200, content: callbackResult };\n        return true;\n      }\n      return true;\n    });\n\n    return result;\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/adapter.js"],"names":["debug","formatMatchingConstraints","matchingConstraints","ret","callbackId","Object","assign","validateConstraints","TypeError","validateActionConstraints","actionConstraints","type","SlackMessageAdapter","verificationToken","callbacks","axios","create","headers","path","Promise","resolve","then","all","express","bodyParser","app","use","urlencoded","extended","post","expressMiddleware","createServer","port","reject","server","on","listen","close","error","Error","callback","message","registerCallback","optionsConstraints","constraints","push","payload","action","actions","result","status","dialogPromiseResolve","respond","response_url","some","fn","callbackResult","callback_id","test","unfurl","is_app_unfurl","call","ms","content","catch"],"mappings":"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","file":"adapter.js","sourcesContent":["import http from 'http';\nimport axios from 'axios';\nimport isString from 'lodash.isstring';\nimport isPlainObject from 'lodash.isplainobject';\nimport isRegExp from 'lodash.isregexp';\nimport isFunction from 'lodash.isfunction';\nimport debugFactory from 'debug';\nimport { createExpressMiddleware } from './express-middleware';\nimport { packageIdentifier, promiseTimeout } from './util';\n\nconst debug = debugFactory('@slack/interactive-messages:adapter');\n\n/**\n * Transforms various forms of matching constraints to a single standard object shape\n * @param {string|RegExp|Object} matchingConstraints - the various forms of matching constraints\n * accepted\n * @returns {Object} - an object where each matching constraint is a property\n */\nfunction formatMatchingConstraints(matchingConstraints) {\n  let ret = {};\n  if (!isPlainObject(matchingConstraints)) {\n    ret.callbackId = matchingConstraints;\n  } else {\n    ret = Object.assign({}, matchingConstraints);\n  }\n  return ret;\n}\n\n/**\n * Validates general properties of a matching constraints object\n * @param {Object} matchingConstraints - object describing the constraints on a callback\n * @return {Error|false} - a false value represents successful validation, otherwise an error to\n * describe why validation failed.\n */\nfunction validateConstraints(matchingConstraints) {\n  if (matchingConstraints.callbackId &&\n      !(isString(matchingConstraints.callbackId) || isRegExp(matchingConstraints.callbackId))) {\n    return new TypeError('Callback ID must be a string or RegExp');\n  }\n\n  return false;\n}\n\n/**\n * Validates properties of a matching constraints object specific to registering an action\n * @param {Object} matchingConstraints - object describing the constraints on a callback\n * @return {Error|false} - a false value represents successful validation, otherwise an error to\n * describe why validation failed.\n */\nfunction validateActionConstraints(actionConstraints) {\n  if (actionConstraints.type &&\n      !(actionConstraints.type === 'select' || actionConstraints.type === 'button')) {\n    return new TypeError('Type must be \\'select\\' or \\'button\\'');\n  }\n\n  // We don't need to validate unfurl, we'll just cooerce it to a boolean\n  return false;\n}\n\nexport default class SlackMessageAdapter {\n  /**\n   * Create a message adapter.\n   *\n   * @param {string} verificationToken - Slack app verification token used to authenticate request\n   */\n  constructor(verificationToken) {\n    if (!isString(verificationToken)) {\n      throw new TypeError('SlackMessageAdapter needs a verification token');\n    }\n\n    this.verificationToken = verificationToken;\n    this.callbacks = [];\n    this.axios = axios.create({\n      headers: {\n        'User-Agent': packageIdentifier(),\n      },\n    });\n\n    debug('instantiated');\n  }\n\n  /**\n   * Create a server that's ready to serve requests from Slack's interactive messages.\n   *\n   * @param {string} [path=/slack/actions] - The path portion of the URL where the server will\n   * listen for requests from Slack's interactive messages.\n   */\n  createServer(path = '/slack/actions') {\n    // TODO: more options (like https)\n    return Promise.resolve().then(() => Promise.all([\n      import('express'),\n      import('body-parser'),\n    ]))\n    .then(([express, bodyParser]) => {\n      const app = express();\n      app.use(bodyParser.urlencoded({ extended: false }));\n      app.post(path, this.expressMiddleware());\n\n      debug('server created - path: %s', path);\n\n      return http.createServer(app);\n    });\n  }\n\n  start(port) {\n    return this.createServer()\n      .then(server => new Promise((resolve, reject) => {\n        this.server = server;\n        server.on('error', reject);\n        server.listen(port, () => resolve(server));\n        debug('server started - port: %s', port);\n      }));\n  }\n\n  stop() {\n    return new Promise((resolve, reject) => {\n      if (this.server) {\n        this.server.close((error) => {\n          delete this.server;\n          if (error) {\n            reject(error);\n          } else {\n            resolve();\n          }\n        });\n      } else {\n        reject(new Error('SlackMessageAdapter cannot stop when it did not start a server'));\n      }\n    });\n  }\n\n  expressMiddleware() {\n    return createExpressMiddleware(this);\n  }\n\n  action(matchingConstraints, callback) {\n    const actionConstraints = formatMatchingConstraints(matchingConstraints);\n\n    const error = validateConstraints(actionConstraints) ||\n      validateActionConstraints(actionConstraints);\n    if (error) {\n      debug('action could not be registered: %s', error.message);\n      throw error;\n    }\n\n    return this.registerCallback(actionConstraints, callback);\n  }\n\n  options(matchingConstraints, callback) {\n    const optionsConstraints = formatMatchingConstraints(matchingConstraints);\n\n    const error = validateConstraints(optionsConstraints);\n    if (error) {\n      debug('options could not be registered: %s', error.message);\n      throw error;\n    }\n\n    return this.registerCallback(optionsConstraints, callback);\n  }\n\n  /* @private */\n\n  registerCallback(constraints, callback) {\n    // Validation\n    if (!isFunction(callback)) {\n      debug('did not register callback because its not a function');\n      throw new TypeError('callback must be a function');\n    }\n\n    this.callbacks.push([constraints, callback]);\n\n    return this;\n  }\n\n  dispatch(payload) {\n    const action = payload.actions && payload.actions[0];\n\n    // The following result value represents \"no replacement\"\n    let result = { status: 200 };\n    // when the matcher finds a dialog submission, it will populate this value with a function\n    let dialogPromiseResolve;\n    const respond = (message) => {\n      if (payload.response_url) {\n        debug('sending async response');\n        return this.axios.post(payload.response_url, message);\n      } else if (dialogPromiseResolve) {\n        dialogPromiseResolve(message);\n      }\n      return true;\n    };\n\n    this.callbacks.some(([constraints, fn]) => {\n      // Returning false in this function continues the iteration, and returning true ends it.\n      // The pattern is that we assign a value to `result` and then return true. We only desire one\n      // result for the response.\n      let callbackResult;\n\n      if (constraints.callbackId) {\n        if (isString(constraints.callbackId) && payload.callback_id !== constraints.callbackId) {\n          return false;\n        }\n        if (isRegExp(constraints.callbackId) && !constraints.callbackId.test(payload.callback_id)) {\n          return false;\n        }\n      }\n\n      if (action && constraints.type && constraints.type !== action.type) {\n        return false;\n      }\n\n      if ('unfurl' in constraints &&\n           (\n             (constraints.unfurl && !payload.is_app_unfurl) ||\n             (!constraints.unfurl && payload.is_app_unfurl)\n           )\n         ) {\n        return false;\n      }\n\n      try {\n        callbackResult = fn.call(this, payload, respond);\n      } catch (error) {\n        debug('callback error: %o', error);\n        result = { status: 500 };\n        return true;\n      }\n\n      // Dialog submissions must be responded to in under 3 seconds\n      // Setting timeout to  2.5 seconds to account for propagation\n      if (payload.type === 'dialog_submission') {\n        const ms = 2500;\n        if (callbackResult) {\n          result = { status: 200, content: promiseTimeout(ms, callbackResult) };\n        } else {\n          result = {\n            status: 200, content: new Promise((resolve) => { dialogPromiseResolve = resolve; }),\n          };\n        }\n        return true;\n      }\n\n      if (callbackResult) {\n        // Checking for Promise type\n        if (typeof callbackResult.then === 'function') {\n          callbackResult.then(respond).catch((error) => {\n            debug('async error for callback. callback_id: %s, error: %s',\n                  payload.callback_id, error.message);\n          });\n          return true;\n        }\n        result = { status: 200, content: callbackResult };\n        return true;\n      }\n      return true;\n    });\n\n    return result;\n  }\n}\n"]}
//# sourceMappingURL=adapter.js.map

36

dist/express-middleware.js

@@ -29,24 +29,26 @@ 'use strict';

function sendResponse(res) {
return function _sendResponse(_ref) {
var status = _ref.status,
content = _ref.content;
return function _sendResponse(dispatchResult) {
var status = dispatchResult.status,
content = dispatchResult.content;
debug('sending response - status: %s, content: %o', status, content);
return new Promise(function (resolve, reject) {
var contentReady = content && typeof content.then === 'function' ? content : Promise.resolve(content);
return contentReady.then(function (c) {
res.status(status);
res.set('X-Slack-Powered-By', poweredBy);
if (content) {
res.json(content);
if (c) {
res.json(c);
} else {
res.end();
}
res.on('finish', function () {
// res._headers is an undocumented property, but we feel comfortable using it because:
// 1. express depends on it and express is so foundational in node
// 2. this is logging code and the risk of this causing a break is minimal
// eslint-disable-next-line no-underscore-dangle
debug('response finished - status: %d, headers: %o', res.statusCode, res._headers);
resolve(res);
return new Promise(function (resolve, reject) {
res.on('finish', function () {
// res._headers is an undocumented property, but we feel comfortable using it because:
// 1. express depends on it and express is so foundational in node
// 2. this is logging code and the risk of this causing a break is minimal
// eslint-disable-next-line no-underscore-dangle
debug('response finished - status: %d, headers: %o', res.statusCode, res._headers);
resolve(res);
});
res.on('error', reject);
});
res.on('error', reject);
});

@@ -57,3 +59,3 @@ };

return function slackMessageAdapterMiddleware(req, res, next) {
debug('request recieved - method: %s, path: %s', req.method, req.path);
debug('request received - method: %s, path: %s', req.method, req.path);

@@ -92,3 +94,3 @@ // Bind a response function to this request's respond object. This may be used in a number of

}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9leHByZXNzLW1pZGRsZXdhcmUuanMiXSwibmFtZXMiOlsiY3JlYXRlRXhwcmVzc01pZGRsZXdhcmUiLCJlcnJvckNvZGVzIiwiTk9fQk9EWV9QQVJTRVIiLCJUT0tFTl9WRVJJRklDQVRJT05fRkFJTFVSRSIsImRlYnVnIiwiYWRhcHRlciIsInBvd2VyZWRCeSIsInNlbmRSZXNwb25zZSIsInJlcyIsIl9zZW5kUmVzcG9uc2UiLCJzdGF0dXMiLCJjb250ZW50IiwiUHJvbWlzZSIsInJlc29sdmUiLCJyZWplY3QiLCJzZXQiLCJqc29uIiwiZW5kIiwib24iLCJzdGF0dXNDb2RlIiwiX2hlYWRlcnMiLCJzbGFja01lc3NhZ2VBZGFwdGVyTWlkZGxld2FyZSIsInJlcSIsIm5leHQiLCJtZXRob2QiLCJwYXRoIiwicmVzcG9uZCIsImJvZHkiLCJlcnJvciIsIkVycm9yIiwiY29kZSIsInNzbF9jaGVjayIsInBheWxvYWQiLCJKU09OIiwicGFyc2UiLCJ0b2tlbiIsInZlcmlmaWNhdGlvblRva2VuIiwiZGlzcGF0Y2giLCJjYXRjaCJdLCJtYXBwaW5ncyI6Ijs7Ozs7O1FBVWdCQSx1QixHQUFBQSx1Qjs7QUFWaEI7Ozs7QUFDQTs7OztBQUVPLElBQU1DLGtDQUFhO0FBQ3hCQyxrQkFBZ0IsdUNBRFE7QUFFeEJDLDhCQUE0QjtBQUZKLENBQW5COztBQUtQLElBQU1DLFFBQVEscUJBQWEsZ0RBQWIsQ0FBZDs7QUFFTyxTQUFTSix1QkFBVCxDQUFpQ0ssT0FBakMsRUFBMEM7QUFDL0MsTUFBTUMsWUFBWSw4QkFBbEI7O0FBRUE7QUFDQSxXQUFTQyxZQUFULENBQXNCQyxHQUF0QixFQUEyQjtBQUN6QixXQUFPLFNBQVNDLGFBQVQsT0FBNEM7QUFBQSxVQUFuQkMsTUFBbUIsUUFBbkJBLE1BQW1CO0FBQUEsVUFBWEMsT0FBVyxRQUFYQSxPQUFXOztBQUNqRFAsWUFBTSw0Q0FBTixFQUFvRE0sTUFBcEQsRUFBNERDLE9BQTVEO0FBQ0EsYUFBTyxJQUFJQyxPQUFKLENBQVksVUFBQ0MsT0FBRCxFQUFVQyxNQUFWLEVBQXFCO0FBQ3RDTixZQUFJRSxNQUFKLENBQVdBLE1BQVg7QUFDQUYsWUFBSU8sR0FBSixDQUFRLG9CQUFSLEVBQThCVCxTQUE5QjtBQUNBLFlBQUlLLE9BQUosRUFBYTtBQUNYSCxjQUFJUSxJQUFKLENBQVNMLE9BQVQ7QUFDRCxTQUZELE1BRU87QUFDTEgsY0FBSVMsR0FBSjtBQUNEO0FBQ0RULFlBQUlVLEVBQUosQ0FBTyxRQUFQLEVBQWlCLFlBQU07QUFDckI7QUFDQTtBQUNBO0FBQ0E7QUFDQWQsZ0JBQU0sNkNBQU4sRUFBcURJLElBQUlXLFVBQXpELEVBQXFFWCxJQUFJWSxRQUF6RTtBQUNBUCxrQkFBUUwsR0FBUjtBQUNELFNBUEQ7QUFRQUEsWUFBSVUsRUFBSixDQUFPLE9BQVAsRUFBZ0JKLE1BQWhCO0FBQ0QsT0FqQk0sQ0FBUDtBQWtCRCxLQXBCRDtBQXFCRDs7QUFFRCxTQUFPLFNBQVNPLDZCQUFULENBQXVDQyxHQUF2QyxFQUE0Q2QsR0FBNUMsRUFBaURlLElBQWpELEVBQXVEO0FBQzVEbkIsVUFBTSx5Q0FBTixFQUFpRGtCLElBQUlFLE1BQXJELEVBQTZERixJQUFJRyxJQUFqRTs7QUFFQTtBQUNBO0FBQ0EsUUFBTUMsVUFBVW5CLGFBQWFDLEdBQWIsQ0FBaEI7O0FBRUE7QUFDQSxRQUFJLENBQUNjLElBQUlLLElBQVQsRUFBZTtBQUNiLFVBQU1DLFFBQVEsSUFBSUMsS0FBSixDQUFVLHVEQUFWLENBQWQ7QUFDQUQsWUFBTUUsSUFBTixHQUFhN0IsV0FBV0MsY0FBeEI7QUFDQXFCLFdBQUtLLEtBQUw7QUFDQTtBQUNEOztBQUVELFFBQUlOLElBQUlLLElBQUosQ0FBU0ksU0FBYixFQUF3QjtBQUN0QkwsY0FBUSxFQUFFaEIsUUFBUSxHQUFWLEVBQVI7QUFDQTtBQUNEOztBQUVELFFBQU1zQixVQUFVQyxLQUFLQyxLQUFMLENBQVdaLElBQUlLLElBQUosQ0FBU0ssT0FBcEIsQ0FBaEI7O0FBRUE7QUFDQSxRQUFJLENBQUNBLFFBQVFHLEtBQVQsSUFBa0JILFFBQVFHLEtBQVIsS0FBa0I5QixRQUFRK0IsaUJBQWhELEVBQW1FO0FBQ2pFaEMsWUFBTSxvQ0FBTjtBQUNBLFVBQU13QixTQUFRLElBQUlDLEtBQUosQ0FBVSwrQ0FBVixDQUFkO0FBQ0FELGFBQU1FLElBQU4sR0FBYTdCLFdBQVdFLDBCQUF4QjtBQUNBb0IsV0FBS0ssTUFBTDtBQUNBO0FBQ0Q7QUFDRHhCLFVBQU0sb0NBQU47O0FBRUFzQixZQUFRckIsUUFBUWdDLFFBQVIsQ0FBaUJMLE9BQWpCLENBQVIsRUFDR00sS0FESCxDQUNTZixJQURUO0FBRUQsR0FsQ0Q7QUFtQ0QiLCJmaWxlIjoiZXhwcmVzcy1taWRkbGV3YXJlLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGRlYnVnRmFjdG9yeSBmcm9tICdkZWJ1Zyc7XG5pbXBvcnQgeyBwYWNrYWdlSWRlbnRpZmllciB9IGZyb20gJy4vdXRpbCc7XG5cbmV4cG9ydCBjb25zdCBlcnJvckNvZGVzID0ge1xuICBOT19CT0RZX1BBUlNFUjogJ1NMQUNLTUVTU0FHRU1JRERMRVdBUkVfTk9fQk9EWV9QQVJTRVInLFxuICBUT0tFTl9WRVJJRklDQVRJT05fRkFJTFVSRTogJ1NMQUNLTUVTU0FHRU1JRERMRVdBUkVfVE9LRU5fVkVSSUZJQ0FUSU9OX0ZBSUxVUkUnLFxufTtcblxuY29uc3QgZGVidWcgPSBkZWJ1Z0ZhY3RvcnkoJ0BzbGFjay9pbnRlcmFjdGl2ZS1tZXNzYWdlczpleHByZXNzLW1pZGRsZXdhcmUnKTtcblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUV4cHJlc3NNaWRkbGV3YXJlKGFkYXB0ZXIpIHtcbiAgY29uc3QgcG93ZXJlZEJ5ID0gcGFja2FnZUlkZW50aWZpZXIoKTtcblxuICAvLyBUaGlzIGZ1bmN0aW9uIGJpbmRzIGEgc3BlY2lmaWMgcmVzcG9uc2UgaW5zdGFuY2UgdG8gYSBmdW5jdGlvblxuICBmdW5jdGlvbiBzZW5kUmVzcG9uc2UocmVzKSB7XG4gICAgcmV0dXJuIGZ1bmN0aW9uIF9zZW5kUmVzcG9uc2UoeyBzdGF0dXMsIGNvbnRlbnQgfSkge1xuICAgICAgZGVidWcoJ3NlbmRpbmcgcmVzcG9uc2UgLSBzdGF0dXM6ICVzLCBjb250ZW50OiAlbycsIHN0YXR1cywgY29udGVudCk7XG4gICAgICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgICAgICByZXMuc3RhdHVzKHN0YXR1cyk7XG4gICAgICAgIHJlcy5zZXQoJ1gtU2xhY2stUG93ZXJlZC1CeScsIHBvd2VyZWRCeSk7XG4gICAgICAgIGlmIChjb250ZW50KSB7XG4gICAgICAgICAgcmVzLmpzb24oY29udGVudCk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgcmVzLmVuZCgpO1xuICAgICAgICB9XG4gICAgICAgIHJlcy5vbignZmluaXNoJywgKCkgPT4ge1xuICAgICAgICAgIC8vIHJlcy5faGVhZGVycyBpcyBhbiB1bmRvY3VtZW50ZWQgcHJvcGVydHksIGJ1dCB3ZSBmZWVsIGNvbWZvcnRhYmxlIHVzaW5nIGl0IGJlY2F1c2U6XG4gICAgICAgICAgLy8gMS4gZXhwcmVzcyBkZXBlbmRzIG9uIGl0IGFuZCBleHByZXNzIGlzIHNvIGZvdW5kYXRpb25hbCBpbiBub2RlXG4gICAgICAgICAgLy8gMi4gdGhpcyBpcyBsb2dnaW5nIGNvZGUgYW5kIHRoZSByaXNrIG9mIHRoaXMgY2F1c2luZyBhIGJyZWFrIGlzIG1pbmltYWxcbiAgICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgbm8tdW5kZXJzY29yZS1kYW5nbGVcbiAgICAgICAgICBkZWJ1ZygncmVzcG9uc2UgZmluaXNoZWQgLSBzdGF0dXM6ICVkLCBoZWFkZXJzOiAlbycsIHJlcy5zdGF0dXNDb2RlLCByZXMuX2hlYWRlcnMpO1xuICAgICAgICAgIHJlc29sdmUocmVzKTtcbiAgICAgICAgfSk7XG4gICAgICAgIHJlcy5vbignZXJyb3InLCByZWplY3QpO1xuICAgICAgfSk7XG4gICAgfTtcbiAgfVxuXG4gIHJldHVybiBmdW5jdGlvbiBzbGFja01lc3NhZ2VBZGFwdGVyTWlkZGxld2FyZShyZXEsIHJlcywgbmV4dCkge1xuICAgIGRlYnVnKCdyZXF1ZXN0IHJlY2lldmVkIC0gbWV0aG9kOiAlcywgcGF0aDogJXMnLCByZXEubWV0aG9kLCByZXEucGF0aCk7XG5cbiAgICAvLyBCaW5kIGEgcmVzcG9uc2UgZnVuY3Rpb24gdG8gdGhpcyByZXF1ZXN0J3MgcmVzcG9uZCBvYmplY3QuIFRoaXMgbWF5IGJlIHVzZWQgaW4gYSBudW1iZXIgb2ZcbiAgICAvLyBwbGFjZXNcbiAgICBjb25zdCByZXNwb25kID0gc2VuZFJlc3BvbnNlKHJlcyk7XG5cbiAgICAvLyBDaGVjayB0aGF0IHRoZSByZXF1ZXN0IGJvZHkgaGFzIGJlZW4gcGFyc2VkXG4gICAgaWYgKCFyZXEuYm9keSkge1xuICAgICAgY29uc3QgZXJyb3IgPSBuZXcgRXJyb3IoJ1RoZSBpbmNvbWluZyBIVFRQIHJlcXVlc3QgZGlkIG5vdCBoYXZlIGEgcGFyc2VkIGJvZHkuJyk7XG4gICAgICBlcnJvci5jb2RlID0gZXJyb3JDb2Rlcy5OT19CT0RZX1BBUlNFUjtcbiAgICAgIG5leHQoZXJyb3IpO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGlmIChyZXEuYm9keS5zc2xfY2hlY2spIHtcbiAgICAgIHJlc3BvbmQoeyBzdGF0dXM6IDIwMCB9KTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBwYXlsb2FkID0gSlNPTi5wYXJzZShyZXEuYm9keS5wYXlsb2FkKTtcblxuICAgIC8vIEhhbmRsZSByZXF1ZXN0IHRva2VuIHZlcmlmaWNhdGlvblxuICAgIGlmICghcGF5bG9hZC50b2tlbiB8fCBwYXlsb2FkLnRva2VuICE9PSBhZGFwdGVyLnZlcmlmaWNhdGlvblRva2VuKSB7XG4gICAgICBkZWJ1ZygncmVxdWVzdCB0b2tlbiB2ZXJpZmljYXRpb24gZmFpbHVyZScpO1xuICAgICAgY29uc3QgZXJyb3IgPSBuZXcgRXJyb3IoJ1NsYWNrIGludGVyYWN0aXZlIG1lc3NhZ2UgdmVyaWZpY2F0aW9uIGZhaWxlZCcpO1xuICAgICAgZXJyb3IuY29kZSA9IGVycm9yQ29kZXMuVE9LRU5fVkVSSUZJQ0FUSU9OX0ZBSUxVUkU7XG4gICAgICBuZXh0KGVycm9yKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgZGVidWcoJ3JlcXVlc3QgdG9rZW4gdmVyaWZpY2F0aW9uIHN1Y2Nlc3MnKTtcblxuICAgIHJlc3BvbmQoYWRhcHRlci5kaXNwYXRjaChwYXlsb2FkKSlcbiAgICAgIC5jYXRjaChuZXh0KTtcbiAgfTtcbn1cbiJdfQ==
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9leHByZXNzLW1pZGRsZXdhcmUuanMiXSwibmFtZXMiOlsiY3JlYXRlRXhwcmVzc01pZGRsZXdhcmUiLCJlcnJvckNvZGVzIiwiTk9fQk9EWV9QQVJTRVIiLCJUT0tFTl9WRVJJRklDQVRJT05fRkFJTFVSRSIsImRlYnVnIiwiYWRhcHRlciIsInBvd2VyZWRCeSIsInNlbmRSZXNwb25zZSIsInJlcyIsIl9zZW5kUmVzcG9uc2UiLCJkaXNwYXRjaFJlc3VsdCIsInN0YXR1cyIsImNvbnRlbnQiLCJjb250ZW50UmVhZHkiLCJ0aGVuIiwiUHJvbWlzZSIsInJlc29sdmUiLCJjIiwic2V0IiwianNvbiIsImVuZCIsInJlamVjdCIsIm9uIiwic3RhdHVzQ29kZSIsIl9oZWFkZXJzIiwic2xhY2tNZXNzYWdlQWRhcHRlck1pZGRsZXdhcmUiLCJyZXEiLCJuZXh0IiwibWV0aG9kIiwicGF0aCIsInJlc3BvbmQiLCJib2R5IiwiZXJyb3IiLCJFcnJvciIsImNvZGUiLCJzc2xfY2hlY2siLCJwYXlsb2FkIiwiSlNPTiIsInBhcnNlIiwidG9rZW4iLCJ2ZXJpZmljYXRpb25Ub2tlbiIsImRpc3BhdGNoIiwiY2F0Y2giXSwibWFwcGluZ3MiOiI7Ozs7OztRQVVnQkEsdUIsR0FBQUEsdUI7O0FBVmhCOzs7O0FBQ0E7Ozs7QUFFTyxJQUFNQyxrQ0FBYTtBQUN4QkMsa0JBQWdCLHVDQURRO0FBRXhCQyw4QkFBNEI7QUFGSixDQUFuQjs7QUFLUCxJQUFNQyxRQUFRLHFCQUFhLGdEQUFiLENBQWQ7O0FBRU8sU0FBU0osdUJBQVQsQ0FBaUNLLE9BQWpDLEVBQTBDO0FBQy9DLE1BQU1DLFlBQVksOEJBQWxCOztBQUVBO0FBQ0EsV0FBU0MsWUFBVCxDQUFzQkMsR0FBdEIsRUFBMkI7QUFDekIsV0FBTyxTQUFTQyxhQUFULENBQXVCQyxjQUF2QixFQUF1QztBQUFBLFVBQ3BDQyxNQURvQyxHQUNoQkQsY0FEZ0IsQ0FDcENDLE1BRG9DO0FBQUEsVUFDNUJDLE9BRDRCLEdBQ2hCRixjQURnQixDQUM1QkUsT0FENEI7O0FBRTVDLFVBQU1DLGVBQWdCRCxXQUFXLE9BQU9BLFFBQVFFLElBQWYsS0FBd0IsVUFBcEMsR0FBa0RGLE9BQWxELEdBQTRERyxRQUFRQyxPQUFSLENBQWdCSixPQUFoQixDQUFqRjtBQUNBLGFBQU9DLGFBQWFDLElBQWIsQ0FBa0IsVUFBQ0csQ0FBRCxFQUFPO0FBQzlCVCxZQUFJRyxNQUFKLENBQVdBLE1BQVg7QUFDQUgsWUFBSVUsR0FBSixDQUFRLG9CQUFSLEVBQThCWixTQUE5QjtBQUNBLFlBQUlXLENBQUosRUFBTztBQUNMVCxjQUFJVyxJQUFKLENBQVNGLENBQVQ7QUFDRCxTQUZELE1BRU87QUFDTFQsY0FBSVksR0FBSjtBQUNEO0FBQ0QsZUFBTyxJQUFJTCxPQUFKLENBQVksVUFBQ0MsT0FBRCxFQUFVSyxNQUFWLEVBQXFCO0FBQ3RDYixjQUFJYyxFQUFKLENBQU8sUUFBUCxFQUFpQixZQUFNO0FBQ3JCO0FBQ0E7QUFDQTtBQUNBO0FBQ0FsQixrQkFBTSw2Q0FBTixFQUFxREksSUFBSWUsVUFBekQsRUFBcUVmLElBQUlnQixRQUF6RTtBQUNBUixvQkFBUVIsR0FBUjtBQUNELFdBUEQ7QUFRQUEsY0FBSWMsRUFBSixDQUFPLE9BQVAsRUFBZ0JELE1BQWhCO0FBQ0QsU0FWTSxDQUFQO0FBV0QsT0FuQk0sQ0FBUDtBQW9CRCxLQXZCRDtBQXdCRDs7QUFFRCxTQUFPLFNBQVNJLDZCQUFULENBQXVDQyxHQUF2QyxFQUE0Q2xCLEdBQTVDLEVBQWlEbUIsSUFBakQsRUFBdUQ7QUFDNUR2QixVQUFNLHlDQUFOLEVBQWlEc0IsSUFBSUUsTUFBckQsRUFBNkRGLElBQUlHLElBQWpFOztBQUVBO0FBQ0E7QUFDQSxRQUFNQyxVQUFVdkIsYUFBYUMsR0FBYixDQUFoQjs7QUFFQTtBQUNBLFFBQUksQ0FBQ2tCLElBQUlLLElBQVQsRUFBZTtBQUNiLFVBQU1DLFFBQVEsSUFBSUMsS0FBSixDQUFVLHVEQUFWLENBQWQ7QUFDQUQsWUFBTUUsSUFBTixHQUFhakMsV0FBV0MsY0FBeEI7QUFDQXlCLFdBQUtLLEtBQUw7QUFDQTtBQUNEOztBQUVELFFBQUlOLElBQUlLLElBQUosQ0FBU0ksU0FBYixFQUF3QjtBQUN0QkwsY0FBUSxFQUFFbkIsUUFBUSxHQUFWLEVBQVI7QUFDQTtBQUNEOztBQUVELFFBQU15QixVQUFVQyxLQUFLQyxLQUFMLENBQVdaLElBQUlLLElBQUosQ0FBU0ssT0FBcEIsQ0FBaEI7O0FBRUE7QUFDQSxRQUFJLENBQUNBLFFBQVFHLEtBQVQsSUFBa0JILFFBQVFHLEtBQVIsS0FBa0JsQyxRQUFRbUMsaUJBQWhELEVBQW1FO0FBQ2pFcEMsWUFBTSxvQ0FBTjtBQUNBLFVBQU00QixTQUFRLElBQUlDLEtBQUosQ0FBVSwrQ0FBVixDQUFkO0FBQ0FELGFBQU1FLElBQU4sR0FBYWpDLFdBQVdFLDBCQUF4QjtBQUNBd0IsV0FBS0ssTUFBTDtBQUNBO0FBQ0Q7QUFDRDVCLFVBQU0sb0NBQU47O0FBRUEwQixZQUFRekIsUUFBUW9DLFFBQVIsQ0FBaUJMLE9BQWpCLENBQVIsRUFDR00sS0FESCxDQUNTZixJQURUO0FBRUQsR0FsQ0Q7QUFtQ0QiLCJmaWxlIjoiZXhwcmVzcy1taWRkbGV3YXJlLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGRlYnVnRmFjdG9yeSBmcm9tICdkZWJ1Zyc7XG5pbXBvcnQgeyBwYWNrYWdlSWRlbnRpZmllciB9IGZyb20gJy4vdXRpbCc7XG5cbmV4cG9ydCBjb25zdCBlcnJvckNvZGVzID0ge1xuICBOT19CT0RZX1BBUlNFUjogJ1NMQUNLTUVTU0FHRU1JRERMRVdBUkVfTk9fQk9EWV9QQVJTRVInLFxuICBUT0tFTl9WRVJJRklDQVRJT05fRkFJTFVSRTogJ1NMQUNLTUVTU0FHRU1JRERMRVdBUkVfVE9LRU5fVkVSSUZJQ0FUSU9OX0ZBSUxVUkUnLFxufTtcblxuY29uc3QgZGVidWcgPSBkZWJ1Z0ZhY3RvcnkoJ0BzbGFjay9pbnRlcmFjdGl2ZS1tZXNzYWdlczpleHByZXNzLW1pZGRsZXdhcmUnKTtcblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUV4cHJlc3NNaWRkbGV3YXJlKGFkYXB0ZXIpIHtcbiAgY29uc3QgcG93ZXJlZEJ5ID0gcGFja2FnZUlkZW50aWZpZXIoKTtcblxuICAvLyBUaGlzIGZ1bmN0aW9uIGJpbmRzIGEgc3BlY2lmaWMgcmVzcG9uc2UgaW5zdGFuY2UgdG8gYSBmdW5jdGlvblxuICBmdW5jdGlvbiBzZW5kUmVzcG9uc2UocmVzKSB7XG4gICAgcmV0dXJuIGZ1bmN0aW9uIF9zZW5kUmVzcG9uc2UoZGlzcGF0Y2hSZXN1bHQpIHtcbiAgICAgIGNvbnN0IHsgc3RhdHVzLCBjb250ZW50IH0gPSBkaXNwYXRjaFJlc3VsdDtcbiAgICAgIGNvbnN0IGNvbnRlbnRSZWFkeSA9IChjb250ZW50ICYmIHR5cGVvZiBjb250ZW50LnRoZW4gPT09ICdmdW5jdGlvbicpID8gY29udGVudCA6IFByb21pc2UucmVzb2x2ZShjb250ZW50KTtcbiAgICAgIHJldHVybiBjb250ZW50UmVhZHkudGhlbigoYykgPT4ge1xuICAgICAgICByZXMuc3RhdHVzKHN0YXR1cyk7XG4gICAgICAgIHJlcy5zZXQoJ1gtU2xhY2stUG93ZXJlZC1CeScsIHBvd2VyZWRCeSk7XG4gICAgICAgIGlmIChjKSB7XG4gICAgICAgICAgcmVzLmpzb24oYyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgcmVzLmVuZCgpO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICAgICAgcmVzLm9uKCdmaW5pc2gnLCAoKSA9PiB7XG4gICAgICAgICAgICAvLyByZXMuX2hlYWRlcnMgaXMgYW4gdW5kb2N1bWVudGVkIHByb3BlcnR5LCBidXQgd2UgZmVlbCBjb21mb3J0YWJsZSB1c2luZyBpdCBiZWNhdXNlOlxuICAgICAgICAgICAgLy8gMS4gZXhwcmVzcyBkZXBlbmRzIG9uIGl0IGFuZCBleHByZXNzIGlzIHNvIGZvdW5kYXRpb25hbCBpbiBub2RlXG4gICAgICAgICAgICAvLyAyLiB0aGlzIGlzIGxvZ2dpbmcgY29kZSBhbmQgdGhlIHJpc2sgb2YgdGhpcyBjYXVzaW5nIGEgYnJlYWsgaXMgbWluaW1hbFxuICAgICAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIG5vLXVuZGVyc2NvcmUtZGFuZ2xlXG4gICAgICAgICAgICBkZWJ1ZygncmVzcG9uc2UgZmluaXNoZWQgLSBzdGF0dXM6ICVkLCBoZWFkZXJzOiAlbycsIHJlcy5zdGF0dXNDb2RlLCByZXMuX2hlYWRlcnMpO1xuICAgICAgICAgICAgcmVzb2x2ZShyZXMpO1xuICAgICAgICAgIH0pO1xuICAgICAgICAgIHJlcy5vbignZXJyb3InLCByZWplY3QpO1xuICAgICAgICB9KTtcbiAgICAgIH0pO1xuICAgIH07XG4gIH1cblxuICByZXR1cm4gZnVuY3Rpb24gc2xhY2tNZXNzYWdlQWRhcHRlck1pZGRsZXdhcmUocmVxLCByZXMsIG5leHQpIHtcbiAgICBkZWJ1ZygncmVxdWVzdCByZWNlaXZlZCAtIG1ldGhvZDogJXMsIHBhdGg6ICVzJywgcmVxLm1ldGhvZCwgcmVxLnBhdGgpO1xuXG4gICAgLy8gQmluZCBhIHJlc3BvbnNlIGZ1bmN0aW9uIHRvIHRoaXMgcmVxdWVzdCdzIHJlc3BvbmQgb2JqZWN0LiBUaGlzIG1heSBiZSB1c2VkIGluIGEgbnVtYmVyIG9mXG4gICAgLy8gcGxhY2VzXG4gICAgY29uc3QgcmVzcG9uZCA9IHNlbmRSZXNwb25zZShyZXMpO1xuXG4gICAgLy8gQ2hlY2sgdGhhdCB0aGUgcmVxdWVzdCBib2R5IGhhcyBiZWVuIHBhcnNlZFxuICAgIGlmICghcmVxLmJvZHkpIHtcbiAgICAgIGNvbnN0IGVycm9yID0gbmV3IEVycm9yKCdUaGUgaW5jb21pbmcgSFRUUCByZXF1ZXN0IGRpZCBub3QgaGF2ZSBhIHBhcnNlZCBib2R5LicpO1xuICAgICAgZXJyb3IuY29kZSA9IGVycm9yQ29kZXMuTk9fQk9EWV9QQVJTRVI7XG4gICAgICBuZXh0KGVycm9yKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBpZiAocmVxLmJvZHkuc3NsX2NoZWNrKSB7XG4gICAgICByZXNwb25kKHsgc3RhdHVzOiAyMDAgfSk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgcGF5bG9hZCA9IEpTT04ucGFyc2UocmVxLmJvZHkucGF5bG9hZCk7XG5cbiAgICAvLyBIYW5kbGUgcmVxdWVzdCB0b2tlbiB2ZXJpZmljYXRpb25cbiAgICBpZiAoIXBheWxvYWQudG9rZW4gfHwgcGF5bG9hZC50b2tlbiAhPT0gYWRhcHRlci52ZXJpZmljYXRpb25Ub2tlbikge1xuICAgICAgZGVidWcoJ3JlcXVlc3QgdG9rZW4gdmVyaWZpY2F0aW9uIGZhaWx1cmUnKTtcbiAgICAgIGNvbnN0IGVycm9yID0gbmV3IEVycm9yKCdTbGFjayBpbnRlcmFjdGl2ZSBtZXNzYWdlIHZlcmlmaWNhdGlvbiBmYWlsZWQnKTtcbiAgICAgIGVycm9yLmNvZGUgPSBlcnJvckNvZGVzLlRPS0VOX1ZFUklGSUNBVElPTl9GQUlMVVJFO1xuICAgICAgbmV4dChlcnJvcik7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGRlYnVnKCdyZXF1ZXN0IHRva2VuIHZlcmlmaWNhdGlvbiBzdWNjZXNzJyk7XG5cbiAgICByZXNwb25kKGFkYXB0ZXIuZGlzcGF0Y2gocGF5bG9hZCkpXG4gICAgICAuY2F0Y2gobmV4dCk7XG4gIH07XG59XG4iXX0=
//# sourceMappingURL=express-middleware.js.map

@@ -6,2 +6,3 @@ 'use strict';

});
exports.promiseTimeout = promiseTimeout;
exports.packageIdentifier = packageIdentifier;

@@ -23,2 +24,14 @@

function promiseTimeout(ms, promise) {
// Create a promise that rejects in <ms> milliseconds
var timeout = new Promise(function (resolve, reject) {
var id = setTimeout(function () {
clearTimeout(id);
reject('Promise timed out');
}, ms);
});
// Returns a race between our timeout and the passed in promise
return Promise.race([promise, timeout]);
}
// NOTE: before this can be an external module:

@@ -43,3 +56,3 @@ // 1. are all the JS features supported back to a reasonable version?

}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy91dGlsLmpzIl0sIm5hbWVzIjpbInBhY2thZ2VJZGVudGlmaWVyIiwiZXNjYXBlIiwicyIsInJlcGxhY2UiLCJhZGRvbnMiLCJpZGVudGlmaWVyTWFwIiwiT2JqZWN0IiwiYXNzaWduIiwibmFtZSIsInZlcnNpb24iLCJwbGF0Zm9ybSIsInJlbGVhc2UiLCJub2RlIiwicHJvY2VzcyIsImtleXMiLCJyZWR1Y2UiLCJhY2MiLCJrIl0sIm1hcHBpbmdzIjoiOzs7OztRQVlnQkEsaUIsR0FBQUEsaUI7O0FBWmhCOzs7O0FBQ0E7Ozs7OztBQUVBLFNBQVNDLE1BQVQsQ0FBZ0JDLENBQWhCLEVBQW1CO0FBQUUsU0FBT0EsRUFBRUMsT0FBRixDQUFVLEdBQVYsRUFBZSxHQUFmLEVBQW9CQSxPQUFwQixDQUE0QixHQUE1QixFQUFpQyxHQUFqQyxDQUFQO0FBQStDOztBQUVwRTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNPLFNBQVNILGlCQUFULEdBQXdDO0FBQUEsTUFBYkksTUFBYSx1RUFBSixFQUFJOztBQUM3QyxNQUFNQyxnQkFBZ0JDLE9BQU9DLE1BQVAsQ0FBYztBQUNsQyxLQUFFLEdBQUUsa0JBQUlDLElBQUssRUFBYixHQUFpQixrQkFBSUMsT0FEYTtBQUVsQyxLQUFFLEdBQUUsYUFBR0MsUUFBSCxFQUFjLEVBQWxCLEdBQXNCLGFBQUdDLE9BQUgsRUFGWTtBQUdsQ0MsVUFBTUMsUUFBUUosT0FBUixDQUFnQk4sT0FBaEIsQ0FBd0IsR0FBeEIsRUFBNkIsRUFBN0I7QUFINEIsR0FBZCxFQUluQkMsTUFKbUIsQ0FBdEI7QUFLQSxTQUFPRSxPQUFPUSxJQUFQLENBQVlULGFBQVosRUFBMkJVLE1BQTNCLENBQWtDLFVBQUNDLEdBQUQsRUFBTUMsQ0FBTjtBQUFBLFdBQWEsR0FBRUQsR0FBSSxJQUFHZixPQUFPZ0IsQ0FBUCxDQUFVLElBQUdoQixPQUFPSSxjQUFjWSxDQUFkLENBQVAsQ0FBeUIsRUFBNUQ7QUFBQSxHQUFsQyxFQUFpRyxFQUFqRyxDQUFQO0FBQ0QiLCJmaWxlIjoidXRpbC5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBvcyBmcm9tICdvcyc7XG5pbXBvcnQgcGtnIGZyb20gJy4uL3BhY2thZ2UuanNvbic7XG5cbmZ1bmN0aW9uIGVzY2FwZShzKSB7IHJldHVybiBzLnJlcGxhY2UoJy8nLCAnOicpLnJlcGxhY2UoJyAnLCAnXycpOyB9XG5cbi8vIE5PVEU6IGJlZm9yZSB0aGlzIGNhbiBiZSBhbiBleHRlcm5hbCBtb2R1bGU6XG4vLyAxLiBhcmUgYWxsIHRoZSBKUyBmZWF0dXJlcyBzdXBwb3J0ZWQgYmFjayB0byBhIHJlYXNvbmFibGUgdmVyc2lvbj9cbi8vICAgIGRlZmF1bHQgcGFyYW1zLCB0ZW1wbGF0ZSBzdHJpbmdzLCBjb21wdXRlZCBwcm9wZXJ0eSBuYW1lc1xuLy8gMi4gYWNjZXNzIHRvIGBwa2dgIHdpbGwgY2hhbmdlXG4vLyAzLiB0ZXN0c1xuLy8gdGhlcmUgd2lsbCBwb3RlbnRpYWxseSBiZSBtb3JlIG5hbWVkIGV4cG9ydHMgaW4gdGhpcyBmaWxlXG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L3ByZWZlci1kZWZhdWx0LWV4cG9ydFxuZXhwb3J0IGZ1bmN0aW9uIHBhY2thZ2VJZGVudGlmaWVyKGFkZG9ucyA9IHt9KSB7XG4gIGNvbnN0IGlkZW50aWZpZXJNYXAgPSBPYmplY3QuYXNzaWduKHtcbiAgICBbYCR7cGtnLm5hbWV9YF06IHBrZy52ZXJzaW9uLFxuICAgIFtgJHtvcy5wbGF0Zm9ybSgpfWBdOiBvcy5yZWxlYXNlKCksXG4gICAgbm9kZTogcHJvY2Vzcy52ZXJzaW9uLnJlcGxhY2UoJ3YnLCAnJyksXG4gIH0sIGFkZG9ucyk7XG4gIHJldHVybiBPYmplY3Qua2V5cyhpZGVudGlmaWVyTWFwKS5yZWR1Y2UoKGFjYywgaykgPT4gYCR7YWNjfSAke2VzY2FwZShrKX0vJHtlc2NhcGUoaWRlbnRpZmllck1hcFtrXSl9YCwgJycpO1xufVxuIl19
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy91dGlsLmpzIl0sIm5hbWVzIjpbInByb21pc2VUaW1lb3V0IiwicGFja2FnZUlkZW50aWZpZXIiLCJlc2NhcGUiLCJzIiwicmVwbGFjZSIsIm1zIiwicHJvbWlzZSIsInRpbWVvdXQiLCJQcm9taXNlIiwicmVzb2x2ZSIsInJlamVjdCIsImlkIiwic2V0VGltZW91dCIsImNsZWFyVGltZW91dCIsInJhY2UiLCJhZGRvbnMiLCJpZGVudGlmaWVyTWFwIiwiT2JqZWN0IiwiYXNzaWduIiwibmFtZSIsInZlcnNpb24iLCJwbGF0Zm9ybSIsInJlbGVhc2UiLCJub2RlIiwicHJvY2VzcyIsImtleXMiLCJyZWR1Y2UiLCJhY2MiLCJrIl0sIm1hcHBpbmdzIjoiOzs7OztRQUtnQkEsYyxHQUFBQSxjO1FBc0JBQyxpQixHQUFBQSxpQjs7QUEzQmhCOzs7O0FBQ0E7Ozs7OztBQUVBLFNBQVNDLE1BQVQsQ0FBZ0JDLENBQWhCLEVBQW1CO0FBQUUsU0FBT0EsRUFBRUMsT0FBRixDQUFVLEdBQVYsRUFBZSxHQUFmLEVBQW9CQSxPQUFwQixDQUE0QixHQUE1QixFQUFpQyxHQUFqQyxDQUFQO0FBQStDOztBQUU3RCxTQUFTSixjQUFULENBQXdCSyxFQUF4QixFQUE0QkMsT0FBNUIsRUFBcUM7QUFDMUM7QUFDQSxNQUFNQyxVQUFVLElBQUlDLE9BQUosQ0FBWSxVQUFDQyxPQUFELEVBQVVDLE1BQVYsRUFBcUI7QUFDL0MsUUFBTUMsS0FBS0MsV0FBVyxZQUFNO0FBQzFCQyxtQkFBYUYsRUFBYjtBQUNBRCxhQUFPLG1CQUFQO0FBQ0QsS0FIVSxFQUdSTCxFQUhRLENBQVg7QUFJRCxHQUxlLENBQWhCO0FBTUE7QUFDQSxTQUFPRyxRQUFRTSxJQUFSLENBQWEsQ0FDbEJSLE9BRGtCLEVBRWxCQyxPQUZrQixDQUFiLENBQVA7QUFJRDs7QUFFRDtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNPLFNBQVNOLGlCQUFULEdBQXdDO0FBQUEsTUFBYmMsTUFBYSx1RUFBSixFQUFJOztBQUM3QyxNQUFNQyxnQkFBZ0JDLE9BQU9DLE1BQVAsQ0FBYztBQUNsQyxLQUFFLEdBQUUsa0JBQUlDLElBQUssRUFBYixHQUFpQixrQkFBSUMsT0FEYTtBQUVsQyxLQUFFLEdBQUUsYUFBR0MsUUFBSCxFQUFjLEVBQWxCLEdBQXNCLGFBQUdDLE9BQUgsRUFGWTtBQUdsQ0MsVUFBTUMsUUFBUUosT0FBUixDQUFnQmhCLE9BQWhCLENBQXdCLEdBQXhCLEVBQTZCLEVBQTdCO0FBSDRCLEdBQWQsRUFJbkJXLE1BSm1CLENBQXRCO0FBS0EsU0FBT0UsT0FBT1EsSUFBUCxDQUFZVCxhQUFaLEVBQTJCVSxNQUEzQixDQUFrQyxVQUFDQyxHQUFELEVBQU1DLENBQU47QUFBQSxXQUFhLEdBQUVELEdBQUksSUFBR3pCLE9BQU8wQixDQUFQLENBQVUsSUFBRzFCLE9BQU9jLGNBQWNZLENBQWQsQ0FBUCxDQUF5QixFQUE1RDtBQUFBLEdBQWxDLEVBQWlHLEVBQWpHLENBQVA7QUFDRCIsImZpbGUiOiJ1dGlsLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IG9zIGZyb20gJ29zJztcbmltcG9ydCBwa2cgZnJvbSAnLi4vcGFja2FnZS5qc29uJztcblxuZnVuY3Rpb24gZXNjYXBlKHMpIHsgcmV0dXJuIHMucmVwbGFjZSgnLycsICc6JykucmVwbGFjZSgnICcsICdfJyk7IH1cblxuZXhwb3J0IGZ1bmN0aW9uIHByb21pc2VUaW1lb3V0KG1zLCBwcm9taXNlKSB7XG4gIC8vIENyZWF0ZSBhIHByb21pc2UgdGhhdCByZWplY3RzIGluIDxtcz4gbWlsbGlzZWNvbmRzXG4gIGNvbnN0IHRpbWVvdXQgPSBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgY29uc3QgaWQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIGNsZWFyVGltZW91dChpZCk7XG4gICAgICByZWplY3QoJ1Byb21pc2UgdGltZWQgb3V0Jyk7XG4gICAgfSwgbXMpO1xuICB9KTtcbiAgLy8gUmV0dXJucyBhIHJhY2UgYmV0d2VlbiBvdXIgdGltZW91dCBhbmQgdGhlIHBhc3NlZCBpbiBwcm9taXNlXG4gIHJldHVybiBQcm9taXNlLnJhY2UoW1xuICAgIHByb21pc2UsXG4gICAgdGltZW91dCxcbiAgXSk7XG59XG5cbi8vIE5PVEU6IGJlZm9yZSB0aGlzIGNhbiBiZSBhbiBleHRlcm5hbCBtb2R1bGU6XG4vLyAxLiBhcmUgYWxsIHRoZSBKUyBmZWF0dXJlcyBzdXBwb3J0ZWQgYmFjayB0byBhIHJlYXNvbmFibGUgdmVyc2lvbj9cbi8vICAgIGRlZmF1bHQgcGFyYW1zLCB0ZW1wbGF0ZSBzdHJpbmdzLCBjb21wdXRlZCBwcm9wZXJ0eSBuYW1lc1xuLy8gMi4gYWNjZXNzIHRvIGBwa2dgIHdpbGwgY2hhbmdlXG4vLyAzLiB0ZXN0c1xuLy8gdGhlcmUgd2lsbCBwb3RlbnRpYWxseSBiZSBtb3JlIG5hbWVkIGV4cG9ydHMgaW4gdGhpcyBmaWxlXG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgaW1wb3J0L3ByZWZlci1kZWZhdWx0LWV4cG9ydFxuZXhwb3J0IGZ1bmN0aW9uIHBhY2thZ2VJZGVudGlmaWVyKGFkZG9ucyA9IHt9KSB7XG4gIGNvbnN0IGlkZW50aWZpZXJNYXAgPSBPYmplY3QuYXNzaWduKHtcbiAgICBbYCR7cGtnLm5hbWV9YF06IHBrZy52ZXJzaW9uLFxuICAgIFtgJHtvcy5wbGF0Zm9ybSgpfWBdOiBvcy5yZWxlYXNlKCksXG4gICAgbm9kZTogcHJvY2Vzcy52ZXJzaW9uLnJlcGxhY2UoJ3YnLCAnJyksXG4gIH0sIGFkZG9ucyk7XG4gIHJldHVybiBPYmplY3Qua2V5cyhpZGVudGlmaWVyTWFwKS5yZWR1Y2UoKGFjYywgaykgPT4gYCR7YWNjfSAke2VzY2FwZShrKX0vJHtlc2NhcGUoaWRlbnRpZmllck1hcFtrXSl9YCwgJycpO1xufVxuIl19
//# sourceMappingURL=util.js.map
{
"name": "@slack/interactive-messages",
"version": "0.1.2",
"version": "0.2.0",
"description": "Slack Interactive Messages module",

@@ -5,0 +5,0 @@ "main": "dist/index.js",

@@ -30,3 +30,3 @@ # Interactive Message Adapter for Node and Express

Select the **Interactive Messages** feature, and enable it. Input your **Request URL**. If your app
will use dynamic menus, you also need to input a **Options URL**.
will use dynamic message menus, you also need to input a **Options URL**.

@@ -240,2 +240,7 @@ ![Configuring a request URL](support/interactive-messages.gif)

### Dialogs
This adapter can be used in conjunction with Slack dialogs. Because any dialog submission request requires a synchronous response, a response is needed within 3 seconds. If your asynchronous code cannot return within sufficient time, the adapter will fallback to responding with a `200 OK` successful response to avoid any user-facing errors.
### Error handling

@@ -242,0 +247,0 @@

@@ -9,3 +9,3 @@ import http from 'http';

import { createExpressMiddleware } from './express-middleware';
import { packageIdentifier } from './util';
import { packageIdentifier, promiseTimeout } from './util';

@@ -178,7 +178,15 @@ const debug = debugFactory('@slack/interactive-messages:adapter');

const action = payload.actions && payload.actions[0];
// The following result value represents "no replacement"
let result = { status: 200 };
// when the matcher finds a dialog submission, it will populate this value with a function
let dialogPromiseResolve;
const respond = (message) => {
debug('sending async response');
return this.axios.post(payload.response_url, message);
if (payload.response_url) {
debug('sending async response');
return this.axios.post(payload.response_url, message);
} else if (dialogPromiseResolve) {
dialogPromiseResolve(message);
}
return true;
};

@@ -222,2 +230,16 @@

// Dialog submissions must be responded to in under 3 seconds
// Setting timeout to 2.5 seconds to account for propagation
if (payload.type === 'dialog_submission') {
const ms = 2500;
if (callbackResult) {
result = { status: 200, content: promiseTimeout(ms, callbackResult) };
} else {
result = {
status: 200, content: new Promise((resolve) => { dialogPromiseResolve = resolve; }),
};
}
return true;
}
if (callbackResult) {

@@ -224,0 +246,0 @@ // Checking for Promise type

@@ -16,21 +16,24 @@ import debugFactory from 'debug';

function sendResponse(res) {
return function _sendResponse({ status, content }) {
debug('sending response - status: %s, content: %o', status, content);
return new Promise((resolve, reject) => {
return function _sendResponse(dispatchResult) {
const { status, content } = dispatchResult;
const contentReady = (content && typeof content.then === 'function') ? content : Promise.resolve(content);
return contentReady.then((c) => {
res.status(status);
res.set('X-Slack-Powered-By', poweredBy);
if (content) {
res.json(content);
if (c) {
res.json(c);
} else {
res.end();
}
res.on('finish', () => {
// res._headers is an undocumented property, but we feel comfortable using it because:
// 1. express depends on it and express is so foundational in node
// 2. this is logging code and the risk of this causing a break is minimal
// eslint-disable-next-line no-underscore-dangle
debug('response finished - status: %d, headers: %o', res.statusCode, res._headers);
resolve(res);
return new Promise((resolve, reject) => {
res.on('finish', () => {
// res._headers is an undocumented property, but we feel comfortable using it because:
// 1. express depends on it and express is so foundational in node
// 2. this is logging code and the risk of this causing a break is minimal
// eslint-disable-next-line no-underscore-dangle
debug('response finished - status: %d, headers: %o', res.statusCode, res._headers);
resolve(res);
});
res.on('error', reject);
});
res.on('error', reject);
});

@@ -41,3 +44,3 @@ };

return function slackMessageAdapterMiddleware(req, res, next) {
debug('request recieved - method: %s, path: %s', req.method, req.path);
debug('request received - method: %s, path: %s', req.method, req.path);

@@ -44,0 +47,0 @@ // Bind a response function to this request's respond object. This may be used in a number of

@@ -6,2 +6,17 @@ import os from 'os';

export function promiseTimeout(ms, promise) {
// Create a promise that rejects in <ms> milliseconds
const timeout = new Promise((resolve, reject) => {
const id = setTimeout(() => {
clearTimeout(id);
reject('Promise timed out');
}, ms);
});
// Returns a race between our timeout and the passed in promise
return Promise.race([
promise,
timeout,
]);
}
// NOTE: before this can be an external module:

@@ -8,0 +23,0 @@ // 1. are all the JS features supported back to a reasonable version?

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc