parse-server 9.6.0-alpha.8 → 9.6.0

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.
Files changed (49) hide show
  1. package/lib/AccountLockout.js +14 -42
  2. package/lib/Adapters/Auth/index.js +2 -2
  3. package/lib/Adapters/Auth/oauth2.js +2 -2
  4. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +3 -3
  5. package/lib/Adapters/Storage/Mongo/MongoTransform.js +30 -30
  6. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +53 -16
  7. package/lib/Auth.js +35 -9
  8. package/lib/Config.js +6 -5
  9. package/lib/Controllers/DatabaseController.js +134 -27
  10. package/lib/Controllers/FilesController.js +2 -2
  11. package/lib/Controllers/LiveQueryController.js +2 -2
  12. package/lib/Controllers/ParseGraphQLController.js +3 -2
  13. package/lib/Controllers/SchemaController.js +3 -7
  14. package/lib/Controllers/UserController.js +12 -3
  15. package/lib/Deprecator/Deprecations.js +41 -1
  16. package/lib/GraphQL/ParseGraphQLServer.js +1 -12
  17. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +6 -5
  18. package/lib/GraphQL/loaders/functionsMutations.js +3 -4
  19. package/lib/GraphQL/loaders/parseClassMutations.js +4 -5
  20. package/lib/GraphQL/loaders/parseClassQueries.js +3 -4
  21. package/lib/GraphQL/loaders/schemaMutations.js +4 -5
  22. package/lib/GraphQL/loaders/schemaQueries.js +2 -3
  23. package/lib/GraphQL/loaders/usersMutations.js +5 -5
  24. package/lib/GraphQL/parseGraphQLUtils.js +9 -1
  25. package/lib/GraphQL/transformers/query.js +2 -2
  26. package/lib/LiveQuery/ParseLiveQueryServer.js +192 -13
  27. package/lib/LiveQuery/QueryTools.js +28 -19
  28. package/lib/Options/Definitions.js +41 -17
  29. package/lib/Options/docs.js +15 -11
  30. package/lib/Options/index.js +1 -1
  31. package/lib/ParseServer.js +9 -4
  32. package/lib/Push/PushWorker.js +2 -4
  33. package/lib/Push/utils.js +3 -4
  34. package/lib/RestQuery.js +32 -4
  35. package/lib/RestWrite.js +50 -24
  36. package/lib/Routers/AggregateRouter.js +9 -2
  37. package/lib/Routers/ClassesRouter.js +6 -2
  38. package/lib/Routers/FilesRouter.js +4 -3
  39. package/lib/Routers/PagesRouter.js +44 -20
  40. package/lib/Routers/UsersRouter.js +64 -30
  41. package/lib/Security/Check.js +2 -2
  42. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +2 -2
  43. package/lib/StatusHandler.js +3 -2
  44. package/lib/Utils.js +74 -8
  45. package/lib/batch.js +39 -36
  46. package/lib/cloud-code/Parse.Cloud.js +2 -2
  47. package/lib/middlewares.js +36 -11
  48. package/lib/triggers.js +9 -2
  49. package/package.json +1 -3
@@ -28,38 +28,7 @@ class AccountLockout {
28
28
  }
29
29
 
30
30
  /**
31
- * check if the _failed_login_count field has been set
32
- */
33
- _isFailedLoginCountSet() {
34
- const query = {
35
- username: this._user.username,
36
- _failed_login_count: {
37
- $exists: true
38
- }
39
- };
40
- return this._config.database.find('_User', query).then(users => {
41
- if (Array.isArray(users) && users.length > 0) {
42
- return true;
43
- } else {
44
- return false;
45
- }
46
- });
47
- }
48
-
49
- /**
50
- * if _failed_login_count is NOT set then set it to 0
51
- * else do nothing
52
- */
53
- _initFailedLoginCount() {
54
- return this._isFailedLoginCountSet().then(failedLoginCountIsSet => {
55
- if (!failedLoginCountIsSet) {
56
- return this._setFailedLoginCount(0);
57
- }
58
- });
59
- }
60
-
61
- /**
62
- * increment _failed_login_count by 1
31
+ * increment _failed_login_count by 1 and return the updated document
63
32
  */
64
33
  _incrementFailedLoginCount() {
65
34
  const query = {
@@ -123,17 +92,20 @@ class AccountLockout {
123
92
  }
124
93
 
125
94
  /**
126
- * set and/or increment _failed_login_count
127
- * if _failed_login_count > threshold
128
- * set the _account_lockout_expires_at to current_time + accountPolicy.duration
129
- * else
130
- * do nothing
95
+ * Atomically increment _failed_login_count and enforce lockout threshold.
96
+ * Uses the atomic increment result to determine the exact post-increment
97
+ * count, eliminating the TOCTOU race between checking and updating.
131
98
  */
132
99
  _handleFailedLoginAttempt() {
133
- return this._initFailedLoginCount().then(() => {
134
- return this._incrementFailedLoginCount();
135
- }).then(() => {
136
- return this._setLockoutExpiration();
100
+ return this._incrementFailedLoginCount().then(result => {
101
+ const count = result._failed_login_count;
102
+ if (count >= this._config.accountLockout.threshold) {
103
+ return this._setLockoutExpiration().then(() => {
104
+ if (count > this._config.accountLockout.threshold) {
105
+ throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Your account is locked due to multiple failed login attempts. Please try again after ' + this._config.accountLockout.duration + ' minute(s)');
106
+ }
107
+ });
108
+ }
137
109
  });
138
110
  }
139
111
 
@@ -174,4 +146,4 @@ class AccountLockout {
174
146
  }
175
147
  exports.AccountLockout = AccountLockout;
176
148
  var _default = exports.default = AccountLockout;
177
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","_interopRequireDefault","require","e","__esModule","default","AccountLockout","constructor","user","config","_user","_config","_setFailedLoginCount","value","query","username","updateFields","_failed_login_count","database","update","_isFailedLoginCountSet","$exists","find","then","users","Array","isArray","length","_initFailedLoginCount","failedLoginCountIsSet","_incrementFailedLoginCount","__op","amount","_setLockoutExpiration","$gte","accountLockout","threshold","now","Date","_account_lockout_expires_at","Parse","_encode","getTime","duration","catch","err","code","message","Error","OBJECT_NOT_FOUND","_notLocked","$gt","_handleFailedLoginAttempt","handleLoginAttempt","loginSuccessful","Promise","resolve","unlockAccount","unlockOnPasswordReset","exports","_default"],"sources":["../src/AccountLockout.js"],"sourcesContent":["// This class handles the Account Lockout Policy settings.\nimport Parse from 'parse/node';\n\nexport class AccountLockout {\n  constructor(user, config) {\n    this._user = user;\n    this._config = config;\n  }\n\n  /**\n   * set _failed_login_count to value\n   */\n  _setFailedLoginCount(value) {\n    const query = {\n      username: this._user.username,\n    };\n\n    const updateFields = {\n      _failed_login_count: value,\n    };\n\n    return this._config.database.update('_User', query, updateFields);\n  }\n\n  /**\n   * check if the _failed_login_count field has been set\n   */\n  _isFailedLoginCountSet() {\n    const query = {\n      username: this._user.username,\n      _failed_login_count: { $exists: true },\n    };\n\n    return this._config.database.find('_User', query).then(users => {\n      if (Array.isArray(users) && users.length > 0) {\n        return true;\n      } else {\n        return false;\n      }\n    });\n  }\n\n  /**\n   * if _failed_login_count is NOT set then set it to 0\n   * else do nothing\n   */\n  _initFailedLoginCount() {\n    return this._isFailedLoginCountSet().then(failedLoginCountIsSet => {\n      if (!failedLoginCountIsSet) {\n        return this._setFailedLoginCount(0);\n      }\n    });\n  }\n\n  /**\n   * increment _failed_login_count by 1\n   */\n  _incrementFailedLoginCount() {\n    const query = {\n      username: this._user.username,\n    };\n\n    const updateFields = {\n      _failed_login_count: { __op: 'Increment', amount: 1 },\n    };\n\n    return this._config.database.update('_User', query, updateFields);\n  }\n\n  /**\n   * if the failed login count is greater than the threshold\n   * then sets lockout expiration to 'currenttime + accountPolicy.duration', i.e., account is locked out for the next 'accountPolicy.duration' minutes\n   * else do nothing\n   */\n  _setLockoutExpiration() {\n    const query = {\n      username: this._user.username,\n      _failed_login_count: { $gte: this._config.accountLockout.threshold },\n    };\n\n    const now = new Date();\n\n    const updateFields = {\n      _account_lockout_expires_at: Parse._encode(\n        new Date(now.getTime() + this._config.accountLockout.duration * 60 * 1000)\n      ),\n    };\n\n    return this._config.database.update('_User', query, updateFields).catch(err => {\n      if (\n        err &&\n        err.code &&\n        err.message &&\n        err.code === Parse.Error.OBJECT_NOT_FOUND &&\n        err.message === 'Object not found.'\n      ) {\n        return; // nothing to update so we are good\n      } else {\n        throw err; // unknown error\n      }\n    });\n  }\n\n  /**\n   * if _account_lockout_expires_at > current_time and _failed_login_count > threshold\n   *   reject with account locked error\n   * else\n   *   resolve\n   */\n  _notLocked() {\n    const query = {\n      username: this._user.username,\n      _account_lockout_expires_at: { $gt: Parse._encode(new Date()) },\n      _failed_login_count: { $gte: this._config.accountLockout.threshold },\n    };\n\n    return this._config.database.find('_User', query).then(users => {\n      if (Array.isArray(users) && users.length > 0) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Your account is locked due to multiple failed login attempts. Please try again after ' +\n            this._config.accountLockout.duration +\n            ' minute(s)'\n        );\n      }\n    });\n  }\n\n  /**\n   * set and/or increment _failed_login_count\n   * if _failed_login_count > threshold\n   *   set the _account_lockout_expires_at to current_time + accountPolicy.duration\n   * else\n   *   do nothing\n   */\n  _handleFailedLoginAttempt() {\n    return this._initFailedLoginCount()\n      .then(() => {\n        return this._incrementFailedLoginCount();\n      })\n      .then(() => {\n        return this._setLockoutExpiration();\n      });\n  }\n\n  /**\n   * handle login attempt if the Account Lockout Policy is enabled\n   */\n  handleLoginAttempt(loginSuccessful) {\n    if (!this._config.accountLockout) {\n      return Promise.resolve();\n    }\n    return this._notLocked().then(() => {\n      if (loginSuccessful) {\n        return this._setFailedLoginCount(0);\n      } else {\n        return this._handleFailedLoginAttempt();\n      }\n    });\n  }\n\n  /**\n   * Removes the account lockout.\n   */\n  unlockAccount() {\n    if (!this._config.accountLockout || !this._config.accountLockout.unlockOnPasswordReset) {\n      return Promise.resolve();\n    }\n    return this._config.database.update(\n      '_User',\n      { username: this._user.username },\n      {\n        _failed_login_count: { __op: 'Delete' },\n        _account_lockout_expires_at: { __op: 'Delete' },\n      }\n    );\n  }\n}\n\nexport default AccountLockout;\n"],"mappings":"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","ignoreList":[]}
149
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","_interopRequireDefault","require","e","__esModule","default","AccountLockout","constructor","user","config","_user","_config","_setFailedLoginCount","value","query","username","updateFields","_failed_login_count","database","update","_incrementFailedLoginCount","__op","amount","_setLockoutExpiration","$gte","accountLockout","threshold","now","Date","_account_lockout_expires_at","Parse","_encode","getTime","duration","catch","err","code","message","Error","OBJECT_NOT_FOUND","_notLocked","$gt","find","then","users","Array","isArray","length","_handleFailedLoginAttempt","result","count","handleLoginAttempt","loginSuccessful","Promise","resolve","unlockAccount","unlockOnPasswordReset","exports","_default"],"sources":["../src/AccountLockout.js"],"sourcesContent":["// This class handles the Account Lockout Policy settings.\nimport Parse from 'parse/node';\n\nexport class AccountLockout {\n  constructor(user, config) {\n    this._user = user;\n    this._config = config;\n  }\n\n  /**\n   * set _failed_login_count to value\n   */\n  _setFailedLoginCount(value) {\n    const query = {\n      username: this._user.username,\n    };\n\n    const updateFields = {\n      _failed_login_count: value,\n    };\n\n    return this._config.database.update('_User', query, updateFields);\n  }\n\n  /**\n   * increment _failed_login_count by 1 and return the updated document\n   */\n  _incrementFailedLoginCount() {\n    const query = {\n      username: this._user.username,\n    };\n\n    const updateFields = {\n      _failed_login_count: { __op: 'Increment', amount: 1 },\n    };\n\n    return this._config.database.update('_User', query, updateFields);\n  }\n\n  /**\n   * if the failed login count is greater than the threshold\n   * then sets lockout expiration to 'currenttime + accountPolicy.duration', i.e., account is locked out for the next 'accountPolicy.duration' minutes\n   * else do nothing\n   */\n  _setLockoutExpiration() {\n    const query = {\n      username: this._user.username,\n      _failed_login_count: { $gte: this._config.accountLockout.threshold },\n    };\n\n    const now = new Date();\n\n    const updateFields = {\n      _account_lockout_expires_at: Parse._encode(\n        new Date(now.getTime() + this._config.accountLockout.duration * 60 * 1000)\n      ),\n    };\n\n    return this._config.database.update('_User', query, updateFields).catch(err => {\n      if (\n        err &&\n        err.code &&\n        err.message &&\n        err.code === Parse.Error.OBJECT_NOT_FOUND &&\n        err.message === 'Object not found.'\n      ) {\n        return; // nothing to update so we are good\n      } else {\n        throw err; // unknown error\n      }\n    });\n  }\n\n  /**\n   * if _account_lockout_expires_at > current_time and _failed_login_count > threshold\n   *   reject with account locked error\n   * else\n   *   resolve\n   */\n  _notLocked() {\n    const query = {\n      username: this._user.username,\n      _account_lockout_expires_at: { $gt: Parse._encode(new Date()) },\n      _failed_login_count: { $gte: this._config.accountLockout.threshold },\n    };\n\n    return this._config.database.find('_User', query).then(users => {\n      if (Array.isArray(users) && users.length > 0) {\n        throw new Parse.Error(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Your account is locked due to multiple failed login attempts. Please try again after ' +\n            this._config.accountLockout.duration +\n            ' minute(s)'\n        );\n      }\n    });\n  }\n\n  /**\n   * Atomically increment _failed_login_count and enforce lockout threshold.\n   * Uses the atomic increment result to determine the exact post-increment\n   * count, eliminating the TOCTOU race between checking and updating.\n   */\n  _handleFailedLoginAttempt() {\n    return this._incrementFailedLoginCount().then(result => {\n      const count = result._failed_login_count;\n      if (count >= this._config.accountLockout.threshold) {\n        return this._setLockoutExpiration().then(() => {\n          if (count > this._config.accountLockout.threshold) {\n            throw new Parse.Error(\n              Parse.Error.OBJECT_NOT_FOUND,\n              'Your account is locked due to multiple failed login attempts. Please try again after ' +\n                this._config.accountLockout.duration +\n                ' minute(s)'\n            );\n          }\n        });\n      }\n    });\n  }\n\n  /**\n   * handle login attempt if the Account Lockout Policy is enabled\n   */\n  handleLoginAttempt(loginSuccessful) {\n    if (!this._config.accountLockout) {\n      return Promise.resolve();\n    }\n    return this._notLocked().then(() => {\n      if (loginSuccessful) {\n        return this._setFailedLoginCount(0);\n      } else {\n        return this._handleFailedLoginAttempt();\n      }\n    });\n  }\n\n  /**\n   * Removes the account lockout.\n   */\n  unlockAccount() {\n    if (!this._config.accountLockout || !this._config.accountLockout.unlockOnPasswordReset) {\n      return Promise.resolve();\n    }\n    return this._config.database.update(\n      '_User',\n      { username: this._user.username },\n      {\n        _failed_login_count: { __op: 'Delete' },\n        _account_lockout_expires_at: { __op: 'Delete' },\n      }\n    );\n  }\n}\n\nexport default AccountLockout;\n"],"mappings":";;;;;;AACA,IAAAA,KAAA,GAAAC,sBAAA,CAAAC,OAAA;AAA+B,SAAAD,uBAAAE,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAD/B;;AAGO,MAAMG,cAAc,CAAC;EAC1BC,WAAWA,CAACC,IAAI,EAAEC,MAAM,EAAE;IACxB,IAAI,CAACC,KAAK,GAAGF,IAAI;IACjB,IAAI,CAACG,OAAO,GAAGF,MAAM;EACvB;;EAEA;AACF;AACA;EACEG,oBAAoBA,CAACC,KAAK,EAAE;IAC1B,MAAMC,KAAK,GAAG;MACZC,QAAQ,EAAE,IAAI,CAACL,KAAK,CAACK;IACvB,CAAC;IAED,MAAMC,YAAY,GAAG;MACnBC,mBAAmB,EAAEJ;IACvB,CAAC;IAED,OAAO,IAAI,CAACF,OAAO,CAACO,QAAQ,CAACC,MAAM,CAAC,OAAO,EAAEL,KAAK,EAAEE,YAAY,CAAC;EACnE;;EAEA;AACF;AACA;EACEI,0BAA0BA,CAAA,EAAG;IAC3B,MAAMN,KAAK,GAAG;MACZC,QAAQ,EAAE,IAAI,CAACL,KAAK,CAACK;IACvB,CAAC;IAED,MAAMC,YAAY,GAAG;MACnBC,mBAAmB,EAAE;QAAEI,IAAI,EAAE,WAAW;QAAEC,MAAM,EAAE;MAAE;IACtD,CAAC;IAED,OAAO,IAAI,CAACX,OAAO,CAACO,QAAQ,CAACC,MAAM,CAAC,OAAO,EAAEL,KAAK,EAAEE,YAAY,CAAC;EACnE;;EAEA;AACF;AACA;AACA;AACA;EACEO,qBAAqBA,CAAA,EAAG;IACtB,MAAMT,KAAK,GAAG;MACZC,QAAQ,EAAE,IAAI,CAACL,KAAK,CAACK,QAAQ;MAC7BE,mBAAmB,EAAE;QAAEO,IAAI,EAAE,IAAI,CAACb,OAAO,CAACc,cAAc,CAACC;MAAU;IACrE,CAAC;IAED,MAAMC,GAAG,GAAG,IAAIC,IAAI,CAAC,CAAC;IAEtB,MAAMZ,YAAY,GAAG;MACnBa,2BAA2B,EAAEC,aAAK,CAACC,OAAO,CACxC,IAAIH,IAAI,CAACD,GAAG,CAACK,OAAO,CAAC,CAAC,GAAG,IAAI,CAACrB,OAAO,CAACc,cAAc,CAACQ,QAAQ,GAAG,EAAE,GAAG,IAAI,CAC3E;IACF,CAAC;IAED,OAAO,IAAI,CAACtB,OAAO,CAACO,QAAQ,CAACC,MAAM,CAAC,OAAO,EAAEL,KAAK,EAAEE,YAAY,CAAC,CAACkB,KAAK,CAACC,GAAG,IAAI;MAC7E,IACEA,GAAG,IACHA,GAAG,CAACC,IAAI,IACRD,GAAG,CAACE,OAAO,IACXF,GAAG,CAACC,IAAI,KAAKN,aAAK,CAACQ,KAAK,CAACC,gBAAgB,IACzCJ,GAAG,CAACE,OAAO,KAAK,mBAAmB,EACnC;QACA,OAAO,CAAC;MACV,CAAC,MAAM;QACL,MAAMF,GAAG,CAAC,CAAC;MACb;IACF,CAAC,CAAC;EACJ;;EAEA;AACF;AACA;AACA;AACA;AACA;EACEK,UAAUA,CAAA,EAAG;IACX,MAAM1B,KAAK,GAAG;MACZC,QAAQ,EAAE,IAAI,CAACL,KAAK,CAACK,QAAQ;MAC7Bc,2BAA2B,EAAE;QAAEY,GAAG,EAAEX,aAAK,CAACC,OAAO,CAAC,IAAIH,IAAI,CAAC,CAAC;MAAE,CAAC;MAC/DX,mBAAmB,EAAE;QAAEO,IAAI,EAAE,IAAI,CAACb,OAAO,CAACc,cAAc,CAACC;MAAU;IACrE,CAAC;IAED,OAAO,IAAI,CAACf,OAAO,CAACO,QAAQ,CAACwB,IAAI,CAAC,OAAO,EAAE5B,KAAK,CAAC,CAAC6B,IAAI,CAACC,KAAK,IAAI;MAC9D,IAAIC,KAAK,CAACC,OAAO,CAACF,KAAK,CAAC,IAAIA,KAAK,CAACG,MAAM,GAAG,CAAC,EAAE;QAC5C,MAAM,IAAIjB,aAAK,CAACQ,KAAK,CACnBR,aAAK,CAACQ,KAAK,CAACC,gBAAgB,EAC5B,uFAAuF,GACrF,IAAI,CAAC5B,OAAO,CAACc,cAAc,CAACQ,QAAQ,GACpC,YACJ,CAAC;MACH;IACF,CAAC,CAAC;EACJ;;EAEA;AACF;AACA;AACA;AACA;EACEe,yBAAyBA,CAAA,EAAG;IAC1B,OAAO,IAAI,CAAC5B,0BAA0B,CAAC,CAAC,CAACuB,IAAI,CAACM,MAAM,IAAI;MACtD,MAAMC,KAAK,GAAGD,MAAM,CAAChC,mBAAmB;MACxC,IAAIiC,KAAK,IAAI,IAAI,CAACvC,OAAO,CAACc,cAAc,CAACC,SAAS,EAAE;QAClD,OAAO,IAAI,CAACH,qBAAqB,CAAC,CAAC,CAACoB,IAAI,CAAC,MAAM;UAC7C,IAAIO,KAAK,GAAG,IAAI,CAACvC,OAAO,CAACc,cAAc,CAACC,SAAS,EAAE;YACjD,MAAM,IAAII,aAAK,CAACQ,KAAK,CACnBR,aAAK,CAACQ,KAAK,CAACC,gBAAgB,EAC5B,uFAAuF,GACrF,IAAI,CAAC5B,OAAO,CAACc,cAAc,CAACQ,QAAQ,GACpC,YACJ,CAAC;UACH;QACF,CAAC,CAAC;MACJ;IACF,CAAC,CAAC;EACJ;;EAEA;AACF;AACA;EACEkB,kBAAkBA,CAACC,eAAe,EAAE;IAClC,IAAI,CAAC,IAAI,CAACzC,OAAO,CAACc,cAAc,EAAE;MAChC,OAAO4B,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B;IACA,OAAO,IAAI,CAACd,UAAU,CAAC,CAAC,CAACG,IAAI,CAAC,MAAM;MAClC,IAAIS,eAAe,EAAE;QACnB,OAAO,IAAI,CAACxC,oBAAoB,CAAC,CAAC,CAAC;MACrC,CAAC,MAAM;QACL,OAAO,IAAI,CAACoC,yBAAyB,CAAC,CAAC;MACzC;IACF,CAAC,CAAC;EACJ;;EAEA;AACF;AACA;EACEO,aAAaA,CAAA,EAAG;IACd,IAAI,CAAC,IAAI,CAAC5C,OAAO,CAACc,cAAc,IAAI,CAAC,IAAI,CAACd,OAAO,CAACc,cAAc,CAAC+B,qBAAqB,EAAE;MACtF,OAAOH,OAAO,CAACC,OAAO,CAAC,CAAC;IAC1B;IACA,OAAO,IAAI,CAAC3C,OAAO,CAACO,QAAQ,CAACC,MAAM,CACjC,OAAO,EACP;MAAEJ,QAAQ,EAAE,IAAI,CAACL,KAAK,CAACK;IAAS,CAAC,EACjC;MACEE,mBAAmB,EAAE;QAAEI,IAAI,EAAE;MAAS,CAAC;MACvCQ,2BAA2B,EAAE;QAAER,IAAI,EAAE;MAAS;IAChD,CACF,CAAC;EACH;AACF;AAACoC,OAAA,CAAAnD,cAAA,GAAAA,cAAA;AAAA,IAAAoD,QAAA,GAAAD,OAAA,CAAApD,OAAA,GAEcC,cAAc","ignoreList":[]}
@@ -136,7 +136,7 @@ function loadAuthAdapter(provider, authOptions) {
136
136
  if (!defaultAdapter && !providerOptions) {
137
137
  return;
138
138
  }
139
- const adapter = defaultAdapter instanceof _AuthAdapter.default ? defaultAdapter : Object.assign({}, defaultAdapter);
139
+ const adapter = defaultAdapter instanceof _AuthAdapter.default ? new defaultAdapter.constructor() : Object.assign({}, defaultAdapter);
140
140
  const keys = ['validateAuthData', 'validateAppId', 'validateSetUp', 'validateLogin', 'validateUpdate', 'challenge', 'validateOptions', 'policy', 'afterFind'];
141
141
  const defaultAuthAdapter = new _AuthAdapter.default();
142
142
  keys.forEach(key => {
@@ -245,4 +245,4 @@ module.exports = function (authOptions = {}, enableAnonymousUsers = true) {
245
245
  });
246
246
  };
247
247
  module.exports.loadAuthAdapter = loadAuthAdapter;
248
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_AdapterLoader","_interopRequireDefault","require","_node","_AuthAdapter","_gcenter","_github","_gpgames","_instagram","_line","_linkedin","_mfa","_microsoft","_oauth","_qq","_spotify","_twitter","_wechat","_weibo","e","__esModule","default","apple","digits","facebook","google","janraincapture","janrainengage","keycloak","ldap","meetup","phantauth","vkontakte","anonymous","validateAuthData","Promise","resolve","validateAppId","providers","gcenter","gpgames","instagram","linkedin","mfa","github","twitter","spotify","line","qq","wechat","weibo","microsoft","authAdapterPolicies","solo","additional","authDataValidator","provider","adapter","appIds","options","authData","req","user","requestObject","policy","Parse","Error","OTHER_CAUSE","validateSetUp","validateLogin","validateUpdate","isLoggedIn","auth","id","isMaster","hasAuthDataConfigured","get","method","validator","loadAuthAdapter","authOptions","defaultAdapter","providerOptions","Object","prototype","hasOwnProperty","call","oauth2","AuthAdapter","assign","keys","defaultAuthAdapter","forEach","key","existing","toString","undefined","optionalAdapter","loadAdapter","validateOptions","module","exports","enableAnonymousUsers","_enableAnonymousUsers","setEnableAnonymousUsers","enable","getValidatorForProvider","authAdapter","runAfterFind","adapters","all","map","afterFind","ip","config","master","result","getProviders","allProviders","Set","delete","filter","freeze"],"sources":["../../../src/Adapters/Auth/index.js"],"sourcesContent":["import loadAdapter from '../AdapterLoader';\nimport Parse from 'parse/node';\nimport AuthAdapter from './AuthAdapter';\n\nconst apple = require('./apple');\nconst digits = require('./twitter'); // digits tokens are validated by twitter\nconst facebook = require('./facebook');\nimport gcenter from './gcenter';\nimport github from './github';\nconst google = require('./google');\nimport gpgames from './gpgames';\nimport instagram from './instagram';\nconst janraincapture = require('./janraincapture');\nconst janrainengage = require('./janrainengage');\nconst keycloak = require('./keycloak');\nconst ldap = require('./ldap');\nimport line from './line';\nimport linkedin from './linkedin';\nconst meetup = require('./meetup');\nimport mfa from './mfa';\nimport microsoft from './microsoft';\nimport oauth2 from './oauth2';\nconst phantauth = require('./phantauth');\nimport qq from './qq';\nimport spotify from './spotify';\nimport twitter from './twitter';\nconst vkontakte = require('./vkontakte');\nimport wechat from './wechat';\nimport weibo from './weibo';\n\n\nconst anonymous = {\n  validateAuthData: () => {\n    return Promise.resolve();\n  },\n  validateAppId: () => {\n    return Promise.resolve();\n  },\n};\n\nconst providers = {\n  apple,\n  gcenter,\n  gpgames,\n  facebook,\n  instagram,\n  linkedin,\n  meetup,\n  mfa,\n  google,\n  github,\n  twitter,\n  spotify,\n  anonymous,\n  digits,\n  janrainengage,\n  janraincapture,\n  line,\n  vkontakte,\n  qq,\n  wechat,\n  weibo,\n  phantauth,\n  microsoft,\n  keycloak,\n  ldap,\n};\n\n// Indexed auth policies\nconst authAdapterPolicies = {\n  default: true,\n  solo: true,\n  additional: true,\n};\n\nfunction authDataValidator(provider, adapter, appIds, options) {\n  return async function (authData, req, user, requestObject) {\n    if (appIds && typeof adapter.validateAppId === 'function') {\n      await Promise.resolve(adapter.validateAppId(appIds, authData, options, requestObject));\n    }\n    if (\n      adapter.policy &&\n      !authAdapterPolicies[adapter.policy] &&\n      typeof adapter.policy !== 'function'\n    ) {\n      throw new Parse.Error(\n        Parse.Error.OTHER_CAUSE,\n        'AuthAdapter policy is not configured correctly. The value must be either \"solo\", \"additional\", \"default\" or undefined (will be handled as \"default\")'\n      );\n    }\n    if (typeof adapter.validateAuthData === 'function') {\n      return adapter.validateAuthData(authData, options, requestObject);\n    }\n    if (\n      typeof adapter.validateSetUp !== 'function' ||\n      typeof adapter.validateLogin !== 'function' ||\n      typeof adapter.validateUpdate !== 'function'\n    ) {\n      throw new Parse.Error(\n        Parse.Error.OTHER_CAUSE,\n        'Adapter is not configured. Implement either validateAuthData or all of the following: validateSetUp, validateLogin and validateUpdate'\n      );\n    }\n    // When masterKey is detected, we should trigger a logged in user\n    const isLoggedIn =\n      (req.auth.user && user && req.auth.user.id === user.id) || (user && req.auth.isMaster);\n    let hasAuthDataConfigured = false;\n\n    if (user && user.get('authData') && user.get('authData')[provider]) {\n      hasAuthDataConfigured = true;\n    }\n\n    if (isLoggedIn) {\n      // User is updating their authData\n      if (hasAuthDataConfigured) {\n        return {\n          method: 'validateUpdate',\n          validator: () => adapter.validateUpdate(authData, options, requestObject),\n        };\n      }\n      // Set up if the user does not have the provider configured\n      return {\n        method: 'validateSetUp',\n        validator: () => adapter.validateSetUp(authData, options, requestObject),\n      };\n    }\n\n    // Not logged in and authData is configured on the user\n    if (hasAuthDataConfigured) {\n      return {\n        method: 'validateLogin',\n        validator: () => adapter.validateLogin(authData, options, requestObject),\n      };\n    }\n\n    // User not logged in and the provider is not set up, for example when a new user\n    // signs up or an existing user uses a new auth provider\n    return {\n      method: 'validateSetUp',\n      validator: () => adapter.validateSetUp(authData, options, requestObject),\n    };\n  };\n}\n\nfunction loadAuthAdapter(provider, authOptions) {\n  // providers are auth providers implemented by default\n  let defaultAdapter = providers[provider];\n  // authOptions can contain complete custom auth adapters or\n  // a default auth adapter like Facebook\n  const providerOptions = authOptions[provider];\n  if (\n    providerOptions &&\n    Object.prototype.hasOwnProperty.call(providerOptions, 'oauth2') &&\n    providerOptions['oauth2'] === true\n  ) {\n    defaultAdapter = oauth2;\n  }\n\n  // Default provider not found and a custom auth provider was not provided\n  if (!defaultAdapter && !providerOptions) {\n    return;\n  }\n\n  const adapter =\n    defaultAdapter instanceof AuthAdapter ? defaultAdapter : Object.assign({}, defaultAdapter);\n  const keys = [\n    'validateAuthData',\n    'validateAppId',\n    'validateSetUp',\n    'validateLogin',\n    'validateUpdate',\n    'challenge',\n    'validateOptions',\n    'policy',\n    'afterFind',\n  ];\n  const defaultAuthAdapter = new AuthAdapter();\n  keys.forEach(key => {\n    const existing = adapter?.[key];\n    if (\n      existing &&\n      typeof existing === 'function' &&\n      existing.toString() === defaultAuthAdapter[key].toString()\n    ) {\n      adapter[key] = null;\n    }\n  });\n  const appIds = providerOptions ? providerOptions.appIds : undefined;\n\n  // Try the configuration methods\n  if (providerOptions) {\n    const optionalAdapter = loadAdapter(providerOptions, undefined, providerOptions);\n    if (optionalAdapter) {\n      keys.forEach(key => {\n        if (optionalAdapter[key]) {\n          adapter[key] = optionalAdapter[key];\n        }\n      });\n    }\n  }\n  if (adapter.validateOptions) {\n    adapter.validateOptions(providerOptions);\n  }\n\n  return { adapter, appIds, providerOptions };\n}\n\nmodule.exports = function (authOptions = {}, enableAnonymousUsers = true) {\n  let _enableAnonymousUsers = enableAnonymousUsers;\n  const setEnableAnonymousUsers = function (enable) {\n    _enableAnonymousUsers = enable;\n  };\n  // To handle the test cases on configuration\n  const getValidatorForProvider = function (provider) {\n    if (provider === 'anonymous' && !_enableAnonymousUsers) {\n      return { validator: undefined };\n    }\n    const authAdapter = loadAuthAdapter(provider, authOptions);\n    if (!authAdapter) { return; }\n    const { adapter, appIds, providerOptions } = authAdapter;\n    return { validator: authDataValidator(provider, adapter, appIds, providerOptions), adapter };\n  };\n\n  const runAfterFind = async (req, authData) => {\n    if (!authData) {\n      return;\n    }\n    const adapters = Object.keys(authData);\n    await Promise.all(\n      adapters.map(async provider => {\n        const authAdapter = getValidatorForProvider(provider);\n        if (!authAdapter) {\n          return;\n        }\n        const { adapter, providerOptions } = authAdapter;\n        const afterFind = adapter.afterFind;\n        if (afterFind && typeof afterFind === 'function') {\n          const requestObject = {\n            ip: req.config.ip,\n            user: req.auth.user,\n            master: req.auth.isMaster,\n          };\n          const result = afterFind.call(\n            adapter,\n            authData[provider],\n            providerOptions,\n            requestObject,\n          );\n          if (result) {\n            authData[provider] = result;\n          }\n        }\n      })\n    );\n  };\n\n  // Returns the list of auth provider names that have a valid adapter configured.\n  // This includes both built-in providers and custom providers from authOptions.\n  const getProviders = function () {\n    const allProviders = new Set([...Object.keys(providers), ...Object.keys(authOptions)]);\n    if (!_enableAnonymousUsers) {\n      allProviders.delete('anonymous');\n    }\n    return [...allProviders].filter(provider => {\n      try {\n        return !!loadAuthAdapter(provider, authOptions);\n      } catch {\n        return false;\n      }\n    });\n  };\n\n  return Object.freeze({\n    getValidatorForProvider,\n    getProviders,\n    setEnableAnonymousUsers,\n    runAfterFind,\n  });\n};\n\nmodule.exports.loadAuthAdapter = loadAuthAdapter;\n"],"mappings":"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","ignoreList":[]}
248
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_AdapterLoader","_interopRequireDefault","require","_node","_AuthAdapter","_gcenter","_github","_gpgames","_instagram","_line","_linkedin","_mfa","_microsoft","_oauth","_qq","_spotify","_twitter","_wechat","_weibo","e","__esModule","default","apple","digits","facebook","google","janraincapture","janrainengage","keycloak","ldap","meetup","phantauth","vkontakte","anonymous","validateAuthData","Promise","resolve","validateAppId","providers","gcenter","gpgames","instagram","linkedin","mfa","github","twitter","spotify","line","qq","wechat","weibo","microsoft","authAdapterPolicies","solo","additional","authDataValidator","provider","adapter","appIds","options","authData","req","user","requestObject","policy","Parse","Error","OTHER_CAUSE","validateSetUp","validateLogin","validateUpdate","isLoggedIn","auth","id","isMaster","hasAuthDataConfigured","get","method","validator","loadAuthAdapter","authOptions","defaultAdapter","providerOptions","Object","prototype","hasOwnProperty","call","oauth2","AuthAdapter","constructor","assign","keys","defaultAuthAdapter","forEach","key","existing","toString","undefined","optionalAdapter","loadAdapter","validateOptions","module","exports","enableAnonymousUsers","_enableAnonymousUsers","setEnableAnonymousUsers","enable","getValidatorForProvider","authAdapter","runAfterFind","adapters","all","map","afterFind","ip","config","master","result","getProviders","allProviders","Set","delete","filter","freeze"],"sources":["../../../src/Adapters/Auth/index.js"],"sourcesContent":["import loadAdapter from '../AdapterLoader';\nimport Parse from 'parse/node';\nimport AuthAdapter from './AuthAdapter';\n\nconst apple = require('./apple');\nconst digits = require('./twitter'); // digits tokens are validated by twitter\nconst facebook = require('./facebook');\nimport gcenter from './gcenter';\nimport github from './github';\nconst google = require('./google');\nimport gpgames from './gpgames';\nimport instagram from './instagram';\nconst janraincapture = require('./janraincapture');\nconst janrainengage = require('./janrainengage');\nconst keycloak = require('./keycloak');\nconst ldap = require('./ldap');\nimport line from './line';\nimport linkedin from './linkedin';\nconst meetup = require('./meetup');\nimport mfa from './mfa';\nimport microsoft from './microsoft';\nimport oauth2 from './oauth2';\nconst phantauth = require('./phantauth');\nimport qq from './qq';\nimport spotify from './spotify';\nimport twitter from './twitter';\nconst vkontakte = require('./vkontakte');\nimport wechat from './wechat';\nimport weibo from './weibo';\n\n\nconst anonymous = {\n  validateAuthData: () => {\n    return Promise.resolve();\n  },\n  validateAppId: () => {\n    return Promise.resolve();\n  },\n};\n\nconst providers = {\n  apple,\n  gcenter,\n  gpgames,\n  facebook,\n  instagram,\n  linkedin,\n  meetup,\n  mfa,\n  google,\n  github,\n  twitter,\n  spotify,\n  anonymous,\n  digits,\n  janrainengage,\n  janraincapture,\n  line,\n  vkontakte,\n  qq,\n  wechat,\n  weibo,\n  phantauth,\n  microsoft,\n  keycloak,\n  ldap,\n};\n\n// Indexed auth policies\nconst authAdapterPolicies = {\n  default: true,\n  solo: true,\n  additional: true,\n};\n\nfunction authDataValidator(provider, adapter, appIds, options) {\n  return async function (authData, req, user, requestObject) {\n    if (appIds && typeof adapter.validateAppId === 'function') {\n      await Promise.resolve(adapter.validateAppId(appIds, authData, options, requestObject));\n    }\n    if (\n      adapter.policy &&\n      !authAdapterPolicies[adapter.policy] &&\n      typeof adapter.policy !== 'function'\n    ) {\n      throw new Parse.Error(\n        Parse.Error.OTHER_CAUSE,\n        'AuthAdapter policy is not configured correctly. The value must be either \"solo\", \"additional\", \"default\" or undefined (will be handled as \"default\")'\n      );\n    }\n    if (typeof adapter.validateAuthData === 'function') {\n      return adapter.validateAuthData(authData, options, requestObject);\n    }\n    if (\n      typeof adapter.validateSetUp !== 'function' ||\n      typeof adapter.validateLogin !== 'function' ||\n      typeof adapter.validateUpdate !== 'function'\n    ) {\n      throw new Parse.Error(\n        Parse.Error.OTHER_CAUSE,\n        'Adapter is not configured. Implement either validateAuthData or all of the following: validateSetUp, validateLogin and validateUpdate'\n      );\n    }\n    // When masterKey is detected, we should trigger a logged in user\n    const isLoggedIn =\n      (req.auth.user && user && req.auth.user.id === user.id) || (user && req.auth.isMaster);\n    let hasAuthDataConfigured = false;\n\n    if (user && user.get('authData') && user.get('authData')[provider]) {\n      hasAuthDataConfigured = true;\n    }\n\n    if (isLoggedIn) {\n      // User is updating their authData\n      if (hasAuthDataConfigured) {\n        return {\n          method: 'validateUpdate',\n          validator: () => adapter.validateUpdate(authData, options, requestObject),\n        };\n      }\n      // Set up if the user does not have the provider configured\n      return {\n        method: 'validateSetUp',\n        validator: () => adapter.validateSetUp(authData, options, requestObject),\n      };\n    }\n\n    // Not logged in and authData is configured on the user\n    if (hasAuthDataConfigured) {\n      return {\n        method: 'validateLogin',\n        validator: () => adapter.validateLogin(authData, options, requestObject),\n      };\n    }\n\n    // User not logged in and the provider is not set up, for example when a new user\n    // signs up or an existing user uses a new auth provider\n    return {\n      method: 'validateSetUp',\n      validator: () => adapter.validateSetUp(authData, options, requestObject),\n    };\n  };\n}\n\nfunction loadAuthAdapter(provider, authOptions) {\n  // providers are auth providers implemented by default\n  let defaultAdapter = providers[provider];\n  // authOptions can contain complete custom auth adapters or\n  // a default auth adapter like Facebook\n  const providerOptions = authOptions[provider];\n  if (\n    providerOptions &&\n    Object.prototype.hasOwnProperty.call(providerOptions, 'oauth2') &&\n    providerOptions['oauth2'] === true\n  ) {\n    defaultAdapter = oauth2;\n  }\n\n  // Default provider not found and a custom auth provider was not provided\n  if (!defaultAdapter && !providerOptions) {\n    return;\n  }\n\n  const adapter =\n    defaultAdapter instanceof AuthAdapter ? new defaultAdapter.constructor() : Object.assign({}, defaultAdapter);\n  const keys = [\n    'validateAuthData',\n    'validateAppId',\n    'validateSetUp',\n    'validateLogin',\n    'validateUpdate',\n    'challenge',\n    'validateOptions',\n    'policy',\n    'afterFind',\n  ];\n  const defaultAuthAdapter = new AuthAdapter();\n  keys.forEach(key => {\n    const existing = adapter?.[key];\n    if (\n      existing &&\n      typeof existing === 'function' &&\n      existing.toString() === defaultAuthAdapter[key].toString()\n    ) {\n      adapter[key] = null;\n    }\n  });\n  const appIds = providerOptions ? providerOptions.appIds : undefined;\n\n  // Try the configuration methods\n  if (providerOptions) {\n    const optionalAdapter = loadAdapter(providerOptions, undefined, providerOptions);\n    if (optionalAdapter) {\n      keys.forEach(key => {\n        if (optionalAdapter[key]) {\n          adapter[key] = optionalAdapter[key];\n        }\n      });\n    }\n  }\n  if (adapter.validateOptions) {\n    adapter.validateOptions(providerOptions);\n  }\n\n  return { adapter, appIds, providerOptions };\n}\n\nmodule.exports = function (authOptions = {}, enableAnonymousUsers = true) {\n  let _enableAnonymousUsers = enableAnonymousUsers;\n  const setEnableAnonymousUsers = function (enable) {\n    _enableAnonymousUsers = enable;\n  };\n  // To handle the test cases on configuration\n  const getValidatorForProvider = function (provider) {\n    if (provider === 'anonymous' && !_enableAnonymousUsers) {\n      return { validator: undefined };\n    }\n    const authAdapter = loadAuthAdapter(provider, authOptions);\n    if (!authAdapter) { return; }\n    const { adapter, appIds, providerOptions } = authAdapter;\n    return { validator: authDataValidator(provider, adapter, appIds, providerOptions), adapter };\n  };\n\n  const runAfterFind = async (req, authData) => {\n    if (!authData) {\n      return;\n    }\n    const adapters = Object.keys(authData);\n    await Promise.all(\n      adapters.map(async provider => {\n        const authAdapter = getValidatorForProvider(provider);\n        if (!authAdapter) {\n          return;\n        }\n        const { adapter, providerOptions } = authAdapter;\n        const afterFind = adapter.afterFind;\n        if (afterFind && typeof afterFind === 'function') {\n          const requestObject = {\n            ip: req.config.ip,\n            user: req.auth.user,\n            master: req.auth.isMaster,\n          };\n          const result = afterFind.call(\n            adapter,\n            authData[provider],\n            providerOptions,\n            requestObject,\n          );\n          if (result) {\n            authData[provider] = result;\n          }\n        }\n      })\n    );\n  };\n\n  // Returns the list of auth provider names that have a valid adapter configured.\n  // This includes both built-in providers and custom providers from authOptions.\n  const getProviders = function () {\n    const allProviders = new Set([...Object.keys(providers), ...Object.keys(authOptions)]);\n    if (!_enableAnonymousUsers) {\n      allProviders.delete('anonymous');\n    }\n    return [...allProviders].filter(provider => {\n      try {\n        return !!loadAuthAdapter(provider, authOptions);\n      } catch {\n        return false;\n      }\n    });\n  };\n\n  return Object.freeze({\n    getValidatorForProvider,\n    getProviders,\n    setEnableAnonymousUsers,\n    runAfterFind,\n  });\n};\n\nmodule.exports.loadAuthAdapter = loadAuthAdapter;\n"],"mappings":"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","ignoreList":[]}
@@ -74,7 +74,7 @@ class OAuth2Adapter extends _AuthAdapter.default {
74
74
  this.appIds = options.appIds;
75
75
  this.authorizationHeader = options.authorizationHeader;
76
76
  }
77
- async validateAppId(authData) {
77
+ async validateAppId(appIds, authData) {
78
78
  if (!this.appidField) {
79
79
  return;
80
80
  }
@@ -112,4 +112,4 @@ class OAuth2Adapter extends _AuthAdapter.default {
112
112
  }
113
113
  }
114
114
  var _default = exports.default = new OAuth2Adapter();
115
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_AuthAdapter","_interopRequireDefault","require","e","__esModule","default","OAuth2Adapter","AuthAdapter","validateOptions","options","tokenIntrospectionEndpointUrl","Parse","Error","OBJECT_NOT_FOUND","appidField","appIds","length","useridField","authorizationHeader","validateAppId","authData","response","requestTokenInfo","access_token","appIdFieldValue","isValidAppId","Array","isArray","some","appId","includes","validateAuthData","active","id","accessToken","fetch","method","headers","Authorization","body","URLSearchParams","token","ok","json","_default","exports"],"sources":["../../../src/Adapters/Auth/oauth2.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for OAuth2 Token Introspection.\n *\n * @class OAuth2Adapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.tokenIntrospectionEndpointUrl - The URL of the token introspection endpoint. Required.\n * @param {boolean} options.oauth2 - Indicates that the request should be handled by the OAuth2 adapter. Required.\n * @param {string} [options.useridField='sub'] - The field in the introspection response that contains the user ID. Defaults to `sub` per RFC 7662.\n * @param {string} [options.appidField] - The field in the introspection response that contains the app ID. Optional.\n * @param {string[]} [options.appIds] - List of allowed app IDs. Required if `appidField` is defined.\n * @param {string} [options.authorizationHeader] - The Authorization header value for the introspection request. Optional.\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for OAuth2 Token Introspection, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *     \"oauth2Provider\": {\n *       \"tokenIntrospectionEndpointUrl\": \"https://provider.com/introspect\",\n *       \"useridField\": \"sub\",\n *       \"appidField\": \"aud\",\n *       \"appIds\": [\"my-app-id\"],\n *       \"authorizationHeader\": \"Basic dXNlcm5hbWU6cGFzc3dvcmQ=\",\n *       \"oauth2\": true\n *     }\n *   }\n * }\n * ```\n *\n * The adapter requires the following `authData` fields:\n * - `id`: The user ID provided by the client.\n * - `access_token`: The access token provided by the client.\n *\n * ## Auth Payload\n * ### Example Auth Payload\n * ```json\n * {\n *   \"oauth2\": {\n *     \"id\": \"user-id\",\n *     \"access_token\": \"access-token\"\n *   }\n * }\n * ```\n *\n * ## Notes\n * - `tokenIntrospectionEndpointUrl` is mandatory and should point to a valid OAuth2 provider's introspection endpoint.\n * - If `appidField` is defined, `appIds` must also be specified to validate the app ID in the introspection response.\n * - `authorizationHeader` can be used to authenticate requests to the token introspection endpoint.\n *\n * @see {@link https://datatracker.ietf.org/doc/html/rfc7662 OAuth 2.0 Token Introspection Specification}\n */\n\n\nimport AuthAdapter from './AuthAdapter';\n\nclass OAuth2Adapter extends AuthAdapter {\n  validateOptions(options) {\n    super.validateOptions(options);\n\n    if (!options.tokenIntrospectionEndpointUrl) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 token introspection endpoint URL is missing.');\n    }\n    if (options.appidField && !options.appIds?.length) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 configuration is missing app IDs.');\n    }\n\n    this.tokenIntrospectionEndpointUrl = options.tokenIntrospectionEndpointUrl;\n    this.useridField = options.useridField || 'sub';\n    this.appidField = options.appidField;\n    this.appIds = options.appIds;\n    this.authorizationHeader = options.authorizationHeader;\n  }\n\n  async validateAppId(authData) {\n    if (!this.appidField) {\n      return;\n    }\n\n    const response = await this.requestTokenInfo(authData.access_token);\n\n    const appIdFieldValue = response[this.appidField];\n    const isValidAppId = Array.isArray(appIdFieldValue)\n      ? appIdFieldValue.some(appId => this.appIds.includes(appId))\n      : this.appIds.includes(appIdFieldValue);\n\n    if (!isValidAppId) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2: Invalid app ID.');\n    }\n  }\n\n  async validateAuthData(authData) {\n    const response = await this.requestTokenInfo(authData.access_token);\n\n    if (!response.active || (this.useridField && authData.id !== response[this.useridField])) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 access token is invalid for this user.');\n    }\n\n    return {};\n  }\n\n  async requestTokenInfo(accessToken) {\n    const response = await fetch(this.tokenIntrospectionEndpointUrl, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded',\n        ...(this.authorizationHeader && { Authorization: this.authorizationHeader })\n      },\n      body: new URLSearchParams({ token: accessToken })\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 token introspection request failed.');\n    }\n\n    return response.json();\n  }\n}\n\nexport default new OAuth2Adapter();\n\n"],"mappings":"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","ignoreList":[]}
115
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_AuthAdapter","_interopRequireDefault","require","e","__esModule","default","OAuth2Adapter","AuthAdapter","validateOptions","options","tokenIntrospectionEndpointUrl","Parse","Error","OBJECT_NOT_FOUND","appidField","appIds","length","useridField","authorizationHeader","validateAppId","authData","response","requestTokenInfo","access_token","appIdFieldValue","isValidAppId","Array","isArray","some","appId","includes","validateAuthData","active","id","accessToken","fetch","method","headers","Authorization","body","URLSearchParams","token","ok","json","_default","exports"],"sources":["../../../src/Adapters/Auth/oauth2.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for OAuth2 Token Introspection.\n *\n * @class OAuth2Adapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.tokenIntrospectionEndpointUrl - The URL of the token introspection endpoint. Required.\n * @param {boolean} options.oauth2 - Indicates that the request should be handled by the OAuth2 adapter. Required.\n * @param {string} [options.useridField='sub'] - The field in the introspection response that contains the user ID. Defaults to `sub` per RFC 7662.\n * @param {string} [options.appidField] - The field in the introspection response that contains the app ID. Optional.\n * @param {string[]} [options.appIds] - List of allowed app IDs. Required if `appidField` is defined.\n * @param {string} [options.authorizationHeader] - The Authorization header value for the introspection request. Optional.\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for OAuth2 Token Introspection, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *     \"oauth2Provider\": {\n *       \"tokenIntrospectionEndpointUrl\": \"https://provider.com/introspect\",\n *       \"useridField\": \"sub\",\n *       \"appidField\": \"aud\",\n *       \"appIds\": [\"my-app-id\"],\n *       \"authorizationHeader\": \"Basic dXNlcm5hbWU6cGFzc3dvcmQ=\",\n *       \"oauth2\": true\n *     }\n *   }\n * }\n * ```\n *\n * The adapter requires the following `authData` fields:\n * - `id`: The user ID provided by the client.\n * - `access_token`: The access token provided by the client.\n *\n * ## Auth Payload\n * ### Example Auth Payload\n * ```json\n * {\n *   \"oauth2\": {\n *     \"id\": \"user-id\",\n *     \"access_token\": \"access-token\"\n *   }\n * }\n * ```\n *\n * ## Notes\n * - `tokenIntrospectionEndpointUrl` is mandatory and should point to a valid OAuth2 provider's introspection endpoint.\n * - If `appidField` is defined, `appIds` must also be specified to validate the app ID in the introspection response.\n * - `authorizationHeader` can be used to authenticate requests to the token introspection endpoint.\n *\n * @see {@link https://datatracker.ietf.org/doc/html/rfc7662 OAuth 2.0 Token Introspection Specification}\n */\n\n\nimport AuthAdapter from './AuthAdapter';\n\nclass OAuth2Adapter extends AuthAdapter {\n  validateOptions(options) {\n    super.validateOptions(options);\n\n    if (!options.tokenIntrospectionEndpointUrl) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 token introspection endpoint URL is missing.');\n    }\n    if (options.appidField && !options.appIds?.length) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 configuration is missing app IDs.');\n    }\n\n    this.tokenIntrospectionEndpointUrl = options.tokenIntrospectionEndpointUrl;\n    this.useridField = options.useridField || 'sub';\n    this.appidField = options.appidField;\n    this.appIds = options.appIds;\n    this.authorizationHeader = options.authorizationHeader;\n  }\n\n  async validateAppId(appIds, authData) {\n    if (!this.appidField) {\n      return;\n    }\n\n    const response = await this.requestTokenInfo(authData.access_token);\n\n    const appIdFieldValue = response[this.appidField];\n    const isValidAppId = Array.isArray(appIdFieldValue)\n      ? appIdFieldValue.some(appId => this.appIds.includes(appId))\n      : this.appIds.includes(appIdFieldValue);\n\n    if (!isValidAppId) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2: Invalid app ID.');\n    }\n  }\n\n  async validateAuthData(authData) {\n    const response = await this.requestTokenInfo(authData.access_token);\n\n    if (!response.active || (this.useridField && authData.id !== response[this.useridField])) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 access token is invalid for this user.');\n    }\n\n    return {};\n  }\n\n  async requestTokenInfo(accessToken) {\n    const response = await fetch(this.tokenIntrospectionEndpointUrl, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded',\n        ...(this.authorizationHeader && { Authorization: this.authorizationHeader })\n      },\n      body: new URLSearchParams({ token: accessToken })\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'OAuth2 token introspection request failed.');\n    }\n\n    return response.json();\n  }\n}\n\nexport default new OAuth2Adapter();\n\n"],"mappings":"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","ignoreList":[]}