parse-server 8.0.1 → 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.
- package/lib/Adapters/Auth/AuthAdapter.js +16 -9
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
- package/lib/Adapters/Auth/apple.js +45 -1
- package/lib/Adapters/Auth/facebook.js +61 -1
- package/lib/Adapters/Auth/gcenter.js +201 -157
- package/lib/Adapters/Auth/github.js +119 -31
- package/lib/Adapters/Auth/google.js +45 -1
- package/lib/Adapters/Auth/gpgames.js +120 -27
- package/lib/Adapters/Auth/index.js +33 -33
- package/lib/Adapters/Auth/instagram.js +114 -24
- package/lib/Adapters/Auth/janraincapture.js +45 -1
- package/lib/Adapters/Auth/janrainengage.js +11 -2
- package/lib/Adapters/Auth/keycloak.js +68 -35
- package/lib/Adapters/Auth/ldap.js +75 -1
- package/lib/Adapters/Auth/line.js +119 -32
- package/lib/Adapters/Auth/linkedin.js +111 -35
- package/lib/Adapters/Auth/meetup.js +16 -8
- package/lib/Adapters/Auth/mfa.js +80 -2
- package/lib/Adapters/Auth/microsoft.js +105 -30
- package/lib/Adapters/Auth/oauth2.js +96 -109
- package/lib/Adapters/Auth/phantauth.js +16 -8
- package/lib/Adapters/Auth/qq.js +107 -36
- package/lib/Adapters/Auth/spotify.js +108 -39
- package/lib/Adapters/Auth/twitter.js +187 -40
- package/lib/Adapters/Auth/vkontakte.js +20 -13
- package/lib/Adapters/Auth/wechat.js +105 -25
- package/lib/Adapters/Auth/weibo.js +135 -37
- package/lib/Auth.js +26 -17
- package/lib/Config.js +14 -1
- package/lib/Deprecator/Deprecations.js +5 -2
- package/lib/Options/Definitions.js +7 -1
- package/lib/Options/docs.js +2 -1
- package/lib/Options/index.js +1 -1
- package/lib/RestWrite.js +4 -5
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +10 -1
- package/lib/cli/parse-server.js +1 -1
- package/package.json +6 -6
|
@@ -1,38 +1,125 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
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
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
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
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
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
|
-
|
|
121
|
+
return userData;
|
|
122
|
+
}
|
|
33
123
|
}
|
|
34
|
-
|
|
35
|
-
|
|
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
|
-
|
|
4
|
-
|
|
5
|
-
|
|
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
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
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
|
-
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
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
|
-
|
|
38
|
-
|
|
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
|
-
|
|
10
|
-
|
|
11
|
-
|
|
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,
|
|
44
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfQ29uZmlnIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsInJlcXVpcmUiLCJfRGVwcmVjYXRvciIsImUiLCJfX2VzTW9kdWxlIiwiZGVmYXVsdCIsIlBhcnNlIiwiaHR0cHNSZXF1ZXN0IiwidmFsaWRhdGVBdXRoRGF0YSIsImF1dGhEYXRhIiwiY29uZmlnIiwiQ29uZmlnIiwiZ2V0IiwiYXBwbGljYXRpb25JZCIsIm1lZXR1cENvbmZpZyIsImF1dGgiLCJtZWV0dXAiLCJEZXByZWNhdG9yIiwibG9nUnVudGltZURlcHJlY2F0aW9uIiwidXNhZ2UiLCJlbmFibGVJbnNlY3VyZUF1dGgiLCJFcnJvciIsImRhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwiaWQiLCJPQkpFQ1RfTk9UX0ZPVU5EIiwidmFsaWRhdGVBcHBJZCIsIlByb21pc2UiLCJyZXNvbHZlIiwicGF0aCIsImhvc3QiLCJoZWFkZXJzIiwiQXV0aG9yaXphdGlvbiIsIm1vZHVsZSIsImV4cG9ydHMiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvQWRhcHRlcnMvQXV0aC9tZWV0dXAuanMiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSGVscGVyIGZ1bmN0aW9ucyBmb3IgYWNjZXNzaW5nIHRoZSBtZWV0dXAgQVBJLlxudmFyIFBhcnNlID0gcmVxdWlyZSgncGFyc2Uvbm9kZScpLlBhcnNlO1xuY29uc3QgaHR0cHNSZXF1ZXN0ID0gcmVxdWlyZSgnLi9odHRwc1JlcXVlc3QnKTtcbmltcG9ydCBDb25maWcgZnJvbSAnLi4vLi4vQ29uZmlnJztcbmltcG9ydCBEZXByZWNhdG9yIGZyb20gJy4uLy4uL0RlcHJlY2F0b3IvRGVwcmVjYXRvcic7XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmFzeW5jIGZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEpIHtcbiAgY29uc3QgY29uZmlnID0gQ29uZmlnLmdldChQYXJzZS5hcHBsaWNhdGlvbklkKTtcbiAgY29uc3QgbWVldHVwQ29uZmlnID0gY29uZmlnLmF1dGgubWVldHVwO1xuXG4gIERlcHJlY2F0b3IubG9nUnVudGltZURlcHJlY2F0aW9uKHsgdXNhZ2U6ICdtZWV0dXAgYWRhcHRlcicgfSk7XG5cbiAgaWYgKCFtZWV0dXBDb25maWc/LmVuYWJsZUluc2VjdXJlQXV0aCkge1xuICAgIHRocm93IG5ldyBQYXJzZS5FcnJvcignTWVldHVwIG9ubHkgd29ya3Mgd2l0aCBlbmFibGVJbnNlY3VyZUF1dGg6IHRydWUnKTtcbiAgfVxuXG4gIGNvbnN0IGRhdGEgPSBhd2FpdCByZXF1ZXN0KCdtZW1iZXIvc2VsZicsIGF1dGhEYXRhLmFjY2Vzc190b2tlbik7XG4gIGlmIChkYXRhPy5pZCAhPT0gYXV0aERhdGEuaWQpIHtcbiAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCwgJ01lZXR1cCBhdXRoIGlzIGludmFsaWQgZm9yIHRoaXMgdXNlci4nKTtcbiAgfVxufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbn1cblxuLy8gQSBwcm9taXNleSB3cmFwcGVyIGZvciBhcGkgcmVxdWVzdHNcbmZ1bmN0aW9uIHJlcXVlc3QocGF0aCwgYWNjZXNzX3Rva2VuKSB7XG4gIHJldHVybiBodHRwc1JlcXVlc3QuZ2V0KHtcbiAgICBob3N0OiAnYXBpLm1lZXR1cC5jb20nLFxuICAgIHBhdGg6ICcvMi8nICsgcGF0aCxcbiAgICBoZWFkZXJzOiB7XG4gICAgICBBdXRob3JpemF0aW9uOiAnYmVhcmVyICcgKyBhY2Nlc3NfdG9rZW4sXG4gICAgfSxcbiAgfSk7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICB2YWxpZGF0ZUFwcElkOiB2YWxpZGF0ZUFwcElkLFxuICB2YWxpZGF0ZUF1dGhEYXRhOiB2YWxpZGF0ZUF1dGhEYXRhLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7QUFHQSxJQUFBQSxPQUFBLEdBQUFDLHNCQUFBLENBQUFDLE9BQUE7QUFDQSxJQUFBQyxXQUFBLEdBQUFGLHNCQUFBLENBQUFDLE9BQUE7QUFBcUQsU0FBQUQsdUJBQUFHLENBQUEsV0FBQUEsQ0FBQSxJQUFBQSxDQUFBLENBQUFDLFVBQUEsR0FBQUQsQ0FBQSxLQUFBRSxPQUFBLEVBQUFGLENBQUE7QUFKckQ7QUFDQSxJQUFJRyxLQUFLLEdBQUdMLE9BQU8sQ0FBQyxZQUFZLENBQUMsQ0FBQ0ssS0FBSztBQUN2QyxNQUFNQyxZQUFZLEdBQUdOLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQztBQUk5QztBQUNBLGVBQWVPLGdCQUFnQkEsQ0FBQ0MsUUFBUSxFQUFFO0VBQ3hDLE1BQU1DLE1BQU0sR0FBR0MsZUFBTSxDQUFDQyxHQUFHLENBQUNOLEtBQUssQ0FBQ08sYUFBYSxDQUFDO0VBQzlDLE1BQU1DLFlBQVksR0FBR0osTUFBTSxDQUFDSyxJQUFJLENBQUNDLE1BQU07RUFFdkNDLG1CQUFVLENBQUNDLHFCQUFxQixDQUFDO0lBQUVDLEtBQUssRUFBRTtFQUFpQixDQUFDLENBQUM7RUFFN0QsSUFBSSxDQUFDTCxZQUFZLEVBQUVNLGtCQUFrQixFQUFFO0lBQ3JDLE1BQU0sSUFBSWQsS0FBSyxDQUFDZSxLQUFLLENBQUMsaURBQWlELENBQUM7RUFDMUU7RUFFQSxNQUFNQyxJQUFJLEdBQUcsTUFBTUMsT0FBTyxDQUFDLGFBQWEsRUFBRWQsUUFBUSxDQUFDZSxZQUFZLENBQUM7RUFDaEUsSUFBSUYsSUFBSSxFQUFFRyxFQUFFLEtBQUtoQixRQUFRLENBQUNnQixFQUFFLEVBQUU7SUFDNUIsTUFBTSxJQUFJbkIsS0FBSyxDQUFDZSxLQUFLLENBQUNmLEtBQUssQ0FBQ2UsS0FBSyxDQUFDSyxnQkFBZ0IsRUFBRSx1Q0FBdUMsQ0FBQztFQUM5RjtBQUNGOztBQUVBO0FBQ0EsU0FBU0MsYUFBYUEsQ0FBQSxFQUFHO0VBQ3ZCLE9BQU9DLE9BQU8sQ0FBQ0MsT0FBTyxDQUFDLENBQUM7QUFDMUI7O0FBRUE7QUFDQSxTQUFTTixPQUFPQSxDQUFDTyxJQUFJLEVBQUVOLFlBQVksRUFBRTtFQUNuQyxPQUFPakIsWUFBWSxDQUFDSyxHQUFHLENBQUM7SUFDdEJtQixJQUFJLEVBQUUsZ0JBQWdCO0lBQ3RCRCxJQUFJLEVBQUUsS0FBSyxHQUFHQSxJQUFJO0lBQ2xCRSxPQUFPLEVBQUU7TUFDUEMsYUFBYSxFQUFFLFNBQVMsR0FBR1Q7SUFDN0I7RUFDRixDQUFDLENBQUM7QUFDSjtBQUVBVSxNQUFNLENBQUNDLE9BQU8sR0FBRztFQUNmUixhQUFhLEVBQUVBLGFBQWE7RUFDNUJuQixnQkFBZ0IsRUFBRUE7QUFDcEIsQ0FBQyIsImlnbm9yZUxpc3QiOltdfQ==
|
package/lib/Adapters/Auth/mfa.js
CHANGED
|
@@ -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(
|
|
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":";;;;;;AAAA,IAAAA,QAAA,GAAAC,OAAA;AACA,IAAAC,YAAA,GAAAD,OAAA;AACA,IAAAE,YAAA,GAAAC,sBAAA,CAAAH,OAAA;AAAwC,SAAAG,uBAAAC,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AACxC,MAAMG,UAAU,SAASC,oBAAW,CAAC;EACnCC,eAAeA,CAACC,IAAI,EAAE;IACpB,MAAMC,YAAY,GAAGD,IAAI,CAACE,OAAO;IACjC,IAAI,CAACC,KAAK,CAACC,OAAO,CAACH,YAAY,CAAC,EAAE;MAChC,MAAM,8BAA8B;IACtC;IACA,IAAI,CAACI,GAAG,GAAGJ,YAAY,CAACK,QAAQ,CAAC,KAAK,CAAC;IACvC,IAAI,CAACC,IAAI,GAAGN,YAAY,CAACK,QAAQ,CAAC,MAAM,CAAC;IACzC,IAAI,CAAC,IAAI,CAACD,GAAG,IAAI,CAAC,IAAI,CAACE,IAAI,EAAE;MAC3B,MAAM,sCAAsC;IAC9C;IACA,MAAMC,MAAM,GAAGR,IAAI,CAACQ,MAAM,IAAI,CAAC;IAC/B,MAAMC,MAAM,GAAGT,IAAI,CAACS,MAAM,IAAI,EAAE;IAChC,IAAI,OAAOD,MAAM,KAAK,QAAQ,EAAE;MAC9B,MAAM,6BAA6B;IACrC;IACA,IAAI,OAAOC,MAAM,KAAK,QAAQ,EAAE;MAC9B,MAAM,6BAA6B;IACrC;IACA,IAAID,MAAM,GAAG,CAAC,IAAIA,MAAM,GAAG,EAAE,EAAE;MAC7B,MAAM,qCAAqC;IAC7C;IACA,IAAIC,MAAM,GAAG,EAAE,EAAE;MACf,MAAM,oCAAoC;IAC5C;IACA,MAAMC,OAAO,GAAGV,IAAI,CAACU,OAAO;IAC5B,IAAI,IAAI,CAACL,GAAG,IAAI,OAAOK,OAAO,KAAK,UAAU,EAAE;MAC7C,MAAM,0DAA0D;IAClE;IACA,IAAI,CAACC,WAAW,GAAGD,OAAO;IAC1B,IAAI,CAACF,MAAM,GAAGA,MAAM;IACpB,IAAI,CAACC,MAAM,GAAGA,MAAM;IACpB,IAAI,CAACG,SAAS,GAAGZ,IAAI,CAACY,SAAS,IAAI,MAAM;EAC3C;EACAC,aAAaA,CAACC,OAAO,EAAE;IACrB,IAAIA,OAAO,CAACC,MAAM,IAAI,IAAI,CAACV,GAAG,EAAE;MAC9B,OAAO,IAAI,CAACW,cAAc,CAACF,OAAO,CAACC,MAAM,CAAC;IAC5C;IACA,IAAI,IAAI,CAACR,IAAI,EAAE;MACb,OAAO,IAAI,CAACU,SAAS,CAACH,OAAO,CAAC;IAChC;IACA,MAAM,kBAAkB;EAC1B;EACA,MAAMI,aAAaA,CAACC,SAAS,EAAEC,CAAC,EAAEC,GAAG,EAAE;IACrC,MAAMC,YAAY,GAAG;MACnBC,SAAS,EAAE;IACb,CAAC;IACD,MAAMC,KAAK,GAAGL,SAAS,CAACK,KAAK;IAC7B,MAAMC,IAAI,GAAGJ,GAAG,CAACK,QAAQ,CAACC,GAAG,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IAC/C,MAAM;MAAEC,MAAM;MAAEC,QAAQ;MAAEd,MAAM;MAAES,KAAK,EAAEM,KAAK;MAAEC;IAAO,CAAC,GAAGN,IAAI,CAACO,GAAG,IAAI,CAAC,CAAC;IACzE,IAAI,IAAI,CAAC3B,GAAG,IAAIU,MAAM,EAAE;MACtB,IAAIS,KAAK,KAAK,SAAS,EAAE;QACvB,MAAM;UAAEA,KAAK,EAAES,SAAS;UAAEF;QAAO,CAAC,GAAG,MAAM,IAAI,CAACrB,OAAO,CAACK,MAAM,CAAC;QAC/DU,IAAI,CAACO,GAAG,CAACR,KAAK,GAAGS,SAAS;QAC1BR,IAAI,CAACO,GAAG,CAACD,MAAM,GAAGA,MAAM;QACxBV,GAAG,CAACa,MAAM,CAACC,GAAG,CAAC,UAAU,EAAEV,IAAI,CAAC;QAChC,MAAMJ,GAAG,CAACa,MAAM,CAACE,IAAI,CAAC,IAAI,EAAE;UAAEC,YAAY,EAAE;QAAK,CAAC,CAAC;QACnD,MAAM,wBAAwB;MAChC;MACA,IAAI,CAACP,KAAK,IAAIN,KAAK,KAAKM,KAAK,EAAE;QAC7B,MAAM,qBAAqB;MAC7B;MACA,IAAI,IAAIQ,IAAI,CAAC,CAAC,GAAGP,MAAM,EAAE;QACvB,MAAM,qBAAqB;MAC7B;MACA,OAAON,IAAI,CAACO,GAAG,CAACR,KAAK;MACrB,OAAOC,IAAI,CAACO,GAAG,CAACD,MAAM;MACtB,OAAO;QACLK,IAAI,EAAEX,IAAI,CAACO;MACb,CAAC;IACH;IACA,IAAI,IAAI,CAACzB,IAAI,EAAE;MACb,IAAI,OAAOiB,KAAK,KAAK,QAAQ,EAAE;QAC7B,MAAM,mBAAmB;MAC3B;MACA,IAAI,CAACI,MAAM,EAAE;QACX,OAAON,YAAY;MACrB;MACA,IAAIO,QAAQ,CAAC,CAAC,CAAC,KAAKL,KAAK,IAAIK,QAAQ,CAAC,CAAC,CAAC,KAAKL,KAAK,EAAE;QAClD,OAAOF,YAAY;MACrB;MACA,MAAMf,IAAI,GAAG,IAAIgC,aAAI,CAAC;QACpB3B,SAAS,EAAE,IAAI,CAACA,SAAS;QACzBJ,MAAM,EAAE,IAAI,CAACA,MAAM;QACnBC,MAAM,EAAE,IAAI,CAACA,MAAM;QACnBmB,MAAM,EAAEY,eAAM,CAACC,UAAU,CAACb,MAAM;MAClC,CAAC,CAAC;MACF,MAAMc,KAAK,GAAGnC,IAAI,CAACoC,QAAQ,CAAC;QAC1BnB;MACF,CAAC,CAAC;MACF,IAAIkB,KAAK,KAAK,IAAI,EAAE;QAClB,MAAM,mBAAmB;MAC3B;IACF;IACA,OAAOpB,YAAY;EACrB;EACA,MAAMsB,cAAcA,CAACC,QAAQ,EAAEzB,CAAC,EAAEC,GAAG,EAAE;IACrC,IAAIA,GAAG,CAACyB,MAAM,EAAE;MACd;IACF;IACA,IAAID,QAAQ,CAAC9B,MAAM,IAAI,IAAI,CAACV,GAAG,EAAE;MAC/B,IAAI,CAACwC,QAAQ,CAACrB,KAAK,EAAE;QACnB,MAAM,uCAAuC;MAC/C;MACA,OAAO,IAAI,CAACuB,aAAa,CAACF,QAAQ,EAAExB,GAAG,CAACK,QAAQ,CAACC,GAAG,CAAC,UAAU,CAAC,EAAEK,GAAG,IAAI,CAAC,CAAC,CAAC;IAC9E;IACA,IAAI,IAAI,CAACzB,IAAI,EAAE;MACb,MAAM,IAAI,CAACW,aAAa,CAAC;QAAEM,KAAK,EAAEqB,QAAQ,CAACG;MAAI,CAAC,EAAE,IAAI,EAAE3B,GAAG,CAAC;MAC5D,OAAO,IAAI,CAACR,aAAa,CAACgC,QAAQ,CAAC;IACrC;IACA,MAAM,kBAAkB;EAC1B;EACAI,SAASA,CAAC5B,GAAG,EAAEwB,QAAQ,EAAE;IACvB,IAAIxB,GAAG,CAACyB,MAAM,EAAE;MACd;IACF;IACA,IAAI,IAAI,CAACvC,IAAI,IAAIsC,QAAQ,CAACjB,MAAM,EAAE;MAChC,OAAO;QACLsB,MAAM,EAAE;MACV,CAAC;IACH;IACA,IAAI,IAAI,CAAC7C,GAAG,IAAIwC,QAAQ,CAAC9B,MAAM,EAAE;MAC/B,OAAO;QACLmC,MAAM,EAAE;MACV,CAAC;IACH;IACA,OAAO;MACLA,MAAM,EAAE;IACV,CAAC;EACH;EAEAC,MAAMA,CAAC9B,GAAG,EAAEI,IAAI,EAAE;IAChB,IAAI,IAAI,CAACpB,GAAG,IAAIoB,IAAI,EAAE2B,OAAO,IAAIC,MAAM,CAACC,IAAI,CAAC7B,IAAI,CAAC,CAAC8B,MAAM,KAAK,CAAC,EAAE;MAC/D,OAAO,SAAS;IAClB;IACA,OAAO,YAAY;EACrB;EAEA,MAAMvC,cAAcA,CAACD,MAAM,EAAE;IAC3B,MAAM;MAAES,KAAK;MAAEO;IAAO,CAAC,GAAG,MAAM,IAAI,CAACrB,OAAO,CAACK,MAAM,CAAC;IACpD,OAAO;MACLqB,IAAI,EAAE;QACJgB,OAAO,EAAE;UACP,CAACrC,MAAM,GAAG;YACRS,KAAK;YACLO;UACF;QACF;MACF;IACF,CAAC;EACH;EAEA,MAAMrB,OAAOA,CAACK,MAAM,EAAE;IACpB,IAAI,CAAC,+CAA+C,CAACyC,IAAI,CAACzC,MAAM,CAAC,EAAE;MACjE,MAAM,wBAAwB;IAChC;IACA,IAAIS,KAAK,GAAG,EAAE;IACd,OAAOA,KAAK,CAAC+B,MAAM,GAAG,IAAI,CAAC/C,MAAM,EAAE;MACjCgB,KAAK,IAAI,IAAAiC,yBAAY,EAAC,EAAE,CAAC,CAACC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC;IAC9C;IACAlC,KAAK,GAAGA,KAAK,CAACmC,SAAS,CAAC,CAAC,EAAE,IAAI,CAACnD,MAAM,CAAC;IACvC,MAAMoD,OAAO,CAACC,OAAO,CAAC,IAAI,CAAClD,WAAW,CAACa,KAAK,EAAET,MAAM,CAAC,CAAC;IACtD,MAAMgB,MAAM,GAAG,IAAIO,IAAI,CAAC,IAAIA,IAAI,CAAC,CAAC,CAACwB,OAAO,CAAC,CAAC,GAAG,IAAI,CAACrD,MAAM,GAAG,IAAI,CAAC;IAClE,OAAO;MAAEe,KAAK;MAAEO;IAAO,CAAC;EAC1B;EAEA,MAAMgB,aAAaA,CAACgB,SAAS,EAAElB,QAAQ,EAAE;IACvC,MAAM;MAAE9B,MAAM;MAAES;IAAM,CAAC,GAAGuC,SAAS;IACnC,IAAI,CAAClB,QAAQ,CAACO,OAAO,GAAGrC,MAAM,CAAC,EAAE;MAC/B,MAAM,4BAA4B;IACpC;IACA,MAAMiD,WAAW,GAAGnB,QAAQ,CAACO,OAAO,CAACrC,MAAM,CAAC;IAC5C,IAAIS,KAAK,KAAKwC,WAAW,CAACxC,KAAK,EAAE;MAC/B,MAAM,mBAAmB;IAC3B;IACA,IAAI,IAAIc,IAAI,CAAC,CAAC,GAAG0B,WAAW,CAACjC,MAAM,EAAE;MACnC,MAAM,mBAAmB;IAC3B;IACA,OAAOc,QAAQ,CAACO,OAAO,CAACrC,MAAM,CAAC;IAC/B8B,QAAQ,CAAC9B,MAAM,GAAGA,MAAM;IACxB,OAAO;MACLqB,IAAI,EAAES;IACR,CAAC;EACH;EAEA5B,SAASA,CAACH,OAAO,EAAE;IACjB,MAAM;MAAEc,MAAM;MAAEJ;IAAM,CAAC,GAAGV,OAAO;IACjC,IAAI,CAACc,MAAM,IAAI,CAACJ,KAAK,IAAII,MAAM,CAAC2B,MAAM,GAAG,EAAE,EAAE;MAC3C,MAAM,kBAAkB;IAC1B;IACA,MAAMhD,IAAI,GAAG,IAAIgC,aAAI,CAAC;MACpB3B,SAAS,EAAE,IAAI,CAACA,SAAS;MACzBJ,MAAM,EAAE,IAAI,CAACA,MAAM;MACnBC,MAAM,EAAE,IAAI,CAACA,MAAM;MACnBmB,MAAM,EAAEY,eAAM,CAACC,UAAU,CAACb,MAAM;IAClC,CAAC,CAAC;IACF,MAAMc,KAAK,GAAGnC,IAAI,CAACoC,QAAQ,CAAC;MAC1BnB;IACF,CAAC,CAAC;IACF,IAAIkB,KAAK,KAAK,IAAI,EAAE;MAClB,MAAM,mBAAmB;IAC3B;IACA,MAAMb,QAAQ,GAAG,CAAC,IAAA4B,yBAAY,EAAC,EAAE,CAAC,EAAE,IAAAA,yBAAY,EAAC,EAAE,CAAC,CAAC;IACrD,OAAO;MACLQ,QAAQ,EAAE;QAAEpC,QAAQ,EAAEA,QAAQ,CAACqC,IAAI,CAAC,IAAI;MAAE,CAAC;MAC3C9B,IAAI,EAAE;QAAER,MAAM;QAAEC;MAAS;IAC3B,CAAC;EACH;AACF;AAAC,IAAAsC,QAAA,GAAAC,OAAA,CAAAxE,OAAA,GACc,IAAIC,UAAU,CAAC,CAAC","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":[]}
|