@kaspernj/api-maker 1.0.2122 → 1.0.2125

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -60,7 +60,8 @@ export default class ApiMakerSessionStatusUpdater {
60
60
  logger.debug(() => `Returning CSRF token after updating session status: ${this.csrfToken}`);
61
61
  return this.csrfToken;
62
62
  }
63
- throw new Error("CSRF token hasn't been set");
63
+ logger.debug("No CSRF token available after updating session status");
64
+ return undefined;
64
65
  }
65
66
  /** sessionStatus. */
66
67
  sessionStatus() {
@@ -103,13 +104,23 @@ export default class ApiMakerSessionStatusUpdater {
103
104
  updateSessionStatus = async () => {
104
105
  logger.debug("updateSessionStatus");
105
106
  const result = await this.sessionStatus();
107
+ this.applyResult(result);
108
+ };
109
+ /**
110
+ * @param {Record<string, any>} result
111
+ * @returns {void}
112
+ */
113
+ applyResult(result) {
106
114
  logger.debug(() => `Result: ${JSON.stringify(result, null, 2)}`);
107
115
  this.updateMetaElementsFromResult(result);
108
116
  this.updateUserSessionsFromResult(result);
109
- };
117
+ }
110
118
  /** updateMetaElementsFromResult. */
111
119
  updateMetaElementsFromResult(result) {
112
120
  logger.debug("updateMetaElementsFromResult");
121
+ if (!result.csrf_token) {
122
+ return;
123
+ }
113
124
  this.csrfToken = result.csrf_token;
114
125
  if (this.useMetaElement) {
115
126
  const csrfTokenElement = document.querySelector("meta[name='csrf-token']");
@@ -143,4 +154,4 @@ export default class ApiMakerSessionStatusUpdater {
143
154
  }
144
155
  }
145
156
  }
146
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"session-status-updater.js","sourceRoot":"/src/","sources":["session-status-updater.js"],"names":[],"mappings":"AAAA,OAAO,KAAK,UAAU,MAAM,YAAY,CAAA;AACxC,OAAO,MAAM,MAAM,aAAa,CAAA;AAChC,OAAO,MAAM,MAAM,aAAa,CAAA;AAChC,OAAO,MAAM,MAAM,aAAa,CAAA;AAChC,OAAO,SAAS,MAAM,YAAY,CAAA;AAElC,MAAM,MAAM,GAAG,IAAI,MAAM,CAAC,EAAC,IAAI,EAAE,iCAAiC,EAAC,CAAC,CAAA;AACpE,MAAM,MAAM,GAAG,EAAE,CAAA;AAEjB,wBAAwB;AAExB,+CAA+C;AAC/C,MAAM,CAAC,OAAO,OAAO,4BAA4B;IAC/C,eAAe;IACf,MAAM,CAAC,OAAO,CAAC,IAAI;QACjB,IAAI,CAAC,MAAM,CAAC,4BAA4B,EAAE,CAAC;YACzC,MAAM,CAAC,4BAA4B,GAAG,IAAI,4BAA4B,CAAC,IAAI,CAAC,CAAA;QAC9E,CAAC;QAED,OAAO,MAAM,CAAC,4BAA4B,CAAA;IAC5C,CAAC;IAED,mBAAmB;IACnB,YAAY,IAAI,GAAG,EAAE;QACnB,IAAI,CAAC,MAAM,GAAG,EAAE,CAAA;QAChB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,IAAI,MAAM,CAAA;QAErC,IAAI,gBAAgB,IAAI,IAAI,EAAE,CAAC;YAC7B,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAA;QAC3C,CAAC;aAAM,IAAI,OAAO,QAAQ,KAAK,WAAW,EAAE,CAAC;YAC3C,IAAI,CAAC,cAAc,GAAG,KAAK,CAAA;QAC7B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,cAAc,GAAG,IAAI,CAAA;QAC5B,CAAC;QAED,IAAI,OAAO,MAAM,IAAI,WAAW,EAAE,CAAC;YACjC,IAAI,CAAC,kBAAkB,EAAE,CAAA;QAC3B,CAAC;QAED,IAAI,CAAC,gBAAgB,EAAE,CAAA;IACzB,CAAC;IAED,0BAA0B;IAC1B,kBAAkB;QAChB,MAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,IAAI,CAAC,mBAAmB,EAAE,KAAK,CAAC,CAAA;IACpE,CAAC;IAED,wBAAwB;IACxB,gBAAgB;QACd,SAAS,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAA;IACrC,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,MAAM,CAAC,KAAK,CAAC,qCAAqC,IAAI,CAAC,SAAS,EAAE,CAAC,CAAA;YAEnE,OAAO,IAAI,CAAC,SAAS,CAAA;QACvB,CAAC;QAED,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,MAAM,gBAAgB,GAAG,QAAQ,CAAC,aAAa,CAAC,yBAAyB,CAAC,CAAA;YAE1E,IAAI,gBAAgB,EAAE,CAAC;gBACrB,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,qCAAqC,gBAAgB,CAAC,YAAY,CAAC,SAAS,CAAC,EAAE,CAAC,CAAA;gBAEnG,IAAI,CAAC,SAAS,GAAG,gBAAgB,CAAC,YAAY,CAAC,SAAS,CAAC,CAAA;gBAEzD,OAAO,IAAI,CAAC,SAAS,CAAA;YACvB,CAAC;QACH,CAAC;QAED,MAAM,CAAC,KAAK,CAAC,uDAAuD,CAAC,CAAA;QACrE,MAAM,IAAI,CAAC,mBAAmB,EAAE,CAAA;QAEhC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,uDAAuD,IAAI,CAAC,SAAS,EAAE,CAAC,CAAA;YAE3F,OAAO,IAAI,CAAC,SAAS,CAAA;QACvB,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAA;IAC/C,CAAC;IAED,qBAAqB;IACrB,aAAa;QACX,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,MAAM,IAAI,GAAG,MAAM,CAAC,OAAO,EAAE,CAAA;YAC7B,IAAI,WAAW,GAAG,EAAE,CAAA;YAEpB,IAAI,IAAI;gBAAE,WAAW,IAAI,IAAI,CAAA;YAE7B,WAAW,IAAI,6BAA6B,CAAA;YAE5C,MAAM,GAAG,GAAG,IAAI,cAAc,EAAE,CAAA;YAChC,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,WAAW,EAAE,IAAI,CAAC,CAAA;YACnC,GAAG,CAAC,MAAM,GAAG,GAAG,EAAE;gBAChB,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,CAAA;gBAC7C,OAAO,CAAC,QAAQ,CAAC,CAAA;YACnB,CAAC,CAAA;YACD,GAAG,CAAC,IAAI,EAAE,CAAA;QACZ,CAAC,CAAC,CAAA;IACJ,CAAC;IAED,mBAAmB;IACnB,WAAW,CAAC,QAAQ;QAClB,mBAAmB;QACnB,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE,QAAQ,CAAC,CAAA;IACxC,CAAC;IAED,oBAAoB;IACpB,YAAY;QACV,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAA;QAE5B,IAAI,IAAI,CAAC,aAAa;YACpB,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;QAElC,IAAI,CAAC,aAAa,GAAG,UAAU,CAC7B,GAAG,EAAE;YACH,IAAI,CAAC,YAAY,EAAE,CAAA;YACnB,IAAI,CAAC,mBAAmB,EAAE,CAAA;QAC5B,CAAC,EACD,IAAI,CAAC,OAAO,CACb,CAAA;IACH,CAAC;IAED,mBAAmB;IACnB,WAAW;QACT,IAAI,IAAI,CAAC,aAAa;YACpB,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;IACpC,CAAC;IAED,2BAA2B;IAC3B,mBAAmB,GAAG,KAAK,IAAI,EAAE;QAC/B,MAAM,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAA;QAEnC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,aAAa,EAAE,CAAA;QAEzC,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,WAAW,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC,CAAA;QAChE,IAAI,CAAC,4BAA4B,CAAC,MAAM,CAAC,CAAA;QACzC,IAAI,CAAC,4BAA4B,CAAC,MAAM,CAAC,CAAA;IAC3C,CAAC,CAAA;IAED,oCAAoC;IACpC,4BAA4B,CAAC,MAAM;QACjC,MAAM,CAAC,KAAK,CAAC,8BAA8B,CAAC,CAAA;QAE5C,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,UAAU,CAAA;QAElC,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,MAAM,gBAAgB,GAAG,QAAQ,CAAC,aAAa,CAAC,yBAAyB,CAAC,CAAA;YAE1E,IAAI,gBAAgB,EAAE,CAAC;gBACrB,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,wBAAwB,gBAAgB,CAAC,YAAY,CAAC,SAAS,CAAC,SAAS,MAAM,CAAC,UAAU,GAAG,CAAC,CAAA;gBACjH,gBAAgB,CAAC,YAAY,CAAC,SAAS,EAAE,MAAM,CAAC,UAAU,CAAC,CAAA;YAC7D,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,KAAK,CAAC,sCAAsC,CAAC,CAAA;YACtD,CAAC;QACH,CAAC;IACH,CAAC;IAED,oCAAoC;IACpC,4BAA4B,CAAC,MAAM;QACjC,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;YACtC,IAAI,CAAC,gCAAgC,CAAC,SAAS,EAAE,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAA;QAC5E,CAAC;IACH,CAAC;IAED,wCAAwC;IACxC,gCAAgC,CAAC,SAAS,EAAE,KAAK;QAC/C,MAAM,0BAA0B,GAAG,KAAK,UAAU,CAAC,QAAQ,CAAC,SAAS,CAAC,UAAU,CAAA;QAEhF,IAAI,CAAC,CAAC,0BAA0B,IAAI,MAAM,CAAC,EAAE,CAAC;YAC5C,MAAM,IAAI,KAAK,CAAC,6BAA6B,0BAA0B,EAAE,CAAC,CAAA;QAC5E,CAAC;QAED,MAAM,iBAAiB,GAAG,MAAM,CAAC,0BAA0B,CAAC,EAAE,CAAA;QAC9D,MAAM,iBAAiB,GAAG,KAAK,CAAC,SAAS,CAAA;QAEzC,IAAI,iBAAiB,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC5C,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,GAAG,UAAU,CAAC,QAAQ,CAAC,SAAS,CAAC,4CAA4C,CAAC,CAAA;YAEjG,MAAM,CAAC,YAAY,CAAC,EAAC,KAAK,EAAE,SAAS,EAAC,CAAC,CAAA;YACvC,MAAM,CAAC,gBAAgB,CAAC,EAAC,KAAK,EAAE,SAAS,EAAC,CAAC,CAAA;QAC7C,CAAC;IACH,CAAC;CACF","sourcesContent":["import * as inflection from \"inflection\"\nimport Devise from \"./devise.js\"\nimport Logger from \"./logger.js\"\nimport config from \"./config.js\"\nimport wakeEvent from \"wake-event\"\n\nconst logger = new Logger({name: \"ApiMaker / SessionStatusUpdater\"})\nconst shared = {}\n\n// logger.setDebug(true)\n\n/** Tracks session and CSRF token freshness. */\nexport default class ApiMakerSessionStatusUpdater {\n  /** current. */\n  static current(args) {\n    if (!shared.apiMakerSessionStatusUpdater) {\n      shared.apiMakerSessionStatusUpdater = new ApiMakerSessionStatusUpdater(args)\n    }\n\n    return shared.apiMakerSessionStatusUpdater\n  }\n\n  /** Constructor. */\n  constructor(args = {}) {\n    this.events = {}\n    this.timeout = args.timeout || 600000\n\n    if (\"useMetaElement\" in args) {\n      this.useMetaElement = args.useMetaElement\n    } else if (typeof document === \"undefined\") {\n      this.useMetaElement = false\n    } else {\n      this.useMetaElement = true\n    }\n\n    if (typeof window != \"undefined\") {\n      this.connectOnlineEvent()\n    }\n\n    this.connectWakeEvent()\n  }\n\n  /** connectOnlineEvent. */\n  connectOnlineEvent() {\n    window.addEventListener(\"online\", this.updateSessionStatus, false)\n  }\n\n  /** connectWakeEvent. */\n  connectWakeEvent() {\n    wakeEvent(this.updateSessionStatus)\n  }\n\n  async getCsrfToken() {\n    if (this.csrfToken) {\n      logger.debug(`Get CSRF token from set variable: ${this.csrfToken}`)\n\n      return this.csrfToken\n    }\n\n    if (this.useMetaElement) {\n      const csrfTokenElement = document.querySelector(\"meta[name='csrf-token']\")\n\n      if (csrfTokenElement) {\n        logger.debug(() => `Get CSRF token from meta element: ${csrfTokenElement.getAttribute(\"content\")}`)\n\n        this.csrfToken = csrfTokenElement.getAttribute(\"content\")\n\n        return this.csrfToken\n      }\n    }\n\n    logger.debug(\"Updating session status because no CSRF token set yet\")\n    await this.updateSessionStatus()\n\n    if (this.csrfToken) {\n      logger.debug(() => `Returning CSRF token after updating session status: ${this.csrfToken}`)\n\n      return this.csrfToken\n    }\n\n    throw new Error(\"CSRF token hasn't been set\")\n  }\n\n  /** sessionStatus. */\n  sessionStatus() {\n    return new Promise((resolve) => {\n      const host = config.getHost()\n      let requestPath = \"\"\n\n      if (host) requestPath += host\n\n      requestPath += \"/api_maker/session_statuses\"\n\n      const xhr = new XMLHttpRequest()\n      xhr.open(\"POST\", requestPath, true)\n      xhr.onload = () => {\n        const response = JSON.parse(xhr.responseText)\n        resolve(response)\n      }\n      xhr.send()\n    })\n  }\n\n  /** onSignedOut. */\n  onSignedOut(callback) {\n    // @ts-expect-error\n    this.addEvent(\"onSignedOut\", callback)\n  }\n\n  /** startTimeout. */\n  startTimeout() {\n    logger.debug(\"startTimeout\")\n\n    if (this.updateTimeout)\n      clearTimeout(this.updateTimeout)\n\n    this.updateTimeout = setTimeout(\n      () => {\n        this.startTimeout()\n        this.updateSessionStatus()\n      },\n      this.timeout\n    )\n  }\n\n  /** stopTimeout. */\n  stopTimeout() {\n    if (this.updateTimeout)\n      clearTimeout(this.updateTimeout)\n  }\n\n  /** updateSessionStatus. */\n  updateSessionStatus = async () => {\n    logger.debug(\"updateSessionStatus\")\n\n    const result = await this.sessionStatus()\n\n    logger.debug(() => `Result: ${JSON.stringify(result, null, 2)}`)\n    this.updateMetaElementsFromResult(result)\n    this.updateUserSessionsFromResult(result)\n  }\n\n  /** updateMetaElementsFromResult. */\n  updateMetaElementsFromResult(result) {\n    logger.debug(\"updateMetaElementsFromResult\")\n\n    this.csrfToken = result.csrf_token\n\n    if (this.useMetaElement) {\n      const csrfTokenElement = document.querySelector(\"meta[name='csrf-token']\")\n\n      if (csrfTokenElement) {\n        logger.debug(() => `Changing token from \"${csrfTokenElement.getAttribute(\"content\")}\" to \"${result.csrf_token}\"`)\n        csrfTokenElement.setAttribute(\"content\", result.csrf_token)\n      } else {\n        logger.debug(\"csrf token element couldn't be found\")\n      }\n    }\n  }\n\n  /** updateUserSessionsFromResult. */\n  updateUserSessionsFromResult(result) {\n    for (const scopeName in result.scopes) {\n      this.updateUserSessionScopeFromResult(scopeName, result.scopes[scopeName])\n    }\n  }\n\n  /** updateUserSessionScopeFromResult. */\n  updateUserSessionScopeFromResult(scopeName, scope) {\n    const deviseIsSignedInMethodName = `is${inflection.camelize(scopeName)}SignedIn`\n\n    if (!(deviseIsSignedInMethodName in Devise)) {\n      throw new Error(`No such method in Devise: ${deviseIsSignedInMethodName}`)\n    }\n\n    const currentlySignedIn = Devise[deviseIsSignedInMethodName]()\n    const signedInOnBackend = scope.signed_in\n\n    if (currentlySignedIn && !signedInOnBackend) {\n      logger.debug(() => `${inflection.camelize(scopeName)} signed in on frontend but not in backend!`)\n\n      Devise.setSignedOut({scope: scopeName})\n      Devise.callSignOutEvent({scope: scopeName})\n    }\n  }\n}\n"]}
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"session-status-updater.js","sourceRoot":"/src/","sources":["session-status-updater.js"],"names":[],"mappings":"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","sourcesContent":["import * as inflection from \"inflection\"\nimport Devise from \"./devise.js\"\nimport Logger from \"./logger.js\"\nimport config from \"./config.js\"\nimport wakeEvent from \"wake-event\"\n\nconst logger = new Logger({name: \"ApiMaker / SessionStatusUpdater\"})\nconst shared = {}\n\n// logger.setDebug(true)\n\n/** Tracks session and CSRF token freshness. */\nexport default class ApiMakerSessionStatusUpdater {\n  /** current. */\n  static current(args) {\n    if (!shared.apiMakerSessionStatusUpdater) {\n      shared.apiMakerSessionStatusUpdater = new ApiMakerSessionStatusUpdater(args)\n    }\n\n    return shared.apiMakerSessionStatusUpdater\n  }\n\n  /** Constructor. */\n  constructor(args = {}) {\n    this.events = {}\n    this.timeout = args.timeout || 600000\n\n    if (\"useMetaElement\" in args) {\n      this.useMetaElement = args.useMetaElement\n    } else if (typeof document === \"undefined\") {\n      this.useMetaElement = false\n    } else {\n      this.useMetaElement = true\n    }\n\n    if (typeof window != \"undefined\") {\n      this.connectOnlineEvent()\n    }\n\n    this.connectWakeEvent()\n  }\n\n  /** connectOnlineEvent. */\n  connectOnlineEvent() {\n    window.addEventListener(\"online\", this.updateSessionStatus, false)\n  }\n\n  /** connectWakeEvent. */\n  connectWakeEvent() {\n    wakeEvent(this.updateSessionStatus)\n  }\n\n  async getCsrfToken() {\n    if (this.csrfToken) {\n      logger.debug(`Get CSRF token from set variable: ${this.csrfToken}`)\n\n      return this.csrfToken\n    }\n\n    if (this.useMetaElement) {\n      const csrfTokenElement = document.querySelector(\"meta[name='csrf-token']\")\n\n      if (csrfTokenElement) {\n        logger.debug(() => `Get CSRF token from meta element: ${csrfTokenElement.getAttribute(\"content\")}`)\n\n        this.csrfToken = csrfTokenElement.getAttribute(\"content\")\n\n        return this.csrfToken\n      }\n    }\n\n    logger.debug(\"Updating session status because no CSRF token set yet\")\n    await this.updateSessionStatus()\n\n    if (this.csrfToken) {\n      logger.debug(() => `Returning CSRF token after updating session status: ${this.csrfToken}`)\n\n      return this.csrfToken\n    }\n\n    logger.debug(\"No CSRF token available after updating session status\")\n\n    return undefined\n  }\n\n  /** sessionStatus. */\n  sessionStatus() {\n    return new Promise((resolve) => {\n      const host = config.getHost()\n      let requestPath = \"\"\n\n      if (host) requestPath += host\n\n      requestPath += \"/api_maker/session_statuses\"\n\n      const xhr = new XMLHttpRequest()\n      xhr.open(\"POST\", requestPath, true)\n      xhr.onload = () => {\n        const response = JSON.parse(xhr.responseText)\n        resolve(response)\n      }\n      xhr.send()\n    })\n  }\n\n  /** onSignedOut. */\n  onSignedOut(callback) {\n    // @ts-expect-error\n    this.addEvent(\"onSignedOut\", callback)\n  }\n\n  /** startTimeout. */\n  startTimeout() {\n    logger.debug(\"startTimeout\")\n\n    if (this.updateTimeout)\n      clearTimeout(this.updateTimeout)\n\n    this.updateTimeout = setTimeout(\n      () => {\n        this.startTimeout()\n        this.updateSessionStatus()\n      },\n      this.timeout\n    )\n  }\n\n  /** stopTimeout. */\n  stopTimeout() {\n    if (this.updateTimeout)\n      clearTimeout(this.updateTimeout)\n  }\n\n  /** updateSessionStatus. */\n  updateSessionStatus = async () => {\n    logger.debug(\"updateSessionStatus\")\n\n    const result = await this.sessionStatus()\n\n    this.applyResult(result)\n  }\n\n  /**\n   * @param {Record<string, any>} result\n   * @returns {void}\n   */\n  applyResult(result) {\n    logger.debug(() => `Result: ${JSON.stringify(result, null, 2)}`)\n\n    this.updateMetaElementsFromResult(result)\n    this.updateUserSessionsFromResult(result)\n  }\n\n  /** updateMetaElementsFromResult. */\n  updateMetaElementsFromResult(result) {\n    logger.debug(\"updateMetaElementsFromResult\")\n\n    if (!result.csrf_token) {\n      return\n    }\n\n    this.csrfToken = result.csrf_token\n\n    if (this.useMetaElement) {\n      const csrfTokenElement = document.querySelector(\"meta[name='csrf-token']\")\n\n      if (csrfTokenElement) {\n        logger.debug(() => `Changing token from \"${csrfTokenElement.getAttribute(\"content\")}\" to \"${result.csrf_token}\"`)\n        csrfTokenElement.setAttribute(\"content\", result.csrf_token)\n      } else {\n        logger.debug(\"csrf token element couldn't be found\")\n      }\n    }\n  }\n\n  /** updateUserSessionsFromResult. */\n  updateUserSessionsFromResult(result) {\n    for (const scopeName in result.scopes) {\n      this.updateUserSessionScopeFromResult(scopeName, result.scopes[scopeName])\n    }\n  }\n\n  /** updateUserSessionScopeFromResult. */\n  updateUserSessionScopeFromResult(scopeName, scope) {\n    const deviseIsSignedInMethodName = `is${inflection.camelize(scopeName)}SignedIn`\n\n    if (!(deviseIsSignedInMethodName in Devise)) {\n      throw new Error(`No such method in Devise: ${deviseIsSignedInMethodName}`)\n    }\n\n    const currentlySignedIn = Devise[deviseIsSignedInMethodName]()\n    const signedInOnBackend = scope.signed_in\n\n    if (currentlySignedIn && !signedInOnBackend) {\n      logger.debug(() => `${inflection.camelize(scopeName)} signed in on frontend but not in backend!`)\n\n      Devise.setSignedOut({scope: scopeName})\n      Devise.callSignOutEvent({scope: scopeName})\n    }\n  }\n}\n"]}
@@ -0,0 +1,55 @@
1
+ /** Shared websocket request client for ApiMaker command/service execution. */
2
+ export default class ApiMakerWebsocketRequestClient {
3
+ /** @returns {ApiMakerWebsocketRequestClient} */
4
+ static current(): ApiMakerWebsocketRequestClient;
5
+ currentRequestId: number;
6
+ pendingRequests: {};
7
+ pendingRequestsByFingerprint: {};
8
+ responseCache: {};
9
+ subscriptionState: string;
10
+ /**
11
+ * @param {object} args
12
+ * @param {boolean} [args.cacheResponse]
13
+ * @param {Record<string, any>} [args.global]
14
+ * @param {(value: string) => void} [args.onLog]
15
+ * @param {(value: Record<string, any>) => void} [args.onProgress]
16
+ * @param {(value: Record<string, any>) => void} [args.onReceived]
17
+ * @param {Record<string, any>} args.request
18
+ * @returns {Promise<Record<string, any>>}
19
+ */
20
+ perform({ cacheResponse, global, onLog, onProgress, onReceived, request }: {
21
+ cacheResponse?: boolean;
22
+ global?: Record<string, any>;
23
+ onLog?: (value: string) => void;
24
+ onProgress?: (value: Record<string, any>) => void;
25
+ onReceived?: (value: Record<string, any>) => void;
26
+ request: Record<string, any>;
27
+ }): Promise<Record<string, any>>;
28
+ /** @returns {any} */
29
+ ensureSubscription(): any;
30
+ subscription: any;
31
+ /** @returns {Promise<void>} */
32
+ waitForSubscription(): Promise<void>;
33
+ /** @returns {void} */
34
+ resetSubscriptionReadyPromise(): void;
35
+ subscriptionReadyPromise: Promise<any>;
36
+ resolveSubscriptionReadyPromise: (value: any) => void;
37
+ rejectSubscriptionReadyPromise: (reason?: any) => void;
38
+ /** @returns {void} */
39
+ onConnected: () => void;
40
+ /** @returns {void} */
41
+ onDisconnected: () => void;
42
+ /** @returns {void} */
43
+ onRejected: () => void;
44
+ /**
45
+ * @param {Record<string, any>} data
46
+ * @returns {void}
47
+ */
48
+ onReceived: (data: Record<string, any>) => void;
49
+ /**
50
+ * @param {Error} error
51
+ * @returns {void}
52
+ */
53
+ rejectPendingRequests(error: Error): void;
54
+ }
55
+ //# sourceMappingURL=websocket-request-client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"websocket-request-client.d.ts","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"AAOA,8EAA8E;AAC9E;IACE,gDAAgD;IAChD,kBADc,8BAA8B,CAO3C;IAIC,yBAAyB;IACzB,oBAAyB;IACzB,iCAAsC;IACtC,kBAAuB;IACvB,0BAA8B;IAGhC;;;;;;;;;OASG;IACH,2EARG;QAAuB,aAAa,GAA5B,OAAO;QACoB,MAAM,GAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;QACY,KAAK,GAApC,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI;QACqB,UAAU,GAAtD,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;QACQ,UAAU,GAAtD,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,KAAK,IAAI;QACV,OAAO,EAAjC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;KAC3B,GAAU,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAyDxC;IAED,qBAAqB;IACrB,sBADc,GAAG,CAmBhB;IAZG,kBAQC;IAML,+BAA+B;IAC/B,uBADc,OAAO,CAAC,IAAI,CAAC,CAa1B;IAED,sBAAsB;IACtB,iCADc,IAAI,CAMjB;IAJC,uCAGE;IAFA,sDAA8C;IAC9C,uDAA4C;IAIhD,sBAAsB;IACtB,mBADc,IAAI,CAKjB;IAED,sBAAsB;IACtB,sBADc,IAAI,CAOjB;IAED,sBAAsB;IACtB,kBADc,IAAI,CAUjB;IAED;;;OAGG;IACH,aAAc,MAHH,MAAM,CAAC,MAAM,EAAE,GAAG,CAGX,KAFL,IAAI,CAqChB;IAED;;;OAGG;IACH,6BAHW,KAAK,GACH,IAAI,CAWhB;CACF"}
@@ -0,0 +1,192 @@
1
+ import CustomError from "./custom-error.js";
2
+ import Logger from "./logger.js";
3
+ import channelsConsumer from "./channels-consumer.js";
4
+ const logger = new Logger({ name: "ApiMaker / WebsocketRequestClient" });
5
+ const shared = {};
6
+ /** Shared websocket request client for ApiMaker command/service execution. */
7
+ export default class ApiMakerWebsocketRequestClient {
8
+ /** @returns {ApiMakerWebsocketRequestClient} */
9
+ static current() {
10
+ if (!shared.currentApiMakerWebsocketRequestClient) {
11
+ shared.currentApiMakerWebsocketRequestClient = new ApiMakerWebsocketRequestClient();
12
+ }
13
+ return shared.currentApiMakerWebsocketRequestClient;
14
+ }
15
+ /** Constructor. */
16
+ constructor() {
17
+ this.currentRequestId = 1;
18
+ this.pendingRequests = {};
19
+ this.pendingRequestsByFingerprint = {};
20
+ this.responseCache = {};
21
+ this.subscriptionState = "new";
22
+ }
23
+ /**
24
+ * @param {object} args
25
+ * @param {boolean} [args.cacheResponse]
26
+ * @param {Record<string, any>} [args.global]
27
+ * @param {(value: string) => void} [args.onLog]
28
+ * @param {(value: Record<string, any>) => void} [args.onProgress]
29
+ * @param {(value: Record<string, any>) => void} [args.onReceived]
30
+ * @param {Record<string, any>} args.request
31
+ * @returns {Promise<Record<string, any>>}
32
+ */
33
+ perform({ cacheResponse, global, onLog, onProgress, onReceived, request }) {
34
+ const fingerprint = JSON.stringify({ global, request });
35
+ if (cacheResponse && this.responseCache[fingerprint]) {
36
+ return Promise.resolve(this.responseCache[fingerprint]);
37
+ }
38
+ if (this.pendingRequestsByFingerprint[fingerprint]) {
39
+ const pendingRequestData = this.pendingRequestsByFingerprint[fingerprint];
40
+ const pendingRequest = this.pendingRequests[pendingRequestData.requestId];
41
+ if (onLog)
42
+ pendingRequest?.onLogCallbacks.push(onLog);
43
+ if (onProgress)
44
+ pendingRequest?.onProgressCallbacks.push(onProgress);
45
+ if (onReceived)
46
+ pendingRequest?.onReceivedCallbacks.push(onReceived);
47
+ return pendingRequestData.promise;
48
+ }
49
+ const promise = new Promise((resolve, reject) => {
50
+ const requestId = this.currentRequestId;
51
+ this.currentRequestId += 1;
52
+ this.pendingRequests[requestId] = {
53
+ cacheResponse,
54
+ fingerprint,
55
+ onLogCallbacks: onLog ? [onLog] : [],
56
+ onProgressCallbacks: onProgress ? [onProgress] : [],
57
+ onReceivedCallbacks: onReceived ? [onReceived] : [],
58
+ reject,
59
+ resolve
60
+ };
61
+ this.waitForSubscription().then(() => {
62
+ this.ensureSubscription().perform("execute", {
63
+ cache_response: cacheResponse,
64
+ global,
65
+ request,
66
+ request_id: requestId
67
+ });
68
+ })
69
+ .catch((error) => {
70
+ delete this.pendingRequests[requestId];
71
+ reject(error);
72
+ });
73
+ });
74
+ this.pendingRequestsByFingerprint[fingerprint] = { promise, requestId: this.currentRequestId - 1 };
75
+ promise.then(() => {
76
+ delete this.pendingRequestsByFingerprint[fingerprint];
77
+ }, () => {
78
+ delete this.pendingRequestsByFingerprint[fingerprint];
79
+ });
80
+ return promise;
81
+ }
82
+ /** @returns {any} */
83
+ ensureSubscription() {
84
+ if (!this.subscription) {
85
+ logger.debug("Creating websocket request subscription");
86
+ this.subscriptionState = "connecting";
87
+ this.resetSubscriptionReadyPromise();
88
+ this.subscription = channelsConsumer().subscriptions.create({ channel: "ApiMaker::RequestsChannel" }, {
89
+ connected: this.onConnected,
90
+ disconnected: this.onDisconnected,
91
+ received: this.onReceived,
92
+ rejected: this.onRejected
93
+ });
94
+ }
95
+ return this.subscription;
96
+ }
97
+ /** @returns {Promise<void>} */
98
+ waitForSubscription() {
99
+ this.ensureSubscription();
100
+ if (this.subscriptionState == "connected") {
101
+ return Promise.resolve();
102
+ }
103
+ if (!this.subscriptionReadyPromise) {
104
+ this.resetSubscriptionReadyPromise();
105
+ }
106
+ return this.subscriptionReadyPromise;
107
+ }
108
+ /** @returns {void} */
109
+ resetSubscriptionReadyPromise() {
110
+ this.subscriptionReadyPromise = new Promise((resolve, reject) => {
111
+ this.resolveSubscriptionReadyPromise = resolve;
112
+ this.rejectSubscriptionReadyPromise = reject;
113
+ });
114
+ }
115
+ /** @returns {void} */
116
+ onConnected = () => {
117
+ logger.debug("Websocket request subscription connected");
118
+ this.subscriptionState = "connected";
119
+ this.resolveSubscriptionReadyPromise?.();
120
+ };
121
+ /** @returns {void} */
122
+ onDisconnected = () => {
123
+ logger.debug("Websocket request subscription disconnected");
124
+ this.rejectPendingRequests(new Error("Websocket request subscription disconnected"));
125
+ this.subscriptionState = "disconnected";
126
+ this.subscription = null;
127
+ this.resetSubscriptionReadyPromise();
128
+ };
129
+ /** @returns {void} */
130
+ onRejected = () => {
131
+ const error = new Error("Websocket request subscription was rejected");
132
+ logger.error(error);
133
+ this.rejectPendingRequests(error);
134
+ this.subscriptionState = "rejected";
135
+ this.subscription = null;
136
+ this.rejectSubscriptionReadyPromise?.(error);
137
+ this.resetSubscriptionReadyPromise();
138
+ };
139
+ /**
140
+ * @param {Record<string, any>} data
141
+ * @returns {void}
142
+ */
143
+ onReceived = (data) => {
144
+ const pendingRequest = this.pendingRequests[data.request_id];
145
+ if (!pendingRequest) {
146
+ logger.debug(() => ["Ignoring websocket response without a pending request", { data }]);
147
+ return;
148
+ }
149
+ if (data.type == "api_maker_command_log") {
150
+ pendingRequest.onLogCallbacks.forEach((callback) => callback(data.message));
151
+ }
152
+ else if (data.type == "api_maker_command_progress") {
153
+ const progressData = {
154
+ count: data.count,
155
+ progress: data.progress,
156
+ total: data.total
157
+ };
158
+ pendingRequest.onProgressCallbacks.forEach((callback) => callback(progressData));
159
+ }
160
+ else if (data.type == "api_maker_request_received") {
161
+ pendingRequest.onReceivedCallbacks.forEach((callback) => callback(data));
162
+ }
163
+ else if (data.type == "api_maker_request_response") {
164
+ delete this.pendingRequests[data.request_id];
165
+ if (pendingRequest.cacheResponse) {
166
+ this.responseCache[pendingRequest.fingerprint] = data.response;
167
+ }
168
+ pendingRequest.resolve(data.response);
169
+ }
170
+ else if (data.type == "api_maker_request_error") {
171
+ delete this.pendingRequests[data.request_id];
172
+ pendingRequest.reject(new CustomError("Websocket request failed", { response: data.response }));
173
+ }
174
+ else {
175
+ delete this.pendingRequests[data.request_id];
176
+ pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`));
177
+ }
178
+ };
179
+ /**
180
+ * @param {Error} error
181
+ * @returns {void}
182
+ */
183
+ rejectPendingRequests(error) {
184
+ const pendingRequests = this.pendingRequests;
185
+ this.pendingRequests = {};
186
+ this.pendingRequestsByFingerprint = {};
187
+ Object.values(pendingRequests).forEach((pendingRequest) => {
188
+ queueMicrotask(() => pendingRequest.reject(error));
189
+ });
190
+ }
191
+ }
192
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-request-client.js","sourceRoot":"/src/","sources":["websocket-request-client.js"],"names":[],"mappings":"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","sourcesContent":["import CustomError from \"./custom-error.js\"\nimport Logger from \"./logger.js\"\nimport channelsConsumer from \"./channels-consumer.js\"\n\nconst logger = new Logger({name: \"ApiMaker / WebsocketRequestClient\"})\nconst shared = {}\n\n/** Shared websocket request client for ApiMaker command/service execution. */\nexport default class ApiMakerWebsocketRequestClient {\n  /** @returns {ApiMakerWebsocketRequestClient} */\n  static current () {\n    if (!shared.currentApiMakerWebsocketRequestClient) {\n      shared.currentApiMakerWebsocketRequestClient = new ApiMakerWebsocketRequestClient()\n    }\n\n    return shared.currentApiMakerWebsocketRequestClient\n  }\n\n  /** Constructor. */\n  constructor () {\n    this.currentRequestId = 1\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n    this.responseCache = {}\n    this.subscriptionState = \"new\"\n  }\n\n  /**\n   * @param {object} args\n   * @param {boolean} [args.cacheResponse]\n   * @param {Record<string, any>} [args.global]\n   * @param {(value: string) => void} [args.onLog]\n   * @param {(value: Record<string, any>) => void} [args.onProgress]\n   * @param {(value: Record<string, any>) => void} [args.onReceived]\n   * @param {Record<string, any>} args.request\n   * @returns {Promise<Record<string, any>>}\n   */\n  perform ({cacheResponse, global, onLog, onProgress, onReceived, request}) {\n    const fingerprint = JSON.stringify({global, request})\n\n    if (cacheResponse && this.responseCache[fingerprint]) {\n      return Promise.resolve(this.responseCache[fingerprint])\n    }\n\n    if (this.pendingRequestsByFingerprint[fingerprint]) {\n      const pendingRequestData = this.pendingRequestsByFingerprint[fingerprint]\n      const pendingRequest = this.pendingRequests[pendingRequestData.requestId]\n\n      if (onLog) pendingRequest?.onLogCallbacks.push(onLog)\n      if (onProgress) pendingRequest?.onProgressCallbacks.push(onProgress)\n      if (onReceived) pendingRequest?.onReceivedCallbacks.push(onReceived)\n\n      return pendingRequestData.promise\n    }\n\n    const promise = new Promise((resolve, reject) => {\n      const requestId = this.currentRequestId\n      this.currentRequestId += 1\n\n      this.pendingRequests[requestId] = {\n        cacheResponse,\n        fingerprint,\n        onLogCallbacks: onLog ? [onLog] : [],\n        onProgressCallbacks: onProgress ? [onProgress] : [],\n        onReceivedCallbacks: onReceived ? [onReceived] : [],\n        reject,\n        resolve\n      }\n\n      this.waitForSubscription().then(() => {\n        this.ensureSubscription().perform(\"execute\", {\n          cache_response: cacheResponse,\n          global,\n          request,\n          request_id: requestId\n        })\n      })\n        .catch((error) => {\n          delete this.pendingRequests[requestId]\n          reject(error)\n        })\n    })\n\n    this.pendingRequestsByFingerprint[fingerprint] = {promise, requestId: this.currentRequestId - 1}\n\n    promise.then(() => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    }, () => {\n      delete this.pendingRequestsByFingerprint[fingerprint]\n    })\n\n    return promise\n  }\n\n  /** @returns {any} */\n  ensureSubscription () {\n    if (!this.subscription) {\n      logger.debug(\"Creating websocket request subscription\")\n      this.subscriptionState = \"connecting\"\n      this.resetSubscriptionReadyPromise()\n\n      this.subscription = channelsConsumer().subscriptions.create(\n        {channel: \"ApiMaker::RequestsChannel\"},\n        {\n          connected: this.onConnected,\n          disconnected: this.onDisconnected,\n          received: this.onReceived,\n          rejected: this.onRejected\n        }\n      )\n    }\n\n    return this.subscription\n  }\n\n  /** @returns {Promise<void>} */\n  waitForSubscription () {\n    this.ensureSubscription()\n\n    if (this.subscriptionState == \"connected\") {\n      return Promise.resolve()\n    }\n\n    if (!this.subscriptionReadyPromise) {\n      this.resetSubscriptionReadyPromise()\n    }\n\n    return this.subscriptionReadyPromise\n  }\n\n  /** @returns {void} */\n  resetSubscriptionReadyPromise () {\n    this.subscriptionReadyPromise = new Promise((resolve, reject) => {\n      this.resolveSubscriptionReadyPromise = resolve\n      this.rejectSubscriptionReadyPromise = reject\n    })\n  }\n\n  /** @returns {void} */\n  onConnected = () => {\n    logger.debug(\"Websocket request subscription connected\")\n    this.subscriptionState = \"connected\"\n    this.resolveSubscriptionReadyPromise?.()\n  }\n\n  /** @returns {void} */\n  onDisconnected = () => {\n    logger.debug(\"Websocket request subscription disconnected\")\n    this.rejectPendingRequests(new Error(\"Websocket request subscription disconnected\"))\n    this.subscriptionState = \"disconnected\"\n    this.subscription = null\n    this.resetSubscriptionReadyPromise()\n  }\n\n  /** @returns {void} */\n  onRejected = () => {\n    const error = new Error(\"Websocket request subscription was rejected\")\n\n    logger.error(error)\n    this.rejectPendingRequests(error)\n    this.subscriptionState = \"rejected\"\n    this.subscription = null\n    this.rejectSubscriptionReadyPromise?.(error)\n    this.resetSubscriptionReadyPromise()\n  }\n\n  /**\n   * @param {Record<string, any>} data\n   * @returns {void}\n   */\n  onReceived = (data) => {\n    const pendingRequest = this.pendingRequests[data.request_id]\n\n    if (!pendingRequest) {\n      logger.debug(() => [\"Ignoring websocket response without a pending request\", {data}])\n      return\n    }\n\n    if (data.type == \"api_maker_command_log\") {\n      pendingRequest.onLogCallbacks.forEach((callback) => callback(data.message))\n    } else if (data.type == \"api_maker_command_progress\") {\n      const progressData = {\n        count: data.count,\n        progress: data.progress,\n        total: data.total\n      }\n\n      pendingRequest.onProgressCallbacks.forEach((callback) => callback(progressData))\n    } else if (data.type == \"api_maker_request_received\") {\n      pendingRequest.onReceivedCallbacks.forEach((callback) => callback(data))\n    } else if (data.type == \"api_maker_request_response\") {\n      delete this.pendingRequests[data.request_id]\n\n      if (pendingRequest.cacheResponse) {\n        this.responseCache[pendingRequest.fingerprint] = data.response\n      }\n\n      pendingRequest.resolve(data.response)\n    } else if (data.type == \"api_maker_request_error\") {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new CustomError(\"Websocket request failed\", {response: data.response}))\n    } else {\n      delete this.pendingRequests[data.request_id]\n      pendingRequest.reject(new Error(`Unknown websocket request response type: ${data.type}`))\n    }\n  }\n\n  /**\n   * @param {Error} error\n   * @returns {void}\n   */\n  rejectPendingRequests (error) {\n    const pendingRequests = this.pendingRequests\n\n    this.pendingRequests = {}\n    this.pendingRequestsByFingerprint = {}\n\n    Object.values(pendingRequests).forEach((pendingRequest) => {\n      queueMicrotask(() => pendingRequest.reject(error))\n    })\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@kaspernj/api-maker",
3
3
  "type": "module",
4
- "version": "1.0.2122",
4
+ "version": "1.0.2125",
5
5
  "description": "My new module",
6
6
  "files": [
7
7
  "build/**"