parse-server 8.0.1-alpha.2 → 8.0.2-alpha.1

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 (37) hide show
  1. package/lib/Adapters/Auth/AuthAdapter.js +16 -9
  2. package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
  3. package/lib/Adapters/Auth/apple.js +45 -1
  4. package/lib/Adapters/Auth/facebook.js +61 -1
  5. package/lib/Adapters/Auth/gcenter.js +201 -157
  6. package/lib/Adapters/Auth/github.js +119 -31
  7. package/lib/Adapters/Auth/google.js +45 -1
  8. package/lib/Adapters/Auth/gpgames.js +120 -27
  9. package/lib/Adapters/Auth/index.js +33 -33
  10. package/lib/Adapters/Auth/instagram.js +114 -24
  11. package/lib/Adapters/Auth/janraincapture.js +45 -1
  12. package/lib/Adapters/Auth/janrainengage.js +11 -2
  13. package/lib/Adapters/Auth/keycloak.js +68 -35
  14. package/lib/Adapters/Auth/ldap.js +75 -1
  15. package/lib/Adapters/Auth/line.js +119 -32
  16. package/lib/Adapters/Auth/linkedin.js +111 -35
  17. package/lib/Adapters/Auth/meetup.js +16 -8
  18. package/lib/Adapters/Auth/mfa.js +80 -2
  19. package/lib/Adapters/Auth/microsoft.js +105 -30
  20. package/lib/Adapters/Auth/oauth2.js +96 -109
  21. package/lib/Adapters/Auth/phantauth.js +16 -8
  22. package/lib/Adapters/Auth/qq.js +107 -36
  23. package/lib/Adapters/Auth/spotify.js +108 -39
  24. package/lib/Adapters/Auth/twitter.js +187 -40
  25. package/lib/Adapters/Auth/vkontakte.js +20 -13
  26. package/lib/Adapters/Auth/wechat.js +105 -25
  27. package/lib/Adapters/Auth/weibo.js +135 -37
  28. package/lib/Auth.js +26 -17
  29. package/lib/Config.js +14 -1
  30. package/lib/Deprecator/Deprecations.js +5 -2
  31. package/lib/Options/Definitions.js +7 -1
  32. package/lib/Options/docs.js +2 -1
  33. package/lib/Options/index.js +1 -1
  34. package/lib/RestWrite.js +4 -5
  35. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +10 -1
  36. package/lib/cli/parse-server.js +1 -1
  37. package/package.json +7 -7
@@ -1,38 +1,125 @@
1
1
  "use strict";
2
2
 
3
- // Helper functions for accessing the line API.
4
- var Parse = require('parse/node').Parse;
5
- const httpsRequest = require('./httpsRequest');
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _BaseCodeAuthAdapter = _interopRequireDefault(require("./BaseCodeAuthAdapter"));
8
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
9
+ /**
10
+ * Parse Server authentication adapter for Line.
11
+ *
12
+ * @class LineAdapter
13
+ * @param {Object} options - The adapter configuration options.
14
+ * @param {string} options.clientId - Your Line App Client ID. Required for secure authentication.
15
+ * @param {string} options.clientSecret - Your Line App Client Secret. Required for secure authentication.
16
+ * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).
17
+ *
18
+ * @description
19
+ * ## Parse Server Configuration
20
+ * To configure Parse Server for Line authentication, use the following structure:
21
+ * ### Secure Configuration
22
+ * ```json
23
+ * {
24
+ * "auth": {
25
+ * "line": {
26
+ * "clientId": "your-client-id",
27
+ * "clientSecret": "your-client-secret"
28
+ * }
29
+ * }
30
+ * }
31
+ * ```
32
+ * ### Insecure Configuration (Not Recommended)
33
+ * ```json
34
+ * {
35
+ * "auth": {
36
+ * "line": {
37
+ * "enableInsecureAuth": true
38
+ * }
39
+ * }
40
+ * }
41
+ * ```
42
+ *
43
+ * The adapter requires the following `authData` fields:
44
+ * - **Secure Authentication**: `code`, `redirect_uri`.
45
+ * - **Insecure Authentication (Not Recommended)**: `id`, `access_token`.
46
+ *
47
+ * ## Auth Payloads
48
+ * ### Secure Authentication Payload
49
+ * ```json
50
+ * {
51
+ * "line": {
52
+ * "code": "xxxxxxxxx",
53
+ * "redirect_uri": "https://example.com/callback"
54
+ * }
55
+ * }
56
+ * ```
57
+ *
58
+ * ### Insecure Authentication Payload (Not Recommended)
59
+ * ```json
60
+ * {
61
+ * "line": {
62
+ * "id": "1234567",
63
+ * "access_token": "xxxxxxxxx"
64
+ * }
65
+ * }
66
+ * ```
67
+ *
68
+ * ## Notes
69
+ * - `enableInsecureAuth` is **not recommended** and will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.
70
+ * - Secure authentication exchanges the `code` and `redirect_uri` provided by the client for an access token using Line's OAuth flow.
71
+ *
72
+ * @see {@link https://developers.line.biz/en/docs/line-login/integrate-line-login/ Line Login Documentation}
73
+ */
6
74
 
7
- // Returns a promise that fulfills if this user id is valid.
8
- function validateAuthData(authData) {
9
- return request('profile', authData.access_token).then(response => {
10
- if (response && response.userId && response.userId === authData.id) {
11
- return;
75
+ class LineAdapter extends _BaseCodeAuthAdapter.default {
76
+ constructor() {
77
+ super('Line');
78
+ }
79
+ async getAccessTokenFromCode(authData) {
80
+ if (!authData.code) {
81
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Line auth is invalid for this user.');
12
82
  }
13
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Line auth is invalid for this user.');
14
- });
15
- }
16
-
17
- // Returns a promise that fulfills iff this app id is valid.
18
- function validateAppId() {
19
- return Promise.resolve();
20
- }
21
-
22
- // A promisey wrapper for api requests
23
- function request(path, access_token) {
24
- var options = {
25
- host: 'api.line.me',
26
- path: '/v2/' + path,
27
- method: 'GET',
28
- headers: {
29
- Authorization: 'Bearer ' + access_token
83
+ const tokenUrl = 'https://api.line.me/oauth2/v2.1/token';
84
+ const response = await fetch(tokenUrl, {
85
+ method: 'POST',
86
+ headers: {
87
+ 'Content-Type': 'application/x-www-form-urlencoded'
88
+ },
89
+ body: new URLSearchParams({
90
+ client_id: this.clientId,
91
+ client_secret: this.clientSecret,
92
+ grant_type: 'authorization_code',
93
+ redirect_uri: authData.redirect_uri,
94
+ code: authData.code
95
+ })
96
+ });
97
+ if (!response.ok) {
98
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Failed to exchange code for token: ${response.statusText}`);
99
+ }
100
+ const data = await response.json();
101
+ if (data.error) {
102
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, data.error_description || data.error);
103
+ }
104
+ return data.access_token;
105
+ }
106
+ async getUserFromAccessToken(accessToken) {
107
+ const userApiUrl = 'https://api.line.me/v2/profile';
108
+ const response = await fetch(userApiUrl, {
109
+ method: 'GET',
110
+ headers: {
111
+ Authorization: `Bearer ${accessToken}`
112
+ }
113
+ });
114
+ if (!response.ok) {
115
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Failed to fetch Line user: ${response.statusText}`);
116
+ }
117
+ const userData = await response.json();
118
+ if (!userData?.userId) {
119
+ throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Invalid Line user data received.');
30
120
  }
31
- };
32
- return httpsRequest.get(options);
121
+ return userData;
122
+ }
33
123
  }
34
- module.exports = {
35
- validateAppId: validateAppId,
36
- validateAuthData: validateAuthData
37
- };
38
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJodHRwc1JlcXVlc3QiLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwidGhlbiIsInJlc3BvbnNlIiwidXNlcklkIiwiaWQiLCJFcnJvciIsIk9CSkVDVF9OT1RfRk9VTkQiLCJ2YWxpZGF0ZUFwcElkIiwiUHJvbWlzZSIsInJlc29sdmUiLCJwYXRoIiwib3B0aW9ucyIsImhvc3QiLCJtZXRob2QiLCJoZWFkZXJzIiwiQXV0aG9yaXphdGlvbiIsImdldCIsIm1vZHVsZSIsImV4cG9ydHMiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvQWRhcHRlcnMvQXV0aC9saW5lLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEhlbHBlciBmdW5jdGlvbnMgZm9yIGFjY2Vzc2luZyB0aGUgbGluZSBBUEkuXG52YXIgUGFyc2UgPSByZXF1aXJlKCdwYXJzZS9ub2RlJykuUGFyc2U7XG5jb25zdCBodHRwc1JlcXVlc3QgPSByZXF1aXJlKCcuL2h0dHBzUmVxdWVzdCcpO1xuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEpIHtcbiAgcmV0dXJuIHJlcXVlc3QoJ3Byb2ZpbGUnLCBhdXRoRGF0YS5hY2Nlc3NfdG9rZW4pLnRoZW4ocmVzcG9uc2UgPT4ge1xuICAgIGlmIChyZXNwb25zZSAmJiByZXNwb25zZS51c2VySWQgJiYgcmVzcG9uc2UudXNlcklkID09PSBhdXRoRGF0YS5pZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCwgJ0xpbmUgYXV0aCBpcyBpbnZhbGlkIGZvciB0aGlzIHVzZXIuJyk7XG4gIH0pO1xufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbn1cblxuLy8gQSBwcm9taXNleSB3cmFwcGVyIGZvciBhcGkgcmVxdWVzdHNcbmZ1bmN0aW9uIHJlcXVlc3QocGF0aCwgYWNjZXNzX3Rva2VuKSB7XG4gIHZhciBvcHRpb25zID0ge1xuICAgIGhvc3Q6ICdhcGkubGluZS5tZScsXG4gICAgcGF0aDogJy92Mi8nICsgcGF0aCxcbiAgICBtZXRob2Q6ICdHRVQnLFxuICAgIGhlYWRlcnM6IHtcbiAgICAgIEF1dGhvcml6YXRpb246ICdCZWFyZXIgJyArIGFjY2Vzc190b2tlbixcbiAgICB9LFxuICB9O1xuICByZXR1cm4gaHR0cHNSZXF1ZXN0LmdldChvcHRpb25zKTtcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gIHZhbGlkYXRlQXBwSWQ6IHZhbGlkYXRlQXBwSWQsXG4gIHZhbGlkYXRlQXV0aERhdGE6IHZhbGlkYXRlQXV0aERhdGEsXG59O1xuIl0sIm1hcHBpbmdzIjoiOztBQUFBO0FBQ0EsSUFBSUEsS0FBSyxHQUFHQyxPQUFPLENBQUMsWUFBWSxDQUFDLENBQUNELEtBQUs7QUFDdkMsTUFBTUUsWUFBWSxHQUFHRCxPQUFPLENBQUMsZ0JBQWdCLENBQUM7O0FBRTlDO0FBQ0EsU0FBU0UsZ0JBQWdCQSxDQUFDQyxRQUFRLEVBQUU7RUFDbEMsT0FBT0MsT0FBTyxDQUFDLFNBQVMsRUFBRUQsUUFBUSxDQUFDRSxZQUFZLENBQUMsQ0FBQ0MsSUFBSSxDQUFDQyxRQUFRLElBQUk7SUFDaEUsSUFBSUEsUUFBUSxJQUFJQSxRQUFRLENBQUNDLE1BQU0sSUFBSUQsUUFBUSxDQUFDQyxNQUFNLEtBQUtMLFFBQVEsQ0FBQ00sRUFBRSxFQUFFO01BQ2xFO0lBQ0Y7SUFDQSxNQUFNLElBQUlWLEtBQUssQ0FBQ1csS0FBSyxDQUFDWCxLQUFLLENBQUNXLEtBQUssQ0FBQ0MsZ0JBQWdCLEVBQUUscUNBQXFDLENBQUM7RUFDNUYsQ0FBQyxDQUFDO0FBQ0o7O0FBRUE7QUFDQSxTQUFTQyxhQUFhQSxDQUFBLEVBQUc7RUFDdkIsT0FBT0MsT0FBTyxDQUFDQyxPQUFPLENBQUMsQ0FBQztBQUMxQjs7QUFFQTtBQUNBLFNBQVNWLE9BQU9BLENBQUNXLElBQUksRUFBRVYsWUFBWSxFQUFFO0VBQ25DLElBQUlXLE9BQU8sR0FBRztJQUNaQyxJQUFJLEVBQUUsYUFBYTtJQUNuQkYsSUFBSSxFQUFFLE1BQU0sR0FBR0EsSUFBSTtJQUNuQkcsTUFBTSxFQUFFLEtBQUs7SUFDYkMsT0FBTyxFQUFFO01BQ1BDLGFBQWEsRUFBRSxTQUFTLEdBQUdmO0lBQzdCO0VBQ0YsQ0FBQztFQUNELE9BQU9KLFlBQVksQ0FBQ29CLEdBQUcsQ0FBQ0wsT0FBTyxDQUFDO0FBQ2xDO0FBRUFNLE1BQU0sQ0FBQ0MsT0FBTyxHQUFHO0VBQ2ZYLGFBQWEsRUFBRUEsYUFBYTtFQUM1QlYsZ0JBQWdCLEVBQUVBO0FBQ3BCLENBQUMiLCJpZ25vcmVMaXN0IjpbXX0=
124
+ var _default = exports.default = new LineAdapter();
125
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_BaseCodeAuthAdapter","_interopRequireDefault","require","e","__esModule","default","LineAdapter","BaseCodeAuthAdapter","constructor","getAccessTokenFromCode","authData","code","Parse","Error","OBJECT_NOT_FOUND","tokenUrl","response","fetch","method","headers","body","URLSearchParams","client_id","clientId","client_secret","clientSecret","grant_type","redirect_uri","ok","statusText","data","json","error","error_description","access_token","getUserFromAccessToken","accessToken","userApiUrl","Authorization","userData","userId","VALIDATION_ERROR","_default","exports"],"sources":["../../../src/Adapters/Auth/line.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Line.\n *\n * @class LineAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.clientId - Your Line App Client ID. Required for secure authentication.\n * @param {string} options.clientSecret - Your Line App Client Secret. Required for secure authentication.\n * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for Line authentication, use the following structure:\n * ### Secure Configuration\n * ```json\n * {\n *   \"auth\": {\n *     \"line\": {\n *       \"clientId\": \"your-client-id\",\n *       \"clientSecret\": \"your-client-secret\"\n *     }\n *   }\n * }\n * ```\n * ### Insecure Configuration (Not Recommended)\n * ```json\n * {\n *   \"auth\": {\n *     \"line\": {\n *       \"enableInsecureAuth\": true\n *     }\n *   }\n * }\n * ```\n *\n * The adapter requires the following `authData` fields:\n * - **Secure Authentication**: `code`, `redirect_uri`.\n * - **Insecure Authentication (Not Recommended)**: `id`, `access_token`.\n *\n * ## Auth Payloads\n * ### Secure Authentication Payload\n * ```json\n * {\n *   \"line\": {\n *     \"code\": \"xxxxxxxxx\",\n *     \"redirect_uri\": \"https://example.com/callback\"\n *   }\n * }\n * ```\n *\n * ### Insecure Authentication Payload (Not Recommended)\n * ```json\n * {\n *   \"line\": {\n *     \"id\": \"1234567\",\n *     \"access_token\": \"xxxxxxxxx\"\n *   }\n * }\n * ```\n *\n * ## Notes\n * - `enableInsecureAuth` is **not recommended** and will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.\n * - Secure authentication exchanges the `code` and `redirect_uri` provided by the client for an access token using Line's OAuth flow.\n *\n * @see {@link https://developers.line.biz/en/docs/line-login/integrate-line-login/ Line Login Documentation}\n */\n\nimport BaseCodeAuthAdapter from './BaseCodeAuthAdapter';\n\nclass LineAdapter extends BaseCodeAuthAdapter {\n  constructor() {\n    super('Line');\n  }\n\n  async getAccessTokenFromCode(authData) {\n    if (!authData.code) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        'Line auth is invalid for this user.'\n      );\n    }\n\n    const tokenUrl = 'https://api.line.me/oauth2/v2.1/token';\n    const response = await fetch(tokenUrl, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded',\n      },\n      body: new URLSearchParams({\n        client_id: this.clientId,\n        client_secret: this.clientSecret,\n        grant_type: 'authorization_code',\n        redirect_uri: authData.redirect_uri,\n        code: authData.code,\n      }),\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        `Failed to exchange code for token: ${response.statusText}`\n      );\n    }\n\n    const data = await response.json();\n    if (data.error) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        data.error_description || data.error\n      );\n    }\n\n    return data.access_token;\n  }\n\n  async getUserFromAccessToken(accessToken) {\n    const userApiUrl = 'https://api.line.me/v2/profile';\n    const response = await fetch(userApiUrl, {\n      method: 'GET',\n      headers: {\n        Authorization: `Bearer ${accessToken}`,\n      },\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        `Failed to fetch Line user: ${response.statusText}`\n      );\n    }\n\n    const userData = await response.json();\n    if (!userData?.userId) {\n      throw new Parse.Error(\n        Parse.Error.VALIDATION_ERROR,\n        'Invalid Line user data received.'\n      );\n    }\n\n    return userData;\n  }\n}\n\nexport default new LineAdapter();\n"],"mappings":"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","ignoreList":[]}
@@ -1,41 +1,117 @@
1
1
  "use strict";
2
2
 
3
- // Helper functions for accessing the linkedin API.
4
- var Parse = require('parse/node').Parse;
5
- const httpsRequest = require('./httpsRequest');
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _BaseCodeAuthAdapter = _interopRequireDefault(require("./BaseCodeAuthAdapter"));
8
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
9
+ /**
10
+ * Parse Server authentication adapter for LinkedIn.
11
+ *
12
+ * @class LinkedInAdapter
13
+ * @param {Object} options - The adapter configuration options.
14
+ * @param {string} options.clientId - Your LinkedIn App Client ID. Required for secure authentication.
15
+ * @param {string} options.clientSecret - Your LinkedIn App Client Secret. Required for secure authentication.
16
+ * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).
17
+ *
18
+ * @description
19
+ * ## Parse Server Configuration
20
+ * To configure Parse Server for LinkedIn authentication, use the following structure:
21
+ * ### Secure Configuration
22
+ * ```json
23
+ * {
24
+ * "auth": {
25
+ * "linkedin": {
26
+ * "clientId": "your-client-id",
27
+ * "clientSecret": "your-client-secret"
28
+ * }
29
+ * }
30
+ * }
31
+ * ```
32
+ * ### Insecure Configuration (Not Recommended)
33
+ * ```json
34
+ * {
35
+ * "auth": {
36
+ * "linkedin": {
37
+ * "enableInsecureAuth": true
38
+ * }
39
+ * }
40
+ * }
41
+ * ```
42
+ *
43
+ * The adapter requires the following `authData` fields:
44
+ * - **Secure Authentication**: `code`, `redirect_uri`, and optionally `is_mobile_sdk`.
45
+ * - **Insecure Authentication (Not Recommended)**: `id`, `access_token`, and optionally `is_mobile_sdk`.
46
+ *
47
+ * ## Auth Payloads
48
+ * ### Secure Authentication Payload
49
+ * ```json
50
+ * {
51
+ * "linkedin": {
52
+ * "code": "lmn789opq012rst345uvw",
53
+ * "redirect_uri": "https://your-redirect-uri.com/callback",
54
+ * "is_mobile_sdk": true
55
+ * }
56
+ * }
57
+ * ```
58
+ *
59
+ * ### Insecure Authentication Payload (Not Recommended)
60
+ * ```json
61
+ * {
62
+ * "linkedin": {
63
+ * "id": "7654321",
64
+ * "access_token": "AQXNnd2hIT6z9bHFzZz2Kp1ghiMz_RtyuvwXYZ123abc",
65
+ * "is_mobile_sdk": true
66
+ * }
67
+ * }
68
+ * ```
69
+ *
70
+ * ## Notes
71
+ * - Secure authentication exchanges the `code` and `redirect_uri` provided by the client for an access token using LinkedIn's OAuth API.
72
+ * - Insecure authentication validates the user ID and access token directly, bypassing OAuth flows. This method is **not recommended** and may introduce security vulnerabilities.
73
+ * - `enableInsecureAuth` is **deprecated** and may be removed in future versions.
74
+ *
75
+ * @see {@link https://learn.microsoft.com/en-us/linkedin/shared/authentication/authentication LinkedIn Authentication Documentation}
76
+ */
6
77
 
7
- // Returns a promise that fulfills iff this user id is valid.
8
- function validateAuthData(authData) {
9
- return request('me', authData.access_token, authData.is_mobile_sdk).then(data => {
10
- if (data && data.id == authData.id) {
11
- return;
78
+ class LinkedInAdapter extends _BaseCodeAuthAdapter.default {
79
+ constructor() {
80
+ super('LinkedIn');
81
+ }
82
+ async getUserFromAccessToken(access_token, authData) {
83
+ const response = await fetch('https://api.linkedin.com/v2/me', {
84
+ headers: {
85
+ Authorization: `Bearer ${access_token}`,
86
+ 'x-li-format': 'json',
87
+ 'x-li-src': authData?.is_mobile_sdk ? 'msdk' : undefined
88
+ }
89
+ });
90
+ if (!response.ok) {
91
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LinkedIn API request failed.');
12
92
  }
13
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Linkedin auth is invalid for this user.');
14
- });
15
- }
16
-
17
- // Returns a promise that fulfills iff this app id is valid.
18
- function validateAppId() {
19
- return Promise.resolve();
20
- }
21
-
22
- // A promisey wrapper for api requests
23
- function request(path, access_token, is_mobile_sdk) {
24
- var headers = {
25
- Authorization: 'Bearer ' + access_token,
26
- 'x-li-format': 'json'
27
- };
28
- if (is_mobile_sdk) {
29
- headers['x-li-src'] = 'msdk';
93
+ return response.json();
94
+ }
95
+ async getAccessTokenFromCode(authData) {
96
+ const response = await fetch('https://www.linkedin.com/oauth/v2/accessToken', {
97
+ method: 'POST',
98
+ headers: {
99
+ 'Content-Type': 'application/x-www-form-urlencoded'
100
+ },
101
+ body: new URLSearchParams({
102
+ grant_type: 'authorization_code',
103
+ code: authData.code,
104
+ redirect_uri: authData.redirect_uri,
105
+ client_id: this.clientId,
106
+ client_secret: this.clientSecret
107
+ })
108
+ });
109
+ if (!response.ok) {
110
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LinkedIn API request failed.');
111
+ }
112
+ const json = await response.json();
113
+ return json.access_token;
30
114
  }
31
- return httpsRequest.get({
32
- host: 'api.linkedin.com',
33
- path: '/v2/' + path,
34
- headers: headers
35
- });
36
115
  }
37
- module.exports = {
38
- validateAppId: validateAppId,
39
- validateAuthData: validateAuthData
40
- };
41
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJodHRwc1JlcXVlc3QiLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwiaXNfbW9iaWxlX3NkayIsInRoZW4iLCJkYXRhIiwiaWQiLCJFcnJvciIsIk9CSkVDVF9OT1RfRk9VTkQiLCJ2YWxpZGF0ZUFwcElkIiwiUHJvbWlzZSIsInJlc29sdmUiLCJwYXRoIiwiaGVhZGVycyIsIkF1dGhvcml6YXRpb24iLCJnZXQiLCJob3N0IiwibW9kdWxlIiwiZXhwb3J0cyJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BZGFwdGVycy9BdXRoL2xpbmtlZGluLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEhlbHBlciBmdW5jdGlvbnMgZm9yIGFjY2Vzc2luZyB0aGUgbGlua2VkaW4gQVBJLlxudmFyIFBhcnNlID0gcmVxdWlyZSgncGFyc2Uvbm9kZScpLlBhcnNlO1xuY29uc3QgaHR0cHNSZXF1ZXN0ID0gcmVxdWlyZSgnLi9odHRwc1JlcXVlc3QnKTtcblxuLy8gUmV0dXJucyBhIHByb21pc2UgdGhhdCBmdWxmaWxscyBpZmYgdGhpcyB1c2VyIGlkIGlzIHZhbGlkLlxuZnVuY3Rpb24gdmFsaWRhdGVBdXRoRGF0YShhdXRoRGF0YSkge1xuICByZXR1cm4gcmVxdWVzdCgnbWUnLCBhdXRoRGF0YS5hY2Nlc3NfdG9rZW4sIGF1dGhEYXRhLmlzX21vYmlsZV9zZGspLnRoZW4oZGF0YSA9PiB7XG4gICAgaWYgKGRhdGEgJiYgZGF0YS5pZCA9PSBhdXRoRGF0YS5pZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCwgJ0xpbmtlZGluIGF1dGggaXMgaW52YWxpZCBmb3IgdGhpcyB1c2VyLicpO1xuICB9KTtcbn1cblxuLy8gUmV0dXJucyBhIHByb21pc2UgdGhhdCBmdWxmaWxscyBpZmYgdGhpcyBhcHAgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUFwcElkKCkge1xuICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCk7XG59XG5cbi8vIEEgcHJvbWlzZXkgd3JhcHBlciBmb3IgYXBpIHJlcXVlc3RzXG5mdW5jdGlvbiByZXF1ZXN0KHBhdGgsIGFjY2Vzc190b2tlbiwgaXNfbW9iaWxlX3Nkaykge1xuICB2YXIgaGVhZGVycyA9IHtcbiAgICBBdXRob3JpemF0aW9uOiAnQmVhcmVyICcgKyBhY2Nlc3NfdG9rZW4sXG4gICAgJ3gtbGktZm9ybWF0JzogJ2pzb24nLFxuICB9O1xuXG4gIGlmIChpc19tb2JpbGVfc2RrKSB7XG4gICAgaGVhZGVyc1sneC1saS1zcmMnXSA9ICdtc2RrJztcbiAgfVxuICByZXR1cm4gaHR0cHNSZXF1ZXN0LmdldCh7XG4gICAgaG9zdDogJ2FwaS5saW5rZWRpbi5jb20nLFxuICAgIHBhdGg6ICcvdjIvJyArIHBhdGgsXG4gICAgaGVhZGVyczogaGVhZGVycyxcbiAgfSk7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICB2YWxpZGF0ZUFwcElkOiB2YWxpZGF0ZUFwcElkLFxuICB2YWxpZGF0ZUF1dGhEYXRhOiB2YWxpZGF0ZUF1dGhEYXRhLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7QUFBQTtBQUNBLElBQUlBLEtBQUssR0FBR0MsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDRCxLQUFLO0FBQ3ZDLE1BQU1FLFlBQVksR0FBR0QsT0FBTyxDQUFDLGdCQUFnQixDQUFDOztBQUU5QztBQUNBLFNBQVNFLGdCQUFnQkEsQ0FBQ0MsUUFBUSxFQUFFO0VBQ2xDLE9BQU9DLE9BQU8sQ0FBQyxJQUFJLEVBQUVELFFBQVEsQ0FBQ0UsWUFBWSxFQUFFRixRQUFRLENBQUNHLGFBQWEsQ0FBQyxDQUFDQyxJQUFJLENBQUNDLElBQUksSUFBSTtJQUMvRSxJQUFJQSxJQUFJLElBQUlBLElBQUksQ0FBQ0MsRUFBRSxJQUFJTixRQUFRLENBQUNNLEVBQUUsRUFBRTtNQUNsQztJQUNGO0lBQ0EsTUFBTSxJQUFJVixLQUFLLENBQUNXLEtBQUssQ0FBQ1gsS0FBSyxDQUFDVyxLQUFLLENBQUNDLGdCQUFnQixFQUFFLHlDQUF5QyxDQUFDO0VBQ2hHLENBQUMsQ0FBQztBQUNKOztBQUVBO0FBQ0EsU0FBU0MsYUFBYUEsQ0FBQSxFQUFHO0VBQ3ZCLE9BQU9DLE9BQU8sQ0FBQ0MsT0FBTyxDQUFDLENBQUM7QUFDMUI7O0FBRUE7QUFDQSxTQUFTVixPQUFPQSxDQUFDVyxJQUFJLEVBQUVWLFlBQVksRUFBRUMsYUFBYSxFQUFFO0VBQ2xELElBQUlVLE9BQU8sR0FBRztJQUNaQyxhQUFhLEVBQUUsU0FBUyxHQUFHWixZQUFZO0lBQ3ZDLGFBQWEsRUFBRTtFQUNqQixDQUFDO0VBRUQsSUFBSUMsYUFBYSxFQUFFO0lBQ2pCVSxPQUFPLENBQUMsVUFBVSxDQUFDLEdBQUcsTUFBTTtFQUM5QjtFQUNBLE9BQU9mLFlBQVksQ0FBQ2lCLEdBQUcsQ0FBQztJQUN0QkMsSUFBSSxFQUFFLGtCQUFrQjtJQUN4QkosSUFBSSxFQUFFLE1BQU0sR0FBR0EsSUFBSTtJQUNuQkMsT0FBTyxFQUFFQTtFQUNYLENBQUMsQ0FBQztBQUNKO0FBRUFJLE1BQU0sQ0FBQ0MsT0FBTyxHQUFHO0VBQ2ZULGFBQWEsRUFBRUEsYUFBYTtFQUM1QlYsZ0JBQWdCLEVBQUVBO0FBQ3BCLENBQUMiLCJpZ25vcmVMaXN0IjpbXX0=
116
+ var _default = exports.default = new LinkedInAdapter();
117
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_BaseCodeAuthAdapter","_interopRequireDefault","require","e","__esModule","default","LinkedInAdapter","BaseAuthCodeAdapter","constructor","getUserFromAccessToken","access_token","authData","response","fetch","headers","Authorization","is_mobile_sdk","undefined","ok","Parse","Error","OBJECT_NOT_FOUND","json","getAccessTokenFromCode","method","body","URLSearchParams","grant_type","code","redirect_uri","client_id","clientId","client_secret","clientSecret","_default","exports"],"sources":["../../../src/Adapters/Auth/linkedin.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for LinkedIn.\n *\n * @class LinkedInAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.clientId - Your LinkedIn App Client ID. Required for secure authentication.\n * @param {string} options.clientSecret - Your LinkedIn App Client Secret. Required for secure authentication.\n * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for LinkedIn authentication, use the following structure:\n * ### Secure Configuration\n * ```json\n * {\n *   \"auth\": {\n *     \"linkedin\": {\n *       \"clientId\": \"your-client-id\",\n *       \"clientSecret\": \"your-client-secret\"\n *     }\n *   }\n * }\n * ```\n * ### Insecure Configuration (Not Recommended)\n * ```json\n * {\n *   \"auth\": {\n *     \"linkedin\": {\n *       \"enableInsecureAuth\": true\n *     }\n *   }\n * }\n * ```\n *\n * The adapter requires the following `authData` fields:\n * - **Secure Authentication**: `code`, `redirect_uri`, and optionally `is_mobile_sdk`.\n * - **Insecure Authentication (Not Recommended)**: `id`, `access_token`, and optionally `is_mobile_sdk`.\n *\n * ## Auth Payloads\n * ### Secure Authentication Payload\n * ```json\n * {\n *   \"linkedin\": {\n *     \"code\": \"lmn789opq012rst345uvw\",\n *     \"redirect_uri\": \"https://your-redirect-uri.com/callback\",\n *     \"is_mobile_sdk\": true\n *   }\n * }\n * ```\n *\n * ### Insecure Authentication Payload (Not Recommended)\n * ```json\n * {\n *   \"linkedin\": {\n *     \"id\": \"7654321\",\n *     \"access_token\": \"AQXNnd2hIT6z9bHFzZz2Kp1ghiMz_RtyuvwXYZ123abc\",\n *     \"is_mobile_sdk\": true\n *   }\n * }\n * ```\n *\n * ## Notes\n * - Secure authentication exchanges the `code` and `redirect_uri` provided by the client for an access token using LinkedIn's OAuth API.\n * - Insecure authentication validates the user ID and access token directly, bypassing OAuth flows. This method is **not recommended** and may introduce security vulnerabilities.\n * - `enableInsecureAuth` is **deprecated** and may be removed in future versions.\n *\n * @see {@link https://learn.microsoft.com/en-us/linkedin/shared/authentication/authentication LinkedIn Authentication Documentation}\n */\n\nimport BaseAuthCodeAdapter from './BaseCodeAuthAdapter';\nclass LinkedInAdapter extends BaseAuthCodeAdapter {\n  constructor() {\n    super('LinkedIn');\n  }\n  async getUserFromAccessToken(access_token, authData) {\n    const response = await fetch('https://api.linkedin.com/v2/me', {\n      headers: {\n        Authorization: `Bearer ${access_token}`,\n        'x-li-format': 'json',\n        'x-li-src': authData?.is_mobile_sdk ? 'msdk' : undefined,\n      },\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LinkedIn API request failed.');\n    }\n\n    return response.json();\n  }\n\n  async getAccessTokenFromCode(authData) {\n    const response = await fetch('https://www.linkedin.com/oauth/v2/accessToken', {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded',\n      },\n      body: new URLSearchParams({\n        grant_type: 'authorization_code',\n        code: authData.code,\n        redirect_uri: authData.redirect_uri,\n        client_id: this.clientId,\n        client_secret: this.clientSecret,\n      }),\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LinkedIn API request failed.');\n    }\n\n    const json = await response.json();\n    return json.access_token;\n  }\n}\n\nexport default new LinkedInAdapter();\n"],"mappings":";;;;;;AAqEA,IAAAA,oBAAA,GAAAC,sBAAA,CAAAC,OAAA;AAAwD,SAAAD,uBAAAE,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AArExD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAGA,MAAMG,eAAe,SAASC,4BAAmB,CAAC;EAChDC,WAAWA,CAAA,EAAG;IACZ,KAAK,CAAC,UAAU,CAAC;EACnB;EACA,MAAMC,sBAAsBA,CAACC,YAAY,EAAEC,QAAQ,EAAE;IACnD,MAAMC,QAAQ,GAAG,MAAMC,KAAK,CAAC,gCAAgC,EAAE;MAC7DC,OAAO,EAAE;QACPC,aAAa,EAAE,UAAUL,YAAY,EAAE;QACvC,aAAa,EAAE,MAAM;QACrB,UAAU,EAAEC,QAAQ,EAAEK,aAAa,GAAG,MAAM,GAAGC;MACjD;IACF,CAAC,CAAC;IAEF,IAAI,CAACL,QAAQ,CAACM,EAAE,EAAE;MAChB,MAAM,IAAIC,KAAK,CAACC,KAAK,CAACD,KAAK,CAACC,KAAK,CAACC,gBAAgB,EAAE,8BAA8B,CAAC;IACrF;IAEA,OAAOT,QAAQ,CAACU,IAAI,CAAC,CAAC;EACxB;EAEA,MAAMC,sBAAsBA,CAACZ,QAAQ,EAAE;IACrC,MAAMC,QAAQ,GAAG,MAAMC,KAAK,CAAC,+CAA+C,EAAE;MAC5EW,MAAM,EAAE,MAAM;MACdV,OAAO,EAAE;QACP,cAAc,EAAE;MAClB,CAAC;MACDW,IAAI,EAAE,IAAIC,eAAe,CAAC;QACxBC,UAAU,EAAE,oBAAoB;QAChCC,IAAI,EAAEjB,QAAQ,CAACiB,IAAI;QACnBC,YAAY,EAAElB,QAAQ,CAACkB,YAAY;QACnCC,SAAS,EAAE,IAAI,CAACC,QAAQ;QACxBC,aAAa,EAAE,IAAI,CAACC;MACtB,CAAC;IACH,CAAC,CAAC;IAEF,IAAI,CAACrB,QAAQ,CAACM,EAAE,EAAE;MAChB,MAAM,IAAIC,KAAK,CAACC,KAAK,CAACD,KAAK,CAACC,KAAK,CAACC,gBAAgB,EAAE,8BAA8B,CAAC;IACrF;IAEA,MAAMC,IAAI,GAAG,MAAMV,QAAQ,CAACU,IAAI,CAAC,CAAC;IAClC,OAAOA,IAAI,CAACZ,YAAY;EAC1B;AACF;AAAC,IAAAwB,QAAA,GAAAC,OAAA,CAAA9B,OAAA,GAEc,IAAIC,eAAe,CAAC,CAAC","ignoreList":[]}
@@ -1,17 +1,25 @@
1
1
  "use strict";
2
2
 
3
+ var _Config = _interopRequireDefault(require("../../Config"));
4
+ var _Deprecator = _interopRequireDefault(require("../../Deprecator/Deprecator"));
5
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
3
6
  // Helper functions for accessing the meetup API.
4
7
  var Parse = require('parse/node').Parse;
5
8
  const httpsRequest = require('./httpsRequest');
6
-
7
9
  // Returns a promise that fulfills iff this user id is valid.
8
- function validateAuthData(authData) {
9
- return request('member/self', authData.access_token).then(data => {
10
- if (data && data.id == authData.id) {
11
- return;
12
- }
13
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Meetup auth is invalid for this user.');
10
+ async function validateAuthData(authData) {
11
+ const config = _Config.default.get(Parse.applicationId);
12
+ const meetupConfig = config.auth.meetup;
13
+ _Deprecator.default.logRuntimeDeprecation({
14
+ usage: 'meetup adapter'
14
15
  });
16
+ if (!meetupConfig?.enableInsecureAuth) {
17
+ throw new Parse.Error('Meetup only works with enableInsecureAuth: true');
18
+ }
19
+ const data = await request('member/self', authData.access_token);
20
+ if (data?.id !== authData.id) {
21
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Meetup auth is invalid for this user.');
22
+ }
15
23
  }
16
24
 
17
25
  // Returns a promise that fulfills iff this app id is valid.
@@ -33,4 +41,4 @@ module.exports = {
33
41
  validateAppId: validateAppId,
34
42
  validateAuthData: validateAuthData
35
43
  };
36
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJodHRwc1JlcXVlc3QiLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwidGhlbiIsImRhdGEiLCJpZCIsIkVycm9yIiwiT0JKRUNUX05PVF9GT1VORCIsInZhbGlkYXRlQXBwSWQiLCJQcm9taXNlIiwicmVzb2x2ZSIsInBhdGgiLCJnZXQiLCJob3N0IiwiaGVhZGVycyIsIkF1dGhvcml6YXRpb24iLCJtb2R1bGUiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL0FkYXB0ZXJzL0F1dGgvbWVldHVwLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEhlbHBlciBmdW5jdGlvbnMgZm9yIGFjY2Vzc2luZyB0aGUgbWVldHVwIEFQSS5cbnZhciBQYXJzZSA9IHJlcXVpcmUoJ3BhcnNlL25vZGUnKS5QYXJzZTtcbmNvbnN0IGh0dHBzUmVxdWVzdCA9IHJlcXVpcmUoJy4vaHR0cHNSZXF1ZXN0Jyk7XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEpIHtcbiAgcmV0dXJuIHJlcXVlc3QoJ21lbWJlci9zZWxmJywgYXV0aERhdGEuYWNjZXNzX3Rva2VuKS50aGVuKGRhdGEgPT4ge1xuICAgIGlmIChkYXRhICYmIGRhdGEuaWQgPT0gYXV0aERhdGEuaWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhyb3cgbmV3IFBhcnNlLkVycm9yKFBhcnNlLkVycm9yLk9CSkVDVF9OT1RfRk9VTkQsICdNZWV0dXAgYXV0aCBpcyBpbnZhbGlkIGZvciB0aGlzIHVzZXIuJyk7XG4gIH0pO1xufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbn1cblxuLy8gQSBwcm9taXNleSB3cmFwcGVyIGZvciBhcGkgcmVxdWVzdHNcbmZ1bmN0aW9uIHJlcXVlc3QocGF0aCwgYWNjZXNzX3Rva2VuKSB7XG4gIHJldHVybiBodHRwc1JlcXVlc3QuZ2V0KHtcbiAgICBob3N0OiAnYXBpLm1lZXR1cC5jb20nLFxuICAgIHBhdGg6ICcvMi8nICsgcGF0aCxcbiAgICBoZWFkZXJzOiB7XG4gICAgICBBdXRob3JpemF0aW9uOiAnYmVhcmVyICcgKyBhY2Nlc3NfdG9rZW4sXG4gICAgfSxcbiAgfSk7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICB2YWxpZGF0ZUFwcElkOiB2YWxpZGF0ZUFwcElkLFxuICB2YWxpZGF0ZUF1dGhEYXRhOiB2YWxpZGF0ZUF1dGhEYXRhLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7QUFBQTtBQUNBLElBQUlBLEtBQUssR0FBR0MsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDRCxLQUFLO0FBQ3ZDLE1BQU1FLFlBQVksR0FBR0QsT0FBTyxDQUFDLGdCQUFnQixDQUFDOztBQUU5QztBQUNBLFNBQVNFLGdCQUFnQkEsQ0FBQ0MsUUFBUSxFQUFFO0VBQ2xDLE9BQU9DLE9BQU8sQ0FBQyxhQUFhLEVBQUVELFFBQVEsQ0FBQ0UsWUFBWSxDQUFDLENBQUNDLElBQUksQ0FBQ0MsSUFBSSxJQUFJO0lBQ2hFLElBQUlBLElBQUksSUFBSUEsSUFBSSxDQUFDQyxFQUFFLElBQUlMLFFBQVEsQ0FBQ0ssRUFBRSxFQUFFO01BQ2xDO0lBQ0Y7SUFDQSxNQUFNLElBQUlULEtBQUssQ0FBQ1UsS0FBSyxDQUFDVixLQUFLLENBQUNVLEtBQUssQ0FBQ0MsZ0JBQWdCLEVBQUUsdUNBQXVDLENBQUM7RUFDOUYsQ0FBQyxDQUFDO0FBQ0o7O0FBRUE7QUFDQSxTQUFTQyxhQUFhQSxDQUFBLEVBQUc7RUFDdkIsT0FBT0MsT0FBTyxDQUFDQyxPQUFPLENBQUMsQ0FBQztBQUMxQjs7QUFFQTtBQUNBLFNBQVNULE9BQU9BLENBQUNVLElBQUksRUFBRVQsWUFBWSxFQUFFO0VBQ25DLE9BQU9KLFlBQVksQ0FBQ2MsR0FBRyxDQUFDO0lBQ3RCQyxJQUFJLEVBQUUsZ0JBQWdCO0lBQ3RCRixJQUFJLEVBQUUsS0FBSyxHQUFHQSxJQUFJO0lBQ2xCRyxPQUFPLEVBQUU7TUFDUEMsYUFBYSxFQUFFLFNBQVMsR0FBR2I7SUFDN0I7RUFDRixDQUFDLENBQUM7QUFDSjtBQUVBYyxNQUFNLENBQUNDLE9BQU8sR0FBRztFQUNmVCxhQUFhLEVBQUVBLGFBQWE7RUFDNUJULGdCQUFnQixFQUFFQTtBQUNwQixDQUFDIiwiaWdub3JlTGlzdCI6W119
44
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfQ29uZmlnIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsInJlcXVpcmUiLCJfRGVwcmVjYXRvciIsImUiLCJfX2VzTW9kdWxlIiwiZGVmYXVsdCIsIlBhcnNlIiwiaHR0cHNSZXF1ZXN0IiwidmFsaWRhdGVBdXRoRGF0YSIsImF1dGhEYXRhIiwiY29uZmlnIiwiQ29uZmlnIiwiZ2V0IiwiYXBwbGljYXRpb25JZCIsIm1lZXR1cENvbmZpZyIsImF1dGgiLCJtZWV0dXAiLCJEZXByZWNhdG9yIiwibG9nUnVudGltZURlcHJlY2F0aW9uIiwidXNhZ2UiLCJlbmFibGVJbnNlY3VyZUF1dGgiLCJFcnJvciIsImRhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwiaWQiLCJPQkpFQ1RfTk9UX0ZPVU5EIiwidmFsaWRhdGVBcHBJZCIsIlByb21pc2UiLCJyZXNvbHZlIiwicGF0aCIsImhvc3QiLCJoZWFkZXJzIiwiQXV0aG9yaXphdGlvbiIsIm1vZHVsZSIsImV4cG9ydHMiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvQWRhcHRlcnMvQXV0aC9tZWV0dXAuanMiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSGVscGVyIGZ1bmN0aW9ucyBmb3IgYWNjZXNzaW5nIHRoZSBtZWV0dXAgQVBJLlxudmFyIFBhcnNlID0gcmVxdWlyZSgncGFyc2Uvbm9kZScpLlBhcnNlO1xuY29uc3QgaHR0cHNSZXF1ZXN0ID0gcmVxdWlyZSgnLi9odHRwc1JlcXVlc3QnKTtcbmltcG9ydCBDb25maWcgZnJvbSAnLi4vLi4vQ29uZmlnJztcbmltcG9ydCBEZXByZWNhdG9yIGZyb20gJy4uLy4uL0RlcHJlY2F0b3IvRGVwcmVjYXRvcic7XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmFzeW5jIGZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEpIHtcbiAgY29uc3QgY29uZmlnID0gQ29uZmlnLmdldChQYXJzZS5hcHBsaWNhdGlvbklkKTtcbiAgY29uc3QgbWVldHVwQ29uZmlnID0gY29uZmlnLmF1dGgubWVldHVwO1xuXG4gIERlcHJlY2F0b3IubG9nUnVudGltZURlcHJlY2F0aW9uKHsgdXNhZ2U6ICdtZWV0dXAgYWRhcHRlcicgfSk7XG5cbiAgaWYgKCFtZWV0dXBDb25maWc/LmVuYWJsZUluc2VjdXJlQXV0aCkge1xuICAgIHRocm93IG5ldyBQYXJzZS5FcnJvcignTWVldHVwIG9ubHkgd29ya3Mgd2l0aCBlbmFibGVJbnNlY3VyZUF1dGg6IHRydWUnKTtcbiAgfVxuXG4gIGNvbnN0IGRhdGEgPSBhd2FpdCByZXF1ZXN0KCdtZW1iZXIvc2VsZicsIGF1dGhEYXRhLmFjY2Vzc190b2tlbik7XG4gIGlmIChkYXRhPy5pZCAhPT0gYXV0aERhdGEuaWQpIHtcbiAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCwgJ01lZXR1cCBhdXRoIGlzIGludmFsaWQgZm9yIHRoaXMgdXNlci4nKTtcbiAgfVxufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbn1cblxuLy8gQSBwcm9taXNleSB3cmFwcGVyIGZvciBhcGkgcmVxdWVzdHNcbmZ1bmN0aW9uIHJlcXVlc3QocGF0aCwgYWNjZXNzX3Rva2VuKSB7XG4gIHJldHVybiBodHRwc1JlcXVlc3QuZ2V0KHtcbiAgICBob3N0OiAnYXBpLm1lZXR1cC5jb20nLFxuICAgIHBhdGg6ICcvMi8nICsgcGF0aCxcbiAgICBoZWFkZXJzOiB7XG4gICAgICBBdXRob3JpemF0aW9uOiAnYmVhcmVyICcgKyBhY2Nlc3NfdG9rZW4sXG4gICAgfSxcbiAgfSk7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICB2YWxpZGF0ZUFwcElkOiB2YWxpZGF0ZUFwcElkLFxuICB2YWxpZGF0ZUF1dGhEYXRhOiB2YWxpZGF0ZUF1dGhEYXRhLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7QUFHQSxJQUFBQSxPQUFBLEdBQUFDLHNCQUFBLENBQUFDLE9BQUE7QUFDQSxJQUFBQyxXQUFBLEdBQUFGLHNCQUFBLENBQUFDLE9BQUE7QUFBcUQsU0FBQUQsdUJBQUFHLENBQUEsV0FBQUEsQ0FBQSxJQUFBQSxDQUFBLENBQUFDLFVBQUEsR0FBQUQsQ0FBQSxLQUFBRSxPQUFBLEVBQUFGLENBQUE7QUFKckQ7QUFDQSxJQUFJRyxLQUFLLEdBQUdMLE9BQU8sQ0FBQyxZQUFZLENBQUMsQ0FBQ0ssS0FBSztBQUN2QyxNQUFNQyxZQUFZLEdBQUdOLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQztBQUk5QztBQUNBLGVBQWVPLGdCQUFnQkEsQ0FBQ0MsUUFBUSxFQUFFO0VBQ3hDLE1BQU1DLE1BQU0sR0FBR0MsZUFBTSxDQUFDQyxHQUFHLENBQUNOLEtBQUssQ0FBQ08sYUFBYSxDQUFDO0VBQzlDLE1BQU1DLFlBQVksR0FBR0osTUFBTSxDQUFDSyxJQUFJLENBQUNDLE1BQU07RUFFdkNDLG1CQUFVLENBQUNDLHFCQUFxQixDQUFDO0lBQUVDLEtBQUssRUFBRTtFQUFpQixDQUFDLENBQUM7RUFFN0QsSUFBSSxDQUFDTCxZQUFZLEVBQUVNLGtCQUFrQixFQUFFO0lBQ3JDLE1BQU0sSUFBSWQsS0FBSyxDQUFDZSxLQUFLLENBQUMsaURBQWlELENBQUM7RUFDMUU7RUFFQSxNQUFNQyxJQUFJLEdBQUcsTUFBTUMsT0FBTyxDQUFDLGFBQWEsRUFBRWQsUUFBUSxDQUFDZSxZQUFZLENBQUM7RUFDaEUsSUFBSUYsSUFBSSxFQUFFRyxFQUFFLEtBQUtoQixRQUFRLENBQUNnQixFQUFFLEVBQUU7SUFDNUIsTUFBTSxJQUFJbkIsS0FBSyxDQUFDZSxLQUFLLENBQUNmLEtBQUssQ0FBQ2UsS0FBSyxDQUFDSyxnQkFBZ0IsRUFBRSx1Q0FBdUMsQ0FBQztFQUM5RjtBQUNGOztBQUVBO0FBQ0EsU0FBU0MsYUFBYUEsQ0FBQSxFQUFHO0VBQ3ZCLE9BQU9DLE9BQU8sQ0FBQ0MsT0FBTyxDQUFDLENBQUM7QUFDMUI7O0FBRUE7QUFDQSxTQUFTTixPQUFPQSxDQUFDTyxJQUFJLEVBQUVOLFlBQVksRUFBRTtFQUNuQyxPQUFPakIsWUFBWSxDQUFDSyxHQUFHLENBQUM7SUFDdEJtQixJQUFJLEVBQUUsZ0JBQWdCO0lBQ3RCRCxJQUFJLEVBQUUsS0FBSyxHQUFHQSxJQUFJO0lBQ2xCRSxPQUFPLEVBQUU7TUFDUEMsYUFBYSxFQUFFLFNBQVMsR0FBR1Q7SUFDN0I7RUFDRixDQUFDLENBQUM7QUFDSjtBQUVBVSxNQUFNLENBQUNDLE9BQU8sR0FBRztFQUNmUixhQUFhLEVBQUVBLGFBQWE7RUFDNUJuQixnQkFBZ0IsRUFBRUE7QUFDcEIsQ0FBQyIsImlnbm9yZUxpc3QiOltdfQ==
@@ -8,6 +8,84 @@ var _otpauth = require("otpauth");
8
8
  var _cryptoUtils = require("../../cryptoUtils");
9
9
  var _AuthAdapter = _interopRequireDefault(require("./AuthAdapter"));
10
10
  function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
11
+ /**
12
+ * Parse Server authentication adapter for Multi-Factor Authentication (MFA).
13
+ *
14
+ * @class MFAAdapter
15
+ * @param {Object} options - The adapter options.
16
+ * @param {Array<String>} options.options - Supported MFA methods. Must include `"SMS"` or `"TOTP"`.
17
+ * @param {Number} [options.digits=6] - The number of digits for the one-time password (OTP). Must be between 4 and 10.
18
+ * @param {Number} [options.period=30] - The validity period of the OTP in seconds. Must be greater than 10.
19
+ * @param {String} [options.algorithm="SHA1"] - The algorithm used for TOTP generation. Defaults to `"SHA1"`.
20
+ * @param {Function} [options.sendSMS] - A callback function for sending SMS OTPs. Required if `"SMS"` is included in `options`.
21
+ *
22
+ * @description
23
+ * ## Parse Server Configuration
24
+ * To configure Parse Server for MFA, use the following structure:
25
+ * ```javascript
26
+ * {
27
+ * auth: {
28
+ * mfa: {
29
+ * options: ["SMS", "TOTP"],
30
+ * digits: 6,
31
+ * period: 30,
32
+ * algorithm: "SHA1",
33
+ * sendSMS: (token, mobile) => {
34
+ * // Send the SMS using your preferred SMS provider.
35
+ * console.log(`Sending SMS to ${mobile} with token: ${token}`);
36
+ * }
37
+ * }
38
+ * }
39
+ * }
40
+ * ```
41
+ *
42
+ * ## MFA Methods
43
+ * - **SMS**:
44
+ * - Requires a valid mobile number.
45
+ * - Sends a one-time password (OTP) via SMS for login or verification.
46
+ * - Uses the `sendSMS` callback for sending the OTP.
47
+ *
48
+ * - **TOTP**:
49
+ * - Requires a secret key for setup.
50
+ * - Validates the user's OTP against a time-based one-time password (TOTP) generated using the secret key.
51
+ * - Supports configurable digits, period, and algorithm for TOTP generation.
52
+ *
53
+ * ## MFA Payload
54
+ * The adapter requires the following `authData` fields:
55
+ * - **For SMS-based MFA**:
56
+ * - `mobile`: The user's mobile number (required for setup).
57
+ * - `token`: The OTP provided by the user for login or verification.
58
+ * - **For TOTP-based MFA**:
59
+ * - `secret`: The TOTP secret key for the user (required for setup).
60
+ * - `token`: The OTP provided by the user for login or verification.
61
+ *
62
+ * ## Example Payloads
63
+ * ### SMS Setup Payload
64
+ * ```json
65
+ * {
66
+ * "mobile": "+1234567890"
67
+ * }
68
+ * ```
69
+ *
70
+ * ### TOTP Setup Payload
71
+ * ```json
72
+ * {
73
+ * "secret": "BASE32ENCODEDSECRET",
74
+ * "token": "123456"
75
+ * }
76
+ * ```
77
+ *
78
+ * ### Login Payload
79
+ * ```json
80
+ * {
81
+ * "token": "123456"
82
+ * }
83
+ * ```
84
+ *
85
+ * @see {@link https://en.wikipedia.org/wiki/Time-based_One-Time_Password_algorithm Time-based One-Time Password Algorithm (TOTP)}
86
+ * @see {@link https://tools.ietf.org/html/rfc6238 RFC 6238: TOTP: Time-Based One-Time Password Algorithm}
87
+ */
88
+
11
89
  class MFAAdapter extends _AuthAdapter.default {
12
90
  validateOptions(opts) {
13
91
  const validOptions = opts.options;
@@ -133,7 +211,7 @@ class MFAAdapter extends _AuthAdapter.default {
133
211
  }
134
212
  throw 'Invalid MFA data';
135
213
  }
136
- afterFind(req, authData) {
214
+ afterFind(authData, options, req) {
137
215
  if (req.master) {
138
216
  return;
139
217
  }
@@ -243,4 +321,4 @@ class MFAAdapter extends _AuthAdapter.default {
243
321
  }
244
322
  }
245
323
  var _default = exports.default = new MFAAdapter();
246
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_otpauth","require","_cryptoUtils","_AuthAdapter","_interopRequireDefault","e","__esModule","default","MFAAdapter","AuthAdapter","validateOptions","opts","validOptions","options","Array","isArray","sms","includes","totp","digits","period","sendSMS","smsCallback","algorithm","validateSetUp","mfaData","mobile","setupMobileOTP","setupTOTP","validateLogin","loginData","_","req","saveResponse","doNotSave","token","auth","original","get","secret","recovery","saved","expiry","mfa","sendToken","object","set","save","useMasterKey","Date","TOTP","Secret","fromBase32","valid","validate","validateUpdate","authData","master","confirmSMSOTP","old","afterFind","status","policy","pending","Object","keys","length","test","randomString","replace","substring","Promise","resolve","getTime","inputData","pendingData","response","join","_default","exports"],"sources":["../../../src/Adapters/Auth/mfa.js"],"sourcesContent":["import { TOTP, Secret } from 'otpauth';\nimport { randomString } from '../../cryptoUtils';\nimport AuthAdapter from './AuthAdapter';\nclass MFAAdapter extends AuthAdapter {\n  validateOptions(opts) {\n    const validOptions = opts.options;\n    if (!Array.isArray(validOptions)) {\n      throw 'mfa.options must be an array';\n    }\n    this.sms = validOptions.includes('SMS');\n    this.totp = validOptions.includes('TOTP');\n    if (!this.sms && !this.totp) {\n      throw 'mfa.options must include SMS or TOTP';\n    }\n    const digits = opts.digits || 6;\n    const period = opts.period || 30;\n    if (typeof digits !== 'number') {\n      throw 'mfa.digits must be a number';\n    }\n    if (typeof period !== 'number') {\n      throw 'mfa.period must be a number';\n    }\n    if (digits < 4 || digits > 10) {\n      throw 'mfa.digits must be between 4 and 10';\n    }\n    if (period < 10) {\n      throw 'mfa.period must be greater than 10';\n    }\n    const sendSMS = opts.sendSMS;\n    if (this.sms && typeof sendSMS !== 'function') {\n      throw 'mfa.sendSMS callback must be defined when using SMS OTPs';\n    }\n    this.smsCallback = sendSMS;\n    this.digits = digits;\n    this.period = period;\n    this.algorithm = opts.algorithm || 'SHA1';\n  }\n  validateSetUp(mfaData) {\n    if (mfaData.mobile && this.sms) {\n      return this.setupMobileOTP(mfaData.mobile);\n    }\n    if (this.totp) {\n      return this.setupTOTP(mfaData);\n    }\n    throw 'Invalid MFA data';\n  }\n  async validateLogin(loginData, _, req) {\n    const saveResponse = {\n      doNotSave: true,\n    };\n    const token = loginData.token;\n    const auth = req.original.get('authData') || {};\n    const { secret, recovery, mobile, token: saved, expiry } = auth.mfa || {};\n    if (this.sms && mobile) {\n      if (token === 'request') {\n        const { token: sendToken, expiry } = await this.sendSMS(mobile);\n        auth.mfa.token = sendToken;\n        auth.mfa.expiry = expiry;\n        req.object.set('authData', auth);\n        await req.object.save(null, { useMasterKey: true });\n        throw 'Please enter the token';\n      }\n      if (!saved || token !== saved) {\n        throw 'Invalid MFA token 1';\n      }\n      if (new Date() > expiry) {\n        throw 'Invalid MFA token 2';\n      }\n      delete auth.mfa.token;\n      delete auth.mfa.expiry;\n      return {\n        save: auth.mfa,\n      };\n    }\n    if (this.totp) {\n      if (typeof token !== 'string') {\n        throw 'Invalid MFA token';\n      }\n      if (!secret) {\n        return saveResponse;\n      }\n      if (recovery[0] === token || recovery[1] === token) {\n        return saveResponse;\n      }\n      const totp = new TOTP({\n        algorithm: this.algorithm,\n        digits: this.digits,\n        period: this.period,\n        secret: Secret.fromBase32(secret),\n      });\n      const valid = totp.validate({\n        token,\n      });\n      if (valid === null) {\n        throw 'Invalid MFA token';\n      }\n    }\n    return saveResponse;\n  }\n  async validateUpdate(authData, _, req) {\n    if (req.master) {\n      return;\n    }\n    if (authData.mobile && this.sms) {\n      if (!authData.token) {\n        throw 'MFA is already set up on this account';\n      }\n      return this.confirmSMSOTP(authData, req.original.get('authData')?.mfa || {});\n    }\n    if (this.totp) {\n      await this.validateLogin({ token: authData.old }, null, req);\n      return this.validateSetUp(authData);\n    }\n    throw 'Invalid MFA data';\n  }\n  afterFind(req, authData) {\n    if (req.master) {\n      return;\n    }\n    if (this.totp && authData.secret) {\n      return {\n        status: 'enabled',\n      };\n    }\n    if (this.sms && authData.mobile) {\n      return {\n        status: 'enabled',\n      };\n    }\n    return {\n      status: 'disabled',\n    };\n  }\n\n  policy(req, auth) {\n    if (this.sms && auth?.pending && Object.keys(auth).length === 1) {\n      return 'default';\n    }\n    return 'additional';\n  }\n\n  async setupMobileOTP(mobile) {\n    const { token, expiry } = await this.sendSMS(mobile);\n    return {\n      save: {\n        pending: {\n          [mobile]: {\n            token,\n            expiry,\n          },\n        },\n      },\n    };\n  }\n\n  async sendSMS(mobile) {\n    if (!/^[+]*[(]{0,1}[0-9]{1,3}[)]{0,1}[-\\s\\./0-9]*$/g.test(mobile)) {\n      throw 'Invalid mobile number.';\n    }\n    let token = '';\n    while (token.length < this.digits) {\n      token += randomString(10).replace(/\\D/g, '');\n    }\n    token = token.substring(0, this.digits);\n    await Promise.resolve(this.smsCallback(token, mobile));\n    const expiry = new Date(new Date().getTime() + this.period * 1000);\n    return { token, expiry };\n  }\n\n  async confirmSMSOTP(inputData, authData) {\n    const { mobile, token } = inputData;\n    if (!authData.pending?.[mobile]) {\n      throw 'This number is not pending';\n    }\n    const pendingData = authData.pending[mobile];\n    if (token !== pendingData.token) {\n      throw 'Invalid MFA token';\n    }\n    if (new Date() > pendingData.expiry) {\n      throw 'Invalid MFA token';\n    }\n    delete authData.pending[mobile];\n    authData.mobile = mobile;\n    return {\n      save: authData,\n    };\n  }\n\n  setupTOTP(mfaData) {\n    const { secret, token } = mfaData;\n    if (!secret || !token || secret.length < 20) {\n      throw 'Invalid MFA data';\n    }\n    const totp = new TOTP({\n      algorithm: this.algorithm,\n      digits: this.digits,\n      period: this.period,\n      secret: Secret.fromBase32(secret),\n    });\n    const valid = totp.validate({\n      token,\n    });\n    if (valid === null) {\n      throw 'Invalid MFA token';\n    }\n    const recovery = [randomString(30), randomString(30)];\n    return {\n      response: { recovery: recovery.join(', ') },\n      save: { secret, recovery },\n    };\n  }\n}\nexport default new MFAAdapter();\n"],"mappings":"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","ignoreList":[]}
324
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_otpauth","require","_cryptoUtils","_AuthAdapter","_interopRequireDefault","e","__esModule","default","MFAAdapter","AuthAdapter","validateOptions","opts","validOptions","options","Array","isArray","sms","includes","totp","digits","period","sendSMS","smsCallback","algorithm","validateSetUp","mfaData","mobile","setupMobileOTP","setupTOTP","validateLogin","loginData","_","req","saveResponse","doNotSave","token","auth","original","get","secret","recovery","saved","expiry","mfa","sendToken","object","set","save","useMasterKey","Date","TOTP","Secret","fromBase32","valid","validate","validateUpdate","authData","master","confirmSMSOTP","old","afterFind","status","policy","pending","Object","keys","length","test","randomString","replace","substring","Promise","resolve","getTime","inputData","pendingData","response","join","_default","exports"],"sources":["../../../src/Adapters/Auth/mfa.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Multi-Factor Authentication (MFA).\n *\n * @class MFAAdapter\n * @param {Object} options - The adapter options.\n * @param {Array<String>} options.options - Supported MFA methods. Must include `\"SMS\"` or `\"TOTP\"`.\n * @param {Number} [options.digits=6] - The number of digits for the one-time password (OTP). Must be between 4 and 10.\n * @param {Number} [options.period=30] - The validity period of the OTP in seconds. Must be greater than 10.\n * @param {String} [options.algorithm=\"SHA1\"] - The algorithm used for TOTP generation. Defaults to `\"SHA1\"`.\n * @param {Function} [options.sendSMS] - A callback function for sending SMS OTPs. Required if `\"SMS\"` is included in `options`.\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for MFA, use the following structure:\n * ```javascript\n * {\n *   auth: {\n *     mfa: {\n *       options: [\"SMS\", \"TOTP\"],\n *       digits: 6,\n *       period: 30,\n *       algorithm: \"SHA1\",\n *       sendSMS: (token, mobile) => {\n *         // Send the SMS using your preferred SMS provider.\n *         console.log(`Sending SMS to ${mobile} with token: ${token}`);\n *       }\n *     }\n *   }\n * }\n * ```\n *\n * ## MFA Methods\n * - **SMS**:\n *   - Requires a valid mobile number.\n *   - Sends a one-time password (OTP) via SMS for login or verification.\n *   - Uses the `sendSMS` callback for sending the OTP.\n *\n * - **TOTP**:\n *   - Requires a secret key for setup.\n *   - Validates the user's OTP against a time-based one-time password (TOTP) generated using the secret key.\n *   - Supports configurable digits, period, and algorithm for TOTP generation.\n *\n * ## MFA Payload\n * The adapter requires the following `authData` fields:\n * - **For SMS-based MFA**:\n *   - `mobile`: The user's mobile number (required for setup).\n *   - `token`: The OTP provided by the user for login or verification.\n * - **For TOTP-based MFA**:\n *   - `secret`: The TOTP secret key for the user (required for setup).\n *   - `token`: The OTP provided by the user for login or verification.\n *\n * ## Example Payloads\n * ### SMS Setup Payload\n * ```json\n * {\n *   \"mobile\": \"+1234567890\"\n * }\n * ```\n *\n * ### TOTP Setup Payload\n * ```json\n * {\n *   \"secret\": \"BASE32ENCODEDSECRET\",\n *   \"token\": \"123456\"\n * }\n * ```\n *\n * ### Login Payload\n * ```json\n * {\n *   \"token\": \"123456\"\n * }\n * ```\n *\n * @see {@link https://en.wikipedia.org/wiki/Time-based_One-Time_Password_algorithm Time-based One-Time Password Algorithm (TOTP)}\n * @see {@link https://tools.ietf.org/html/rfc6238 RFC 6238: TOTP: Time-Based One-Time Password Algorithm}\n */\n\nimport { TOTP, Secret } from 'otpauth';\nimport { randomString } from '../../cryptoUtils';\nimport AuthAdapter from './AuthAdapter';\nclass MFAAdapter extends AuthAdapter {\n  validateOptions(opts) {\n    const validOptions = opts.options;\n    if (!Array.isArray(validOptions)) {\n      throw 'mfa.options must be an array';\n    }\n    this.sms = validOptions.includes('SMS');\n    this.totp = validOptions.includes('TOTP');\n    if (!this.sms && !this.totp) {\n      throw 'mfa.options must include SMS or TOTP';\n    }\n    const digits = opts.digits || 6;\n    const period = opts.period || 30;\n    if (typeof digits !== 'number') {\n      throw 'mfa.digits must be a number';\n    }\n    if (typeof period !== 'number') {\n      throw 'mfa.period must be a number';\n    }\n    if (digits < 4 || digits > 10) {\n      throw 'mfa.digits must be between 4 and 10';\n    }\n    if (period < 10) {\n      throw 'mfa.period must be greater than 10';\n    }\n    const sendSMS = opts.sendSMS;\n    if (this.sms && typeof sendSMS !== 'function') {\n      throw 'mfa.sendSMS callback must be defined when using SMS OTPs';\n    }\n    this.smsCallback = sendSMS;\n    this.digits = digits;\n    this.period = period;\n    this.algorithm = opts.algorithm || 'SHA1';\n  }\n  validateSetUp(mfaData) {\n    if (mfaData.mobile && this.sms) {\n      return this.setupMobileOTP(mfaData.mobile);\n    }\n    if (this.totp) {\n      return this.setupTOTP(mfaData);\n    }\n    throw 'Invalid MFA data';\n  }\n  async validateLogin(loginData, _, req) {\n    const saveResponse = {\n      doNotSave: true,\n    };\n    const token = loginData.token;\n    const auth = req.original.get('authData') || {};\n    const { secret, recovery, mobile, token: saved, expiry } = auth.mfa || {};\n    if (this.sms && mobile) {\n      if (token === 'request') {\n        const { token: sendToken, expiry } = await this.sendSMS(mobile);\n        auth.mfa.token = sendToken;\n        auth.mfa.expiry = expiry;\n        req.object.set('authData', auth);\n        await req.object.save(null, { useMasterKey: true });\n        throw 'Please enter the token';\n      }\n      if (!saved || token !== saved) {\n        throw 'Invalid MFA token 1';\n      }\n      if (new Date() > expiry) {\n        throw 'Invalid MFA token 2';\n      }\n      delete auth.mfa.token;\n      delete auth.mfa.expiry;\n      return {\n        save: auth.mfa,\n      };\n    }\n    if (this.totp) {\n      if (typeof token !== 'string') {\n        throw 'Invalid MFA token';\n      }\n      if (!secret) {\n        return saveResponse;\n      }\n      if (recovery[0] === token || recovery[1] === token) {\n        return saveResponse;\n      }\n      const totp = new TOTP({\n        algorithm: this.algorithm,\n        digits: this.digits,\n        period: this.period,\n        secret: Secret.fromBase32(secret),\n      });\n      const valid = totp.validate({\n        token,\n      });\n      if (valid === null) {\n        throw 'Invalid MFA token';\n      }\n    }\n    return saveResponse;\n  }\n  async validateUpdate(authData, _, req) {\n    if (req.master) {\n      return;\n    }\n    if (authData.mobile && this.sms) {\n      if (!authData.token) {\n        throw 'MFA is already set up on this account';\n      }\n      return this.confirmSMSOTP(authData, req.original.get('authData')?.mfa || {});\n    }\n    if (this.totp) {\n      await this.validateLogin({ token: authData.old }, null, req);\n      return this.validateSetUp(authData);\n    }\n    throw 'Invalid MFA data';\n  }\n  afterFind(authData, options, req) {\n    if (req.master) {\n      return;\n    }\n    if (this.totp && authData.secret) {\n      return {\n        status: 'enabled',\n      };\n    }\n    if (this.sms && authData.mobile) {\n      return {\n        status: 'enabled',\n      };\n    }\n    return {\n      status: 'disabled',\n    };\n  }\n\n  policy(req, auth) {\n    if (this.sms && auth?.pending && Object.keys(auth).length === 1) {\n      return 'default';\n    }\n    return 'additional';\n  }\n\n  async setupMobileOTP(mobile) {\n    const { token, expiry } = await this.sendSMS(mobile);\n    return {\n      save: {\n        pending: {\n          [mobile]: {\n            token,\n            expiry,\n          },\n        },\n      },\n    };\n  }\n\n  async sendSMS(mobile) {\n    if (!/^[+]*[(]{0,1}[0-9]{1,3}[)]{0,1}[-\\s\\./0-9]*$/g.test(mobile)) {\n      throw 'Invalid mobile number.';\n    }\n    let token = '';\n    while (token.length < this.digits) {\n      token += randomString(10).replace(/\\D/g, '');\n    }\n    token = token.substring(0, this.digits);\n    await Promise.resolve(this.smsCallback(token, mobile));\n    const expiry = new Date(new Date().getTime() + this.period * 1000);\n    return { token, expiry };\n  }\n\n  async confirmSMSOTP(inputData, authData) {\n    const { mobile, token } = inputData;\n    if (!authData.pending?.[mobile]) {\n      throw 'This number is not pending';\n    }\n    const pendingData = authData.pending[mobile];\n    if (token !== pendingData.token) {\n      throw 'Invalid MFA token';\n    }\n    if (new Date() > pendingData.expiry) {\n      throw 'Invalid MFA token';\n    }\n    delete authData.pending[mobile];\n    authData.mobile = mobile;\n    return {\n      save: authData,\n    };\n  }\n\n  setupTOTP(mfaData) {\n    const { secret, token } = mfaData;\n    if (!secret || !token || secret.length < 20) {\n      throw 'Invalid MFA data';\n    }\n    const totp = new TOTP({\n      algorithm: this.algorithm,\n      digits: this.digits,\n      period: this.period,\n      secret: Secret.fromBase32(secret),\n    });\n    const valid = totp.validate({\n      token,\n    });\n    if (valid === null) {\n      throw 'Invalid MFA token';\n    }\n    const recovery = [randomString(30), randomString(30)];\n    return {\n      response: { recovery: recovery.join(', ') },\n      save: { secret, recovery },\n    };\n  }\n}\nexport default new MFAAdapter();\n"],"mappings":"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","ignoreList":[]}