parse-server 2.8.4 → 8.6.2
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/LICENSE +167 -25
- package/NOTICE +10 -0
- package/README.md +929 -278
- package/lib/AccountLockout.js +47 -30
- package/lib/Adapters/AdapterLoader.js +21 -6
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
- package/lib/Adapters/Auth/AuthAdapter.js +116 -13
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
- package/lib/Adapters/Auth/OAuth1Client.js +27 -46
- package/lib/Adapters/Auth/apple.js +123 -0
- package/lib/Adapters/Auth/facebook.js +162 -35
- package/lib/Adapters/Auth/gcenter.js +217 -0
- package/lib/Adapters/Auth/github.js +118 -48
- package/lib/Adapters/Auth/google.js +160 -51
- package/lib/Adapters/Auth/gpgames.js +125 -0
- package/lib/Adapters/Auth/httpsRequest.js +6 -7
- package/lib/Adapters/Auth/index.js +170 -62
- package/lib/Adapters/Auth/instagram.js +114 -40
- package/lib/Adapters/Auth/janraincapture.js +52 -23
- package/lib/Adapters/Auth/janrainengage.js +19 -36
- package/lib/Adapters/Auth/keycloak.js +148 -0
- package/lib/Adapters/Auth/ldap.js +167 -0
- package/lib/Adapters/Auth/line.js +125 -0
- package/lib/Adapters/Auth/linkedin.js +111 -55
- package/lib/Adapters/Auth/meetup.js +24 -34
- package/lib/Adapters/Auth/mfa.js +324 -0
- package/lib/Adapters/Auth/microsoft.js +111 -0
- package/lib/Adapters/Auth/oauth2.js +97 -162
- package/lib/Adapters/Auth/phantauth.js +53 -0
- package/lib/Adapters/Auth/qq.js +108 -49
- package/lib/Adapters/Auth/spotify.js +107 -55
- package/lib/Adapters/Auth/twitter.js +188 -48
- package/lib/Adapters/Auth/utils.js +28 -0
- package/lib/Adapters/Auth/vkontakte.js +26 -39
- package/lib/Adapters/Auth/wechat.js +106 -44
- package/lib/Adapters/Auth/weibo.js +132 -58
- package/lib/Adapters/Cache/CacheAdapter.js +13 -8
- package/lib/Adapters/Cache/InMemoryCache.js +3 -13
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
- package/lib/Adapters/Cache/LRUCache.js +13 -27
- package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
- package/lib/Adapters/Cache/SchemaCache.js +25 -0
- package/lib/Adapters/Email/MailAdapter.js +10 -8
- package/lib/Adapters/Files/FilesAdapter.js +83 -25
- package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
- package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
- package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
- package/lib/Adapters/Logger/WinstonLogger.js +69 -88
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
- package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
- package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
- package/lib/Adapters/Push/PushAdapter.js +14 -7
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
- package/lib/Auth.js +488 -125
- package/lib/ClientSDK.js +2 -6
- package/lib/Config.js +525 -94
- package/lib/Controllers/AdaptableController.js +5 -25
- package/lib/Controllers/AnalyticsController.js +22 -23
- package/lib/Controllers/CacheController.js +10 -31
- package/lib/Controllers/DatabaseController.js +767 -313
- package/lib/Controllers/FilesController.js +49 -54
- package/lib/Controllers/HooksController.js +80 -84
- package/lib/Controllers/LiveQueryController.js +35 -22
- package/lib/Controllers/LoggerController.js +22 -58
- package/lib/Controllers/ParseGraphQLController.js +293 -0
- package/lib/Controllers/PushController.js +58 -49
- package/lib/Controllers/SchemaController.js +916 -422
- package/lib/Controllers/UserController.js +265 -180
- package/lib/Controllers/index.js +90 -125
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +30 -0
- package/lib/Deprecator/Deprecator.js +127 -0
- package/lib/Error.js +48 -0
- package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
- package/lib/GraphQL/ParseGraphQLServer.js +214 -0
- package/lib/GraphQL/helpers/objectsMutations.js +30 -0
- package/lib/GraphQL/helpers/objectsQueries.js +246 -0
- package/lib/GraphQL/loaders/configMutations.js +87 -0
- package/lib/GraphQL/loaders/configQueries.js +79 -0
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
- package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
- package/lib/GraphQL/loaders/filesMutations.js +107 -0
- package/lib/GraphQL/loaders/functionsMutations.js +78 -0
- package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
- package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
- package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
- package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
- package/lib/GraphQL/loaders/schemaMutations.js +162 -0
- package/lib/GraphQL/loaders/schemaQueries.js +81 -0
- package/lib/GraphQL/loaders/schemaTypes.js +341 -0
- package/lib/GraphQL/loaders/usersMutations.js +433 -0
- package/lib/GraphQL/loaders/usersQueries.js +90 -0
- package/lib/GraphQL/parseGraphQLUtils.js +63 -0
- package/lib/GraphQL/transformers/className.js +14 -0
- package/lib/GraphQL/transformers/constraintType.js +53 -0
- package/lib/GraphQL/transformers/inputType.js +51 -0
- package/lib/GraphQL/transformers/mutation.js +274 -0
- package/lib/GraphQL/transformers/outputType.js +51 -0
- package/lib/GraphQL/transformers/query.js +237 -0
- package/lib/GraphQL/transformers/schemaFields.js +99 -0
- package/lib/KeyPromiseQueue.js +48 -0
- package/lib/LiveQuery/Client.js +25 -33
- package/lib/LiveQuery/Id.js +2 -5
- package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
- package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
- package/lib/LiveQuery/ParsePubSub.js +7 -16
- package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
- package/lib/LiveQuery/QueryTools.js +76 -15
- package/lib/LiveQuery/RequestSchema.js +111 -97
- package/lib/LiveQuery/SessionTokenCache.js +23 -36
- package/lib/LiveQuery/Subscription.js +8 -17
- package/lib/LiveQuery/equalObjects.js +2 -3
- package/lib/Options/Definitions.js +1355 -382
- package/lib/Options/docs.js +301 -62
- package/lib/Options/index.js +11 -1
- package/lib/Options/parsers.js +14 -10
- package/lib/Page.js +44 -0
- package/lib/ParseMessageQueue.js +6 -13
- package/lib/ParseServer.js +474 -235
- package/lib/ParseServerRESTController.js +102 -40
- package/lib/PromiseRouter.js +39 -50
- package/lib/Push/PushQueue.js +24 -30
- package/lib/Push/PushWorker.js +32 -56
- package/lib/Push/utils.js +22 -35
- package/lib/RestQuery.js +361 -139
- package/lib/RestWrite.js +713 -344
- package/lib/Routers/AggregateRouter.js +97 -71
- package/lib/Routers/AnalyticsRouter.js +8 -14
- package/lib/Routers/AudiencesRouter.js +16 -35
- package/lib/Routers/ClassesRouter.js +86 -72
- package/lib/Routers/CloudCodeRouter.js +28 -37
- package/lib/Routers/FeaturesRouter.js +22 -25
- package/lib/Routers/FilesRouter.js +266 -171
- package/lib/Routers/FunctionsRouter.js +87 -103
- package/lib/Routers/GlobalConfigRouter.js +94 -33
- package/lib/Routers/GraphQLRouter.js +41 -0
- package/lib/Routers/HooksRouter.js +43 -47
- package/lib/Routers/IAPValidationRouter.js +57 -70
- package/lib/Routers/InstallationsRouter.js +17 -25
- package/lib/Routers/LogsRouter.js +10 -25
- package/lib/Routers/PagesRouter.js +647 -0
- package/lib/Routers/PublicAPIRouter.js +104 -112
- package/lib/Routers/PurgeRouter.js +19 -29
- package/lib/Routers/PushRouter.js +14 -28
- package/lib/Routers/RolesRouter.js +7 -14
- package/lib/Routers/SchemasRouter.js +63 -42
- package/lib/Routers/SecurityRouter.js +34 -0
- package/lib/Routers/SessionsRouter.js +25 -38
- package/lib/Routers/UsersRouter.js +463 -190
- package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
- package/lib/SchemaMigrations/Migrations.js +30 -0
- package/lib/Security/Check.js +109 -0
- package/lib/Security/CheckGroup.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
- package/lib/Security/CheckGroups/CheckGroups.js +21 -0
- package/lib/Security/CheckRunner.js +213 -0
- package/lib/SharedRest.js +29 -0
- package/lib/StatusHandler.js +96 -93
- package/lib/TestUtils.js +70 -14
- package/lib/Utils.js +468 -0
- package/lib/batch.js +74 -40
- package/lib/cache.js +8 -8
- package/lib/cli/definitions/parse-live-query-server.js +4 -3
- package/lib/cli/definitions/parse-server.js +4 -3
- package/lib/cli/parse-live-query-server.js +9 -17
- package/lib/cli/parse-server.js +49 -47
- package/lib/cli/utils/commander.js +20 -29
- package/lib/cli/utils/runner.js +31 -32
- package/lib/cloud-code/Parse.Cloud.js +711 -36
- package/lib/cloud-code/Parse.Server.js +21 -0
- package/lib/cryptoUtils.js +6 -11
- package/lib/defaults.js +21 -15
- package/lib/deprecated.js +1 -1
- package/lib/index.js +78 -67
- package/lib/logger.js +12 -20
- package/lib/middlewares.js +484 -160
- package/lib/password.js +10 -6
- package/lib/request.js +175 -0
- package/lib/requiredParameter.js +4 -3
- package/lib/rest.js +157 -82
- package/lib/triggers.js +627 -185
- package/lib/vendor/README.md +3 -3
- package/lib/vendor/mongodbUrl.js +224 -137
- package/package.json +135 -57
- package/postinstall.js +38 -50
- package/public_html/invalid_verification_link.html +3 -3
- package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
- package/types/@types/deepcopy/index.d.ts +5 -0
- package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
- package/types/Options/index.d.ts +301 -0
- package/types/ParseServer.d.ts +65 -0
- package/types/eslint.config.mjs +30 -0
- package/types/index.d.ts +21 -0
- package/types/logger.d.ts +2 -0
- package/types/tests.ts +44 -0
- package/types/tsconfig.json +24 -0
- package/CHANGELOG.md +0 -1246
- package/PATENTS +0 -37
- package/bin/dev +0 -37
- package/lib/.DS_Store +0 -0
- package/lib/Adapters/Auth/common.js +0 -2
- package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
- package/lib/Controllers/SchemaCache.js +0 -97
- package/lib/LiveQuery/.DS_Store +0 -0
- package/lib/cli/utils/parsers.js +0 -77
- package/lib/cloud-code/.DS_Store +0 -0
- package/lib/cloud-code/HTTPResponse.js +0 -57
- package/lib/cloud-code/Untitled-1 +0 -123
- package/lib/cloud-code/httpRequest.js +0 -102
- package/lib/cloud-code/team.html +0 -123
- package/lib/graphql/ParseClass.js +0 -234
- package/lib/graphql/Schema.js +0 -197
- package/lib/graphql/index.js +0 -1
- package/lib/graphql/types/ACL.js +0 -35
- package/lib/graphql/types/Date.js +0 -25
- package/lib/graphql/types/File.js +0 -24
- package/lib/graphql/types/GeoPoint.js +0 -35
- package/lib/graphql/types/JSONObject.js +0 -30
- package/lib/graphql/types/NumberInput.js +0 -43
- package/lib/graphql/types/NumberQuery.js +0 -42
- package/lib/graphql/types/Pointer.js +0 -35
- package/lib/graphql/types/QueryConstraint.js +0 -61
- package/lib/graphql/types/StringQuery.js +0 -39
- package/lib/graphql/types/index.js +0 -110
|
@@ -1,55 +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 GitHub.
|
|
11
|
+
* @class GitHubAdapter
|
|
12
|
+
* @param {Object} options - The adapter configuration options.
|
|
13
|
+
* @param {string} options.clientId - The GitHub App Client ID. Required for secure authentication.
|
|
14
|
+
* @param {string} options.clientSecret - The GitHub App Client Secret. Required for secure authentication.
|
|
15
|
+
* @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).
|
|
16
|
+
*
|
|
17
|
+
* @param {Object} authData - The authentication data provided by the client.
|
|
18
|
+
* @param {string} authData.code - The authorization code from GitHub. Required for secure authentication.
|
|
19
|
+
* @param {string} [authData.id] - **[DEPRECATED]** The GitHub user ID (required for insecure authentication).
|
|
20
|
+
* @param {string} [authData.access_token] - **[DEPRECATED]** The GitHub access token (required for insecure authentication).
|
|
21
|
+
*
|
|
22
|
+
* @description
|
|
23
|
+
* ## Parse Server Configuration
|
|
24
|
+
* * To configure Parse Server for GitHub authentication, use the following structure:
|
|
25
|
+
* ```json
|
|
26
|
+
* {
|
|
27
|
+
* "auth": {
|
|
28
|
+
* "github": {
|
|
29
|
+
* "clientId": "12345",
|
|
30
|
+
* "clientSecret": "abcde"
|
|
31
|
+
* }
|
|
32
|
+
* }
|
|
33
|
+
* ```
|
|
34
|
+
*
|
|
35
|
+
* The GitHub adapter exchanges the `authData.code` provided by the client for an access token using GitHub's OAuth API. The following `authData` field is required:
|
|
36
|
+
* - `code`
|
|
37
|
+
*
|
|
38
|
+
* ## Insecure Authentication (Not Recommended)
|
|
39
|
+
* Insecure authentication uses the `authData.id` and `authData.access_token` provided by the client. This flow is insecure, deprecated, and poses potential security risks. The following `authData` fields are required:
|
|
40
|
+
* - `id` (**[DEPRECATED]**): The GitHub user ID.
|
|
41
|
+
* - `access_token` (**[DEPRECATED]**): The GitHub access token.
|
|
42
|
+
* To configure Parse Server for insecure authentication, use the following structure:
|
|
43
|
+
* ```json
|
|
44
|
+
* {
|
|
45
|
+
* "auth": {
|
|
46
|
+
* "github": {
|
|
47
|
+
* "enableInsecureAuth": true
|
|
48
|
+
* }
|
|
49
|
+
* }
|
|
50
|
+
* ```
|
|
51
|
+
*
|
|
52
|
+
* ### Deprecation Notice
|
|
53
|
+
* The `enableInsecureAuth` option and insecure `authData` fields (`id`, `access_token`) are deprecated and will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.
|
|
54
|
+
*
|
|
55
|
+
* @example <caption>Secure Authentication Example</caption>
|
|
56
|
+
* // Example authData for secure authentication:
|
|
57
|
+
* const authData = {
|
|
58
|
+
* github: {
|
|
59
|
+
* code: "abc123def456ghi789"
|
|
60
|
+
* }
|
|
61
|
+
* };
|
|
62
|
+
*
|
|
63
|
+
* @example <caption>Insecure Authentication Example (Not Recommended)</caption>
|
|
64
|
+
* // Example authData for insecure authentication:
|
|
65
|
+
* const authData = {
|
|
66
|
+
* github: {
|
|
67
|
+
* id: "1234567",
|
|
68
|
+
* access_token: "abc123def456ghi789" // Deprecated.
|
|
69
|
+
* }
|
|
70
|
+
* };
|
|
71
|
+
*
|
|
72
|
+
* @note `enableInsecureAuth` will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.
|
|
73
|
+
* @note Secure authentication exchanges the `code` provided by the client for an access token using GitHub's OAuth API.
|
|
74
|
+
*
|
|
75
|
+
* @see {@link https://docs.github.com/en/developers/apps/authorizing-oauth-apps GitHub OAuth Documentation}
|
|
76
|
+
*/
|
|
6
77
|
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
78
|
+
class GitHubAdapter extends _BaseCodeAuthAdapter.default {
|
|
79
|
+
constructor() {
|
|
80
|
+
super('GitHub');
|
|
81
|
+
}
|
|
82
|
+
async getAccessTokenFromCode(authData) {
|
|
83
|
+
const tokenUrl = 'https://github.com/login/oauth/access_token';
|
|
84
|
+
const response = await fetch(tokenUrl, {
|
|
85
|
+
method: 'POST',
|
|
86
|
+
headers: {
|
|
87
|
+
'Content-Type': 'application/json',
|
|
88
|
+
Accept: 'application/json'
|
|
89
|
+
},
|
|
90
|
+
body: JSON.stringify({
|
|
91
|
+
client_id: this.clientId,
|
|
92
|
+
client_secret: this.clientSecret,
|
|
93
|
+
code: authData.code
|
|
94
|
+
})
|
|
95
|
+
});
|
|
96
|
+
if (!response.ok) {
|
|
97
|
+
throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to exchange code for token: ${response.statusText}`);
|
|
12
98
|
}
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
function request(path, access_token) {
|
|
24
|
-
return new Promise(function (resolve, reject) {
|
|
25
|
-
https.get({
|
|
26
|
-
host: 'api.github.com',
|
|
27
|
-
path: '/' + path,
|
|
99
|
+
const data = await response.json();
|
|
100
|
+
if (data.error) {
|
|
101
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, data.error_description || data.error);
|
|
102
|
+
}
|
|
103
|
+
return data.access_token;
|
|
104
|
+
}
|
|
105
|
+
async getUserFromAccessToken(accessToken) {
|
|
106
|
+
const userApiUrl = 'https://api.github.com/user';
|
|
107
|
+
const response = await fetch(userApiUrl, {
|
|
108
|
+
method: 'GET',
|
|
28
109
|
headers: {
|
|
29
|
-
|
|
30
|
-
|
|
110
|
+
Authorization: `Bearer ${accessToken}`,
|
|
111
|
+
Accept: 'application/json'
|
|
31
112
|
}
|
|
32
|
-
}, function (res) {
|
|
33
|
-
var data = '';
|
|
34
|
-
res.on('data', function (chunk) {
|
|
35
|
-
data += chunk;
|
|
36
|
-
});
|
|
37
|
-
res.on('end', function () {
|
|
38
|
-
try {
|
|
39
|
-
data = JSON.parse(data);
|
|
40
|
-
} catch (e) {
|
|
41
|
-
return reject(e);
|
|
42
|
-
}
|
|
43
|
-
resolve(data);
|
|
44
|
-
});
|
|
45
|
-
}).on('error', function () {
|
|
46
|
-
reject('Failed to validate this access token with Github.');
|
|
47
113
|
});
|
|
48
|
-
|
|
114
|
+
if (!response.ok) {
|
|
115
|
+
throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to fetch GitHub user: ${response.statusText}`);
|
|
116
|
+
}
|
|
117
|
+
const userData = await response.json();
|
|
118
|
+
if (!userData.id || !userData.login) {
|
|
119
|
+
throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Invalid GitHub user data received.');
|
|
120
|
+
}
|
|
121
|
+
return userData;
|
|
122
|
+
}
|
|
49
123
|
}
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
validateAppId: validateAppId,
|
|
53
|
-
validateAuthData: validateAuthData
|
|
54
|
-
};
|
|
55
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BZGFwdGVycy9BdXRoL2dpdGh1Yi5qcyJdLCJuYW1lcyI6WyJodHRwcyIsInJlcXVpcmUiLCJQYXJzZSIsInZhbGlkYXRlQXV0aERhdGEiLCJhdXRoRGF0YSIsInJlcXVlc3QiLCJhY2Nlc3NfdG9rZW4iLCJ0aGVuIiwiZGF0YSIsImlkIiwiRXJyb3IiLCJPQkpFQ1RfTk9UX0ZPVU5EIiwidmFsaWRhdGVBcHBJZCIsIlByb21pc2UiLCJyZXNvbHZlIiwicGF0aCIsInJlamVjdCIsImdldCIsImhvc3QiLCJoZWFkZXJzIiwicmVzIiwib24iLCJjaHVuayIsIkpTT04iLCJwYXJzZSIsImUiLCJtb2R1bGUiLCJleHBvcnRzIl0sIm1hcHBpbmdzIjoiOztBQUFBO0FBQ0EsSUFBSUEsUUFBUUMsUUFBUSxPQUFSLENBQVo7QUFDQSxJQUFJQyxRQUFRRCxRQUFRLFlBQVIsRUFBc0JDLEtBQWxDOztBQUVBO0FBQ0EsU0FBU0MsZ0JBQVQsQ0FBMEJDLFFBQTFCLEVBQW9DO0FBQ2xDLFNBQU9DLFFBQVEsTUFBUixFQUFnQkQsU0FBU0UsWUFBekIsRUFDSkMsSUFESSxDQUNFQyxJQUFELElBQVU7QUFDZCxRQUFJQSxRQUFRQSxLQUFLQyxFQUFMLElBQVdMLFNBQVNLLEVBQWhDLEVBQW9DO0FBQ2xDO0FBQ0Q7QUFDRCxVQUFNLElBQUlQLE1BQU1RLEtBQVYsQ0FDSlIsTUFBTVEsS0FBTixDQUFZQyxnQkFEUixFQUVKLHVDQUZJLENBQU47QUFHRCxHQVJJLENBQVA7QUFTRDs7QUFFRDtBQUNBLFNBQVNDLGFBQVQsR0FBeUI7QUFDdkIsU0FBT0MsUUFBUUMsT0FBUixFQUFQO0FBQ0Q7O0FBRUQ7QUFDQSxTQUFTVCxPQUFULENBQWlCVSxJQUFqQixFQUF1QlQsWUFBdkIsRUFBcUM7QUFDbkMsU0FBTyxJQUFJTyxPQUFKLENBQVksVUFBU0MsT0FBVCxFQUFrQkUsTUFBbEIsRUFBMEI7QUFDM0NoQixVQUFNaUIsR0FBTixDQUFVO0FBQ1JDLFlBQU0sZ0JBREU7QUFFUkgsWUFBTSxNQUFNQSxJQUZKO0FBR1JJLGVBQVM7QUFDUCx5QkFBaUIsWUFBWWIsWUFEdEI7QUFFUCxzQkFBYztBQUZQO0FBSEQsS0FBVixFQU9HLFVBQVNjLEdBQVQsRUFBYztBQUNmLFVBQUlaLE9BQU8sRUFBWDtBQUNBWSxVQUFJQyxFQUFKLENBQU8sTUFBUCxFQUFlLFVBQVNDLEtBQVQsRUFBZ0I7QUFDN0JkLGdCQUFRYyxLQUFSO0FBQ0QsT0FGRDtBQUdBRixVQUFJQyxFQUFKLENBQU8sS0FBUCxFQUFjLFlBQVc7QUFDdkIsWUFBSTtBQUNGYixpQkFBT2UsS0FBS0MsS0FBTCxDQUFXaEIsSUFBWCxDQUFQO0FBQ0QsU0FGRCxDQUVFLE9BQU1pQixDQUFOLEVBQVM7QUFDVCxpQkFBT1QsT0FBT1MsQ0FBUCxDQUFQO0FBQ0Q7QUFDRFgsZ0JBQVFOLElBQVI7QUFDRCxPQVBEO0FBUUQsS0FwQkQsRUFvQkdhLEVBcEJILENBb0JNLE9BcEJOLEVBb0JlLFlBQVc7QUFDeEJMLGFBQU8sbURBQVA7QUFDRCxLQXRCRDtBQXVCRCxHQXhCTSxDQUFQO0FBeUJEOztBQUVEVSxPQUFPQyxPQUFQLEdBQWlCO0FBQ2ZmLGlCQUFlQSxhQURBO0FBRWZULG9CQUFrQkE7QUFGSCxDQUFqQiIsImZpbGUiOiJnaXRodWIuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBIZWxwZXIgZnVuY3Rpb25zIGZvciBhY2Nlc3NpbmcgdGhlIGdpdGh1YiBBUEkuXG52YXIgaHR0cHMgPSByZXF1aXJlKCdodHRwcycpO1xudmFyIFBhcnNlID0gcmVxdWlyZSgncGFyc2Uvbm9kZScpLlBhcnNlO1xuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIHVzZXIgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUF1dGhEYXRhKGF1dGhEYXRhKSB7XG4gIHJldHVybiByZXF1ZXN0KCd1c2VyJywgYXV0aERhdGEuYWNjZXNzX3Rva2VuKVxuICAgIC50aGVuKChkYXRhKSA9PiB7XG4gICAgICBpZiAoZGF0YSAmJiBkYXRhLmlkID09IGF1dGhEYXRhLmlkKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cbiAgICAgIHRocm93IG5ldyBQYXJzZS5FcnJvcihcbiAgICAgICAgUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCxcbiAgICAgICAgJ0dpdGh1YiBhdXRoIGlzIGludmFsaWQgZm9yIHRoaXMgdXNlci4nKTtcbiAgICB9KTtcbn1cblxuLy8gUmV0dXJucyBhIHByb21pc2UgdGhhdCBmdWxmaWxscyBpZmYgdGhpcyBhcHAgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUFwcElkKCkge1xuICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCk7XG59XG5cbi8vIEEgcHJvbWlzZXkgd3JhcHBlciBmb3IgYXBpIHJlcXVlc3RzXG5mdW5jdGlvbiByZXF1ZXN0KHBhdGgsIGFjY2Vzc190b2tlbikge1xuICByZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24ocmVzb2x2ZSwgcmVqZWN0KSB7XG4gICAgaHR0cHMuZ2V0KHtcbiAgICAgIGhvc3Q6ICdhcGkuZ2l0aHViLmNvbScsXG4gICAgICBwYXRoOiAnLycgKyBwYXRoLFxuICAgICAgaGVhZGVyczoge1xuICAgICAgICAnQXV0aG9yaXphdGlvbic6ICdiZWFyZXIgJyArIGFjY2Vzc190b2tlbixcbiAgICAgICAgJ1VzZXItQWdlbnQnOiAncGFyc2Utc2VydmVyJ1xuICAgICAgfVxuICAgIH0sIGZ1bmN0aW9uKHJlcykge1xuICAgICAgdmFyIGRhdGEgPSAnJztcbiAgICAgIHJlcy5vbignZGF0YScsIGZ1bmN0aW9uKGNodW5rKSB7XG4gICAgICAgIGRhdGEgKz0gY2h1bms7XG4gICAgICB9KTtcbiAgICAgIHJlcy5vbignZW5kJywgZnVuY3Rpb24oKSB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgZGF0YSA9IEpTT04ucGFyc2UoZGF0YSk7XG4gICAgICAgIH0gY2F0Y2goZSkge1xuICAgICAgICAgIHJldHVybiByZWplY3QoZSk7XG4gICAgICAgIH1cbiAgICAgICAgcmVzb2x2ZShkYXRhKTtcbiAgICAgIH0pO1xuICAgIH0pLm9uKCdlcnJvcicsIGZ1bmN0aW9uKCkge1xuICAgICAgcmVqZWN0KCdGYWlsZWQgdG8gdmFsaWRhdGUgdGhpcyBhY2Nlc3MgdG9rZW4gd2l0aCBHaXRodWIuJyk7XG4gICAgfSk7XG4gIH0pO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgdmFsaWRhdGVBcHBJZDogdmFsaWRhdGVBcHBJZCxcbiAgdmFsaWRhdGVBdXRoRGF0YTogdmFsaWRhdGVBdXRoRGF0YVxufTtcbiJdfQ==
|
|
124
|
+
var _default = exports.default = new GitHubAdapter();
|
|
125
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_BaseCodeAuthAdapter","_interopRequireDefault","require","e","__esModule","default","GitHubAdapter","BaseCodeAuthAdapter","constructor","getAccessTokenFromCode","authData","tokenUrl","response","fetch","method","headers","Accept","body","JSON","stringify","client_id","clientId","client_secret","clientSecret","code","ok","Parse","Error","VALIDATION_ERROR","statusText","data","json","error","OBJECT_NOT_FOUND","error_description","access_token","getUserFromAccessToken","accessToken","userApiUrl","Authorization","userData","id","login","_default","exports"],"sources":["../../../src/Adapters/Auth/github.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for GitHub.\n * @class GitHubAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.clientId - The GitHub App Client ID. Required for secure authentication.\n * @param {string} options.clientSecret - The GitHub App Client Secret. Required for secure authentication.\n * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {string} authData.code - The authorization code from GitHub. Required for secure authentication.\n * @param {string} [authData.id] - **[DEPRECATED]** The GitHub user ID (required for insecure authentication).\n * @param {string} [authData.access_token] - **[DEPRECATED]** The GitHub access token (required for insecure authentication).\n *\n * @description\n * ## Parse Server Configuration\n * * To configure Parse Server for GitHub authentication, use the following structure:\n * ```json\n * {\n *  \"auth\": {\n *   \"github\": {\n *     \"clientId\": \"12345\",\n *     \"clientSecret\": \"abcde\"\n *   }\n * }\n * ```\n *\n * The GitHub adapter exchanges the `authData.code` provided by the client for an access token using GitHub's OAuth API. The following `authData` field is required:\n * - `code`\n *\n * ## Insecure Authentication (Not Recommended)\n * Insecure authentication uses the `authData.id` and `authData.access_token` provided by the client. This flow is insecure, deprecated, and poses potential security risks. The following `authData` fields are required:\n * - `id` (**[DEPRECATED]**): The GitHub user ID.\n * - `access_token` (**[DEPRECATED]**): The GitHub access token.\n * To configure Parse Server for insecure authentication, use the following structure:\n * ```json\n * {\n *  \"auth\": {\n *    \"github\": {\n *    \"enableInsecureAuth\": true\n *  }\n * }\n * ```\n *\n * ### Deprecation Notice\n * The `enableInsecureAuth` option and insecure `authData` fields (`id`, `access_token`) are deprecated and will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.\n *\n * @example <caption>Secure Authentication Example</caption>\n * // Example authData for secure authentication:\n * const authData = {\n *   github: {\n *     code: \"abc123def456ghi789\"\n *   }\n * };\n *\n * @example <caption>Insecure Authentication Example (Not Recommended)</caption>\n * // Example authData for insecure authentication:\n * const authData = {\n *   github: {\n *     id: \"1234567\",\n *     access_token: \"abc123def456ghi789\" // Deprecated.\n *   }\n * };\n *\n * @note `enableInsecureAuth` will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.\n * @note Secure authentication exchanges the `code` provided by the client for an access token using GitHub's OAuth API.\n *\n * @see {@link https://docs.github.com/en/developers/apps/authorizing-oauth-apps GitHub OAuth Documentation}\n */\n\nimport BaseCodeAuthAdapter from './BaseCodeAuthAdapter';\nclass GitHubAdapter extends BaseCodeAuthAdapter {\n  constructor() {\n    super('GitHub');\n  }\n  async getAccessTokenFromCode(authData) {\n    const tokenUrl = 'https://github.com/login/oauth/access_token';\n    const response = await fetch(tokenUrl, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        Accept: 'application/json',\n      },\n      body: JSON.stringify({\n        client_id: this.clientId,\n        client_secret: this.clientSecret,\n        code: authData.code,\n      }),\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to exchange code for token: ${response.statusText}`);\n    }\n\n    const data = await response.json();\n    if (data.error) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, data.error_description || data.error);\n    }\n\n    return data.access_token;\n  }\n\n  async getUserFromAccessToken(accessToken) {\n    const userApiUrl = 'https://api.github.com/user';\n    const response = await fetch(userApiUrl, {\n      method: 'GET',\n      headers: {\n        Authorization: `Bearer ${accessToken}`,\n        Accept: 'application/json',\n      },\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to fetch GitHub user: ${response.statusText}`);\n    }\n\n    const userData = await response.json();\n    if (!userData.id || !userData.login) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Invalid GitHub user data received.');\n    }\n\n    return userData;\n  }\n\n}\n\nexport default new GitHubAdapter();\n\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,aAAa,SAASC,4BAAmB,CAAC;EAC9CC,WAAWA,CAAA,EAAG;IACZ,KAAK,CAAC,QAAQ,CAAC;EACjB;EACA,MAAMC,sBAAsBA,CAACC,QAAQ,EAAE;IACrC,MAAMC,QAAQ,GAAG,6CAA6C;IAC9D,MAAMC,QAAQ,GAAG,MAAMC,KAAK,CAACF,QAAQ,EAAE;MACrCG,MAAM,EAAE,MAAM;MACdC,OAAO,EAAE;QACP,cAAc,EAAE,kBAAkB;QAClCC,MAAM,EAAE;MACV,CAAC;MACDC,IAAI,EAAEC,IAAI,CAACC,SAAS,CAAC;QACnBC,SAAS,EAAE,IAAI,CAACC,QAAQ;QACxBC,aAAa,EAAE,IAAI,CAACC,YAAY;QAChCC,IAAI,EAAEd,QAAQ,CAACc;MACjB,CAAC;IACH,CAAC,CAAC;IAEF,IAAI,CAACZ,QAAQ,CAACa,EAAE,EAAE;MAChB,MAAM,IAAIC,KAAK,CAACC,KAAK,CAACD,KAAK,CAACC,KAAK,CAACC,gBAAgB,EAAE,sCAAsChB,QAAQ,CAACiB,UAAU,EAAE,CAAC;IAClH;IAEA,MAAMC,IAAI,GAAG,MAAMlB,QAAQ,CAACmB,IAAI,CAAC,CAAC;IAClC,IAAID,IAAI,CAACE,KAAK,EAAE;MACd,MAAM,IAAIN,KAAK,CAACC,KAAK,CAACD,KAAK,CAACC,KAAK,CAACM,gBAAgB,EAAEH,IAAI,CAACI,iBAAiB,IAAIJ,IAAI,CAACE,KAAK,CAAC;IAC3F;IAEA,OAAOF,IAAI,CAACK,YAAY;EAC1B;EAEA,MAAMC,sBAAsBA,CAACC,WAAW,EAAE;IACxC,MAAMC,UAAU,GAAG,6BAA6B;IAChD,MAAM1B,QAAQ,GAAG,MAAMC,KAAK,CAACyB,UAAU,EAAE;MACvCxB,MAAM,EAAE,KAAK;MACbC,OAAO,EAAE;QACPwB,aAAa,EAAE,UAAUF,WAAW,EAAE;QACtCrB,MAAM,EAAE;MACV;IACF,CAAC,CAAC;IAEF,IAAI,CAACJ,QAAQ,CAACa,EAAE,EAAE;MAChB,MAAM,IAAIC,KAAK,CAACC,KAAK,CAACD,KAAK,CAACC,KAAK,CAACC,gBAAgB,EAAE,gCAAgChB,QAAQ,CAACiB,UAAU,EAAE,CAAC;IAC5G;IAEA,MAAMW,QAAQ,GAAG,MAAM5B,QAAQ,CAACmB,IAAI,CAAC,CAAC;IACtC,IAAI,CAACS,QAAQ,CAACC,EAAE,IAAI,CAACD,QAAQ,CAACE,KAAK,EAAE;MACnC,MAAM,IAAIhB,KAAK,CAACC,KAAK,CAACD,KAAK,CAACC,KAAK,CAACC,gBAAgB,EAAE,oCAAoC,CAAC;IAC3F;IAEA,OAAOY,QAAQ;EACjB;AAEF;AAAC,IAAAG,QAAA,GAAAC,OAAA,CAAAvC,OAAA,GAEc,IAAIC,aAAa,CAAC,CAAC","ignoreList":[]}
|
|
@@ -1,71 +1,180 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Parse Server authentication adapter for Google.
|
|
3
|
+
*
|
|
4
|
+
* @class GoogleAdapter
|
|
5
|
+
* @param {Object} options - The adapter configuration options.
|
|
6
|
+
* @param {string} options.clientId - Your Google application Client ID. Required for authentication.
|
|
7
|
+
*
|
|
8
|
+
* @description
|
|
9
|
+
* ## Parse Server Configuration
|
|
10
|
+
* To configure Parse Server for Google authentication, use the following structure:
|
|
11
|
+
* ```json
|
|
12
|
+
* {
|
|
13
|
+
* "auth": {
|
|
14
|
+
* "google": {
|
|
15
|
+
* "clientId": "your-client-id"
|
|
16
|
+
* }
|
|
17
|
+
* }
|
|
18
|
+
* }
|
|
19
|
+
* ```
|
|
20
|
+
*
|
|
21
|
+
* The adapter requires the following `authData` fields:
|
|
22
|
+
* - **id**: The Google user ID.
|
|
23
|
+
* - **id_token**: The Google ID token.
|
|
24
|
+
* - **access_token**: The Google access token.
|
|
25
|
+
*
|
|
26
|
+
* ## Auth Payload
|
|
27
|
+
* ### Example Auth Data Payload
|
|
28
|
+
* ```json
|
|
29
|
+
* {
|
|
30
|
+
* "google": {
|
|
31
|
+
* "id": "1234567",
|
|
32
|
+
* "id_token": "xxxxx.yyyyy.zzzzz",
|
|
33
|
+
* "access_token": "abc123def456ghi789"
|
|
34
|
+
* }
|
|
35
|
+
* }
|
|
36
|
+
* ```
|
|
37
|
+
*
|
|
38
|
+
* ## Notes
|
|
39
|
+
* - Ensure your Google Client ID is configured properly in the Parse Server configuration.
|
|
40
|
+
* - The `id_token` and `access_token` are validated against Google's authentication services.
|
|
41
|
+
*
|
|
42
|
+
* @see {@link https://developers.google.com/identity/sign-in/web/backend-auth Google Authentication Documentation}
|
|
43
|
+
*/
|
|
44
|
+
|
|
1
45
|
'use strict';
|
|
2
46
|
|
|
3
47
|
// Helper functions for accessing the google API.
|
|
4
|
-
var https = require('https');
|
|
5
48
|
var Parse = require('parse/node').Parse;
|
|
49
|
+
const https = require('https');
|
|
50
|
+
const jwt = require('jsonwebtoken');
|
|
51
|
+
const authUtils = require('./utils');
|
|
52
|
+
const TOKEN_ISSUER = 'accounts.google.com';
|
|
53
|
+
const HTTPS_TOKEN_ISSUER = 'https://accounts.google.com';
|
|
54
|
+
let cache = {};
|
|
6
55
|
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
56
|
+
// Retrieve Google Signin Keys (with cache control)
|
|
57
|
+
function getGoogleKeyByKeyId(keyId) {
|
|
58
|
+
if (cache[keyId] && cache.expiresAt > new Date()) {
|
|
59
|
+
return cache[keyId];
|
|
60
|
+
}
|
|
61
|
+
return new Promise((resolve, reject) => {
|
|
62
|
+
https.get(`https://www.googleapis.com/oauth2/v3/certs`, res => {
|
|
63
|
+
let data = '';
|
|
64
|
+
res.on('data', chunk => {
|
|
65
|
+
data += chunk.toString('utf8');
|
|
66
|
+
});
|
|
67
|
+
res.on('end', () => {
|
|
68
|
+
const {
|
|
69
|
+
keys
|
|
70
|
+
} = JSON.parse(data);
|
|
71
|
+
const pems = keys.reduce((pems, {
|
|
72
|
+
n: modulus,
|
|
73
|
+
e: exposant,
|
|
74
|
+
kid
|
|
75
|
+
}) => Object.assign(pems, {
|
|
76
|
+
[kid]: rsaPublicKeyToPEM(modulus, exposant)
|
|
77
|
+
}), {});
|
|
78
|
+
if (res.headers['cache-control']) {
|
|
79
|
+
var expire = res.headers['cache-control'].match(/max-age=([0-9]+)/);
|
|
80
|
+
if (expire) {
|
|
81
|
+
cache = Object.assign({}, pems, {
|
|
82
|
+
expiresAt: new Date(new Date().getTime() + Number(expire[1]) * 1000)
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
resolve(pems[keyId]);
|
|
87
|
+
});
|
|
88
|
+
}).on('error', reject);
|
|
13
89
|
});
|
|
14
90
|
}
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
91
|
+
async function verifyIdToken({
|
|
92
|
+
id_token: token,
|
|
93
|
+
id
|
|
94
|
+
}, {
|
|
95
|
+
clientId
|
|
96
|
+
}) {
|
|
97
|
+
if (!token) {
|
|
98
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `id token is invalid for this user.`);
|
|
99
|
+
}
|
|
100
|
+
const {
|
|
101
|
+
kid: keyId,
|
|
102
|
+
alg: algorithm
|
|
103
|
+
} = authUtils.getHeaderFromToken(token);
|
|
104
|
+
let jwtClaims;
|
|
105
|
+
const googleKey = await getGoogleKeyByKeyId(keyId);
|
|
106
|
+
try {
|
|
107
|
+
jwtClaims = jwt.verify(token, googleKey, {
|
|
108
|
+
algorithms: algorithm,
|
|
109
|
+
audience: clientId
|
|
110
|
+
});
|
|
111
|
+
} catch (exception) {
|
|
112
|
+
const message = exception.message;
|
|
113
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);
|
|
114
|
+
}
|
|
115
|
+
if (jwtClaims.iss !== TOKEN_ISSUER && jwtClaims.iss !== HTTPS_TOKEN_ISSUER) {
|
|
116
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `id token not issued by correct provider - expected: ${TOKEN_ISSUER} or ${HTTPS_TOKEN_ISSUER} | from: ${jwtClaims.iss}`);
|
|
117
|
+
}
|
|
118
|
+
if (jwtClaims.sub !== id) {
|
|
119
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `auth data is invalid for this user.`);
|
|
120
|
+
}
|
|
121
|
+
if (clientId && jwtClaims.aud !== clientId) {
|
|
122
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `id token not authorized for this clientId.`);
|
|
123
|
+
}
|
|
124
|
+
return jwtClaims;
|
|
23
125
|
}
|
|
24
126
|
|
|
25
127
|
// Returns a promise that fulfills if this user id is valid.
|
|
26
|
-
function validateAuthData(authData) {
|
|
27
|
-
|
|
28
|
-
return validateIdToken(authData.id, authData.id_token);
|
|
29
|
-
} else {
|
|
30
|
-
return validateAuthToken(authData.id, authData.access_token).then(() => {
|
|
31
|
-
// Validation with auth token worked
|
|
32
|
-
return;
|
|
33
|
-
}, () => {
|
|
34
|
-
// Try with the id_token param
|
|
35
|
-
return validateIdToken(authData.id, authData.access_token);
|
|
36
|
-
});
|
|
37
|
-
}
|
|
128
|
+
function validateAuthData(authData, options = {}) {
|
|
129
|
+
return verifyIdToken(authData, options);
|
|
38
130
|
}
|
|
39
131
|
|
|
40
132
|
// Returns a promise that fulfills if this app id is valid.
|
|
41
133
|
function validateAppId() {
|
|
42
134
|
return Promise.resolve();
|
|
43
135
|
}
|
|
44
|
-
|
|
45
|
-
// A promisey wrapper for api requests
|
|
46
|
-
function request(path) {
|
|
47
|
-
return new Promise(function (resolve, reject) {
|
|
48
|
-
https.get("https://www.googleapis.com/oauth2/v3/" + path, function (res) {
|
|
49
|
-
var data = '';
|
|
50
|
-
res.on('data', function (chunk) {
|
|
51
|
-
data += chunk;
|
|
52
|
-
});
|
|
53
|
-
res.on('end', function () {
|
|
54
|
-
try {
|
|
55
|
-
data = JSON.parse(data);
|
|
56
|
-
} catch (e) {
|
|
57
|
-
return reject(e);
|
|
58
|
-
}
|
|
59
|
-
resolve(data);
|
|
60
|
-
});
|
|
61
|
-
}).on('error', function () {
|
|
62
|
-
reject('Failed to validate this access token with Google.');
|
|
63
|
-
});
|
|
64
|
-
});
|
|
65
|
-
}
|
|
66
|
-
|
|
67
136
|
module.exports = {
|
|
68
137
|
validateAppId: validateAppId,
|
|
69
138
|
validateAuthData: validateAuthData
|
|
70
139
|
};
|
|
71
|
-
|
|
140
|
+
|
|
141
|
+
// Helpers functions to convert the RSA certs to PEM (from jwks-rsa)
|
|
142
|
+
function rsaPublicKeyToPEM(modulusB64, exponentB64) {
|
|
143
|
+
const modulus = new Buffer(modulusB64, 'base64');
|
|
144
|
+
const exponent = new Buffer(exponentB64, 'base64');
|
|
145
|
+
const modulusHex = prepadSigned(modulus.toString('hex'));
|
|
146
|
+
const exponentHex = prepadSigned(exponent.toString('hex'));
|
|
147
|
+
const modlen = modulusHex.length / 2;
|
|
148
|
+
const explen = exponentHex.length / 2;
|
|
149
|
+
const encodedModlen = encodeLengthHex(modlen);
|
|
150
|
+
const encodedExplen = encodeLengthHex(explen);
|
|
151
|
+
const encodedPubkey = '30' + encodeLengthHex(modlen + explen + encodedModlen.length / 2 + encodedExplen.length / 2 + 2) + '02' + encodedModlen + modulusHex + '02' + encodedExplen + exponentHex;
|
|
152
|
+
const der = new Buffer(encodedPubkey, 'hex').toString('base64');
|
|
153
|
+
let pem = '-----BEGIN RSA PUBLIC KEY-----\n';
|
|
154
|
+
pem += `${der.match(/.{1,64}/g).join('\n')}`;
|
|
155
|
+
pem += '\n-----END RSA PUBLIC KEY-----\n';
|
|
156
|
+
return pem;
|
|
157
|
+
}
|
|
158
|
+
function prepadSigned(hexStr) {
|
|
159
|
+
const msb = hexStr[0];
|
|
160
|
+
if (msb < '0' || msb > '7') {
|
|
161
|
+
return `00${hexStr}`;
|
|
162
|
+
}
|
|
163
|
+
return hexStr;
|
|
164
|
+
}
|
|
165
|
+
function toHex(number) {
|
|
166
|
+
const nstr = number.toString(16);
|
|
167
|
+
if (nstr.length % 2) {
|
|
168
|
+
return `0${nstr}`;
|
|
169
|
+
}
|
|
170
|
+
return nstr;
|
|
171
|
+
}
|
|
172
|
+
function encodeLengthHex(n) {
|
|
173
|
+
if (n <= 127) {
|
|
174
|
+
return toHex(n);
|
|
175
|
+
}
|
|
176
|
+
const nHex = toHex(n);
|
|
177
|
+
const lengthOfLengthByte = 128 + nHex.length / 2;
|
|
178
|
+
return toHex(lengthOfLengthByte) + nHex;
|
|
179
|
+
}
|
|
180
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","https","jwt","authUtils","TOKEN_ISSUER","HTTPS_TOKEN_ISSUER","cache","getGoogleKeyByKeyId","keyId","expiresAt","Date","Promise","resolve","reject","get","res","data","on","chunk","toString","keys","JSON","parse","pems","reduce","n","modulus","e","exposant","kid","Object","assign","rsaPublicKeyToPEM","headers","expire","match","getTime","Number","verifyIdToken","id_token","token","id","clientId","Error","OBJECT_NOT_FOUND","alg","algorithm","getHeaderFromToken","jwtClaims","googleKey","verify","algorithms","audience","exception","message","iss","sub","aud","validateAuthData","authData","options","validateAppId","module","exports","modulusB64","exponentB64","Buffer","exponent","modulusHex","prepadSigned","exponentHex","modlen","length","explen","encodedModlen","encodeLengthHex","encodedExplen","encodedPubkey","der","pem","join","hexStr","msb","toHex","number","nstr","nHex","lengthOfLengthByte"],"sources":["../../../src/Adapters/Auth/google.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Google.\n *\n * @class GoogleAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.clientId - Your Google application Client ID. Required for authentication.\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for Google authentication, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *     \"google\": {\n *       \"clientId\": \"your-client-id\"\n *     }\n *   }\n * }\n * ```\n *\n * The adapter requires the following `authData` fields:\n * - **id**: The Google user ID.\n * - **id_token**: The Google ID token.\n * - **access_token**: The Google access token.\n *\n * ## Auth Payload\n * ### Example Auth Data Payload\n * ```json\n * {\n *   \"google\": {\n *     \"id\": \"1234567\",\n *     \"id_token\": \"xxxxx.yyyyy.zzzzz\",\n *     \"access_token\": \"abc123def456ghi789\"\n *   }\n * }\n * ```\n *\n * ## Notes\n * - Ensure your Google Client ID is configured properly in the Parse Server configuration.\n * - The `id_token` and `access_token` are validated against Google's authentication services.\n *\n * @see {@link https://developers.google.com/identity/sign-in/web/backend-auth Google Authentication Documentation}\n */\n\n'use strict';\n\n// Helper functions for accessing the google API.\nvar Parse = require('parse/node').Parse;\n\nconst https = require('https');\nconst jwt = require('jsonwebtoken');\nconst authUtils = require('./utils');\n\nconst TOKEN_ISSUER = 'accounts.google.com';\nconst HTTPS_TOKEN_ISSUER = 'https://accounts.google.com';\n\nlet cache = {};\n\n// Retrieve Google Signin Keys (with cache control)\nfunction getGoogleKeyByKeyId(keyId) {\n  if (cache[keyId] && cache.expiresAt > new Date()) {\n    return cache[keyId];\n  }\n\n  return new Promise((resolve, reject) => {\n    https\n      .get(`https://www.googleapis.com/oauth2/v3/certs`, res => {\n        let data = '';\n        res.on('data', chunk => {\n          data += chunk.toString('utf8');\n        });\n        res.on('end', () => {\n          const { keys } = JSON.parse(data);\n          const pems = keys.reduce(\n            (pems, { n: modulus, e: exposant, kid }) =>\n              Object.assign(pems, {\n                [kid]: rsaPublicKeyToPEM(modulus, exposant),\n              }),\n            {}\n          );\n\n          if (res.headers['cache-control']) {\n            var expire = res.headers['cache-control'].match(/max-age=([0-9]+)/);\n\n            if (expire) {\n              cache = Object.assign({}, pems, {\n                expiresAt: new Date(new Date().getTime() + Number(expire[1]) * 1000),\n              });\n            }\n          }\n\n          resolve(pems[keyId]);\n        });\n      })\n      .on('error', reject);\n  });\n}\n\nasync function verifyIdToken({ id_token: token, id }, { clientId }) {\n  if (!token) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `id token is invalid for this user.`);\n  }\n\n  const { kid: keyId, alg: algorithm } = authUtils.getHeaderFromToken(token);\n  let jwtClaims;\n  const googleKey = await getGoogleKeyByKeyId(keyId);\n\n  try {\n    jwtClaims = jwt.verify(token, googleKey, {\n      algorithms: algorithm,\n      audience: clientId,\n    });\n  } catch (exception) {\n    const message = exception.message;\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);\n  }\n\n  if (jwtClaims.iss !== TOKEN_ISSUER && jwtClaims.iss !== HTTPS_TOKEN_ISSUER) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `id token not issued by correct provider - expected: ${TOKEN_ISSUER} or ${HTTPS_TOKEN_ISSUER} | from: ${jwtClaims.iss}`\n    );\n  }\n\n  if (jwtClaims.sub !== id) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `auth data is invalid for this user.`);\n  }\n\n  if (clientId && jwtClaims.aud !== clientId) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `id token not authorized for this clientId.`\n    );\n  }\n\n  return jwtClaims;\n}\n\n// Returns a promise that fulfills if this user id is valid.\nfunction validateAuthData(authData, options = {}) {\n  return verifyIdToken(authData, options);\n}\n\n// Returns a promise that fulfills if this app id is valid.\nfunction validateAppId() {\n  return Promise.resolve();\n}\n\nmodule.exports = {\n  validateAppId: validateAppId,\n  validateAuthData: validateAuthData,\n};\n\n// Helpers functions to convert the RSA certs to PEM (from jwks-rsa)\nfunction rsaPublicKeyToPEM(modulusB64, exponentB64) {\n  const modulus = new Buffer(modulusB64, 'base64');\n  const exponent = new Buffer(exponentB64, 'base64');\n  const modulusHex = prepadSigned(modulus.toString('hex'));\n  const exponentHex = prepadSigned(exponent.toString('hex'));\n  const modlen = modulusHex.length / 2;\n  const explen = exponentHex.length / 2;\n\n  const encodedModlen = encodeLengthHex(modlen);\n  const encodedExplen = encodeLengthHex(explen);\n  const encodedPubkey =\n    '30' +\n    encodeLengthHex(modlen + explen + encodedModlen.length / 2 + encodedExplen.length / 2 + 2) +\n    '02' +\n    encodedModlen +\n    modulusHex +\n    '02' +\n    encodedExplen +\n    exponentHex;\n\n  const der = new Buffer(encodedPubkey, 'hex').toString('base64');\n\n  let pem = '-----BEGIN RSA PUBLIC KEY-----\\n';\n  pem += `${der.match(/.{1,64}/g).join('\\n')}`;\n  pem += '\\n-----END RSA PUBLIC KEY-----\\n';\n  return pem;\n}\n\nfunction prepadSigned(hexStr) {\n  const msb = hexStr[0];\n  if (msb < '0' || msb > '7') {\n    return `00${hexStr}`;\n  }\n  return hexStr;\n}\n\nfunction toHex(number) {\n  const nstr = number.toString(16);\n  if (nstr.length % 2) {\n    return `0${nstr}`;\n  }\n  return nstr;\n}\n\nfunction encodeLengthHex(n) {\n  if (n <= 127) {\n    return toHex(n);\n  }\n  const nHex = toHex(n);\n  const lengthOfLengthByte = 128 + nHex.length / 2;\n  return toHex(lengthOfLengthByte) + nHex;\n}\n"],"mappings":"AAAA;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;;AAEA,YAAY;;AAEZ;AACA,IAAIA,KAAK,GAAGC,OAAO,CAAC,YAAY,CAAC,CAACD,KAAK;AAEvC,MAAME,KAAK,GAAGD,OAAO,CAAC,OAAO,CAAC;AAC9B,MAAME,GAAG,GAAGF,OAAO,CAAC,cAAc,CAAC;AACnC,MAAMG,SAAS,GAAGH,OAAO,CAAC,SAAS,CAAC;AAEpC,MAAMI,YAAY,GAAG,qBAAqB;AAC1C,MAAMC,kBAAkB,GAAG,6BAA6B;AAExD,IAAIC,KAAK,GAAG,CAAC,CAAC;;AAEd;AACA,SAASC,mBAAmBA,CAACC,KAAK,EAAE;EAClC,IAAIF,KAAK,CAACE,KAAK,CAAC,IAAIF,KAAK,CAACG,SAAS,GAAG,IAAIC,IAAI,CAAC,CAAC,EAAE;IAChD,OAAOJ,KAAK,CAACE,KAAK,CAAC;EACrB;EAEA,OAAO,IAAIG,OAAO,CAAC,CAACC,OAAO,EAAEC,MAAM,KAAK;IACtCZ,KAAK,CACFa,GAAG,CAAC,4CAA4C,EAAEC,GAAG,IAAI;MACxD,IAAIC,IAAI,GAAG,EAAE;MACbD,GAAG,CAACE,EAAE,CAAC,MAAM,EAAEC,KAAK,IAAI;QACtBF,IAAI,IAAIE,KAAK,CAACC,QAAQ,CAAC,MAAM,CAAC;MAChC,CAAC,CAAC;MACFJ,GAAG,CAACE,EAAE,CAAC,KAAK,EAAE,MAAM;QAClB,MAAM;UAAEG;QAAK,CAAC,GAAGC,IAAI,CAACC,KAAK,CAACN,IAAI,CAAC;QACjC,MAAMO,IAAI,GAAGH,IAAI,CAACI,MAAM,CACtB,CAACD,IAAI,EAAE;UAAEE,CAAC,EAAEC,OAAO;UAAEC,CAAC,EAAEC,QAAQ;UAAEC;QAAI,CAAC,KACrCC,MAAM,CAACC,MAAM,CAACR,IAAI,EAAE;UAClB,CAACM,GAAG,GAAGG,iBAAiB,CAACN,OAAO,EAAEE,QAAQ;QAC5C,CAAC,CAAC,EACJ,CAAC,CACH,CAAC;QAED,IAAIb,GAAG,CAACkB,OAAO,CAAC,eAAe,CAAC,EAAE;UAChC,IAAIC,MAAM,GAAGnB,GAAG,CAACkB,OAAO,CAAC,eAAe,CAAC,CAACE,KAAK,CAAC,kBAAkB,CAAC;UAEnE,IAAID,MAAM,EAAE;YACV5B,KAAK,GAAGwB,MAAM,CAACC,MAAM,CAAC,CAAC,CAAC,EAAER,IAAI,EAAE;cAC9Bd,SAAS,EAAE,IAAIC,IAAI,CAAC,IAAIA,IAAI,CAAC,CAAC,CAAC0B,OAAO,CAAC,CAAC,GAAGC,MAAM,CAACH,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI;YACrE,CAAC,CAAC;UACJ;QACF;QAEAtB,OAAO,CAACW,IAAI,CAACf,KAAK,CAAC,CAAC;MACtB,CAAC,CAAC;IACJ,CAAC,CAAC,CACDS,EAAE,CAAC,OAAO,EAAEJ,MAAM,CAAC;EACxB,CAAC,CAAC;AACJ;AAEA,eAAeyB,aAAaA,CAAC;EAAEC,QAAQ,EAAEC,KAAK;EAAEC;AAAG,CAAC,EAAE;EAAEC;AAAS,CAAC,EAAE;EAClE,IAAI,CAACF,KAAK,EAAE;IACV,MAAM,IAAIzC,KAAK,CAAC4C,KAAK,CAAC5C,KAAK,CAAC4C,KAAK,CAACC,gBAAgB,EAAE,oCAAoC,CAAC;EAC3F;EAEA,MAAM;IAAEf,GAAG,EAAErB,KAAK;IAAEqC,GAAG,EAAEC;EAAU,CAAC,GAAG3C,SAAS,CAAC4C,kBAAkB,CAACP,KAAK,CAAC;EAC1E,IAAIQ,SAAS;EACb,MAAMC,SAAS,GAAG,MAAM1C,mBAAmB,CAACC,KAAK,CAAC;EAElD,IAAI;IACFwC,SAAS,GAAG9C,GAAG,CAACgD,MAAM,CAACV,KAAK,EAAES,SAAS,EAAE;MACvCE,UAAU,EAAEL,SAAS;MACrBM,QAAQ,EAAEV;IACZ,CAAC,CAAC;EACJ,CAAC,CAAC,OAAOW,SAAS,EAAE;IAClB,MAAMC,OAAO,GAAGD,SAAS,CAACC,OAAO;IACjC,MAAM,IAAIvD,KAAK,CAAC4C,KAAK,CAAC5C,KAAK,CAAC4C,KAAK,CAACC,gBAAgB,EAAE,GAAGU,OAAO,EAAE,CAAC;EACnE;EAEA,IAAIN,SAAS,CAACO,GAAG,KAAKnD,YAAY,IAAI4C,SAAS,CAACO,GAAG,KAAKlD,kBAAkB,EAAE;IAC1E,MAAM,IAAIN,KAAK,CAAC4C,KAAK,CACnB5C,KAAK,CAAC4C,KAAK,CAACC,gBAAgB,EAC5B,uDAAuDxC,YAAY,OAAOC,kBAAkB,YAAY2C,SAAS,CAACO,GAAG,EACvH,CAAC;EACH;EAEA,IAAIP,SAAS,CAACQ,GAAG,KAAKf,EAAE,EAAE;IACxB,MAAM,IAAI1C,KAAK,CAAC4C,KAAK,CAAC5C,KAAK,CAAC4C,KAAK,CAACC,gBAAgB,EAAE,qCAAqC,CAAC;EAC5F;EAEA,IAAIF,QAAQ,IAAIM,SAAS,CAACS,GAAG,KAAKf,QAAQ,EAAE;IAC1C,MAAM,IAAI3C,KAAK,CAAC4C,KAAK,CACnB5C,KAAK,CAAC4C,KAAK,CAACC,gBAAgB,EAC5B,4CACF,CAAC;EACH;EAEA,OAAOI,SAAS;AAClB;;AAEA;AACA,SAASU,gBAAgBA,CAACC,QAAQ,EAAEC,OAAO,GAAG,CAAC,CAAC,EAAE;EAChD,OAAOtB,aAAa,CAACqB,QAAQ,EAAEC,OAAO,CAAC;AACzC;;AAEA;AACA,SAASC,aAAaA,CAAA,EAAG;EACvB,OAAOlD,OAAO,CAACC,OAAO,CAAC,CAAC;AAC1B;AAEAkD,MAAM,CAACC,OAAO,GAAG;EACfF,aAAa,EAAEA,aAAa;EAC5BH,gBAAgB,EAAEA;AACpB,CAAC;;AAED;AACA,SAAS1B,iBAAiBA,CAACgC,UAAU,EAAEC,WAAW,EAAE;EAClD,MAAMvC,OAAO,GAAG,IAAIwC,MAAM,CAACF,UAAU,EAAE,QAAQ,CAAC;EAChD,MAAMG,QAAQ,GAAG,IAAID,MAAM,CAACD,WAAW,EAAE,QAAQ,CAAC;EAClD,MAAMG,UAAU,GAAGC,YAAY,CAAC3C,OAAO,CAACP,QAAQ,CAAC,KAAK,CAAC,CAAC;EACxD,MAAMmD,WAAW,GAAGD,YAAY,CAACF,QAAQ,CAAChD,QAAQ,CAAC,KAAK,CAAC,CAAC;EAC1D,MAAMoD,MAAM,GAAGH,UAAU,CAACI,MAAM,GAAG,CAAC;EACpC,MAAMC,MAAM,GAAGH,WAAW,CAACE,MAAM,GAAG,CAAC;EAErC,MAAME,aAAa,GAAGC,eAAe,CAACJ,MAAM,CAAC;EAC7C,MAAMK,aAAa,GAAGD,eAAe,CAACF,MAAM,CAAC;EAC7C,MAAMI,aAAa,GACjB,IAAI,GACJF,eAAe,CAACJ,MAAM,GAAGE,MAAM,GAAGC,aAAa,CAACF,MAAM,GAAG,CAAC,GAAGI,aAAa,CAACJ,MAAM,GAAG,CAAC,GAAG,CAAC,CAAC,GAC1F,IAAI,GACJE,aAAa,GACbN,UAAU,GACV,IAAI,GACJQ,aAAa,GACbN,WAAW;EAEb,MAAMQ,GAAG,GAAG,IAAIZ,MAAM,CAACW,aAAa,EAAE,KAAK,CAAC,CAAC1D,QAAQ,CAAC,QAAQ,CAAC;EAE/D,IAAI4D,GAAG,GAAG,kCAAkC;EAC5CA,GAAG,IAAI,GAAGD,GAAG,CAAC3C,KAAK,CAAC,UAAU,CAAC,CAAC6C,IAAI,CAAC,IAAI,CAAC,EAAE;EAC5CD,GAAG,IAAI,kCAAkC;EACzC,OAAOA,GAAG;AACZ;AAEA,SAASV,YAAYA,CAACY,MAAM,EAAE;EAC5B,MAAMC,GAAG,GAAGD,MAAM,CAAC,CAAC,CAAC;EACrB,IAAIC,GAAG,GAAG,GAAG,IAAIA,GAAG,GAAG,GAAG,EAAE;IAC1B,OAAO,KAAKD,MAAM,EAAE;EACtB;EACA,OAAOA,MAAM;AACf;AAEA,SAASE,KAAKA,CAACC,MAAM,EAAE;EACrB,MAAMC,IAAI,GAAGD,MAAM,CAACjE,QAAQ,CAAC,EAAE,CAAC;EAChC,IAAIkE,IAAI,CAACb,MAAM,GAAG,CAAC,EAAE;IACnB,OAAO,IAAIa,IAAI,EAAE;EACnB;EACA,OAAOA,IAAI;AACb;AAEA,SAASV,eAAeA,CAAClD,CAAC,EAAE;EAC1B,IAAIA,CAAC,IAAI,GAAG,EAAE;IACZ,OAAO0D,KAAK,CAAC1D,CAAC,CAAC;EACjB;EACA,MAAM6D,IAAI,GAAGH,KAAK,CAAC1D,CAAC,CAAC;EACrB,MAAM8D,kBAAkB,GAAG,GAAG,GAAGD,IAAI,CAACd,MAAM,GAAG,CAAC;EAChD,OAAOW,KAAK,CAACI,kBAAkB,CAAC,GAAGD,IAAI;AACzC","ignoreList":[]}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
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 Google Play Games Services.
|
|
11
|
+
*
|
|
12
|
+
* @class GooglePlayGamesServicesAdapter
|
|
13
|
+
* @param {Object} options - The adapter configuration options.
|
|
14
|
+
* @param {string} options.clientId - Your Google Play Games Services App Client ID. Required for secure authentication.
|
|
15
|
+
* @param {string} options.clientSecret - Your Google Play Games Services 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 Google Play Games Services authentication, use the following structure:
|
|
21
|
+
* ```json
|
|
22
|
+
* {
|
|
23
|
+
* "auth": {
|
|
24
|
+
* "gpgames": {
|
|
25
|
+
* "clientId": "your-client-id",
|
|
26
|
+
* "clientSecret": "your-client-secret"
|
|
27
|
+
* }
|
|
28
|
+
* }
|
|
29
|
+
* }
|
|
30
|
+
* ```
|
|
31
|
+
* ### Insecure Configuration (Not Recommended)
|
|
32
|
+
* ```json
|
|
33
|
+
* {
|
|
34
|
+
* "auth": {
|
|
35
|
+
* "gpgames": {
|
|
36
|
+
* "enableInsecureAuth": true
|
|
37
|
+
* }
|
|
38
|
+
* }
|
|
39
|
+
* }
|
|
40
|
+
* ```
|
|
41
|
+
*
|
|
42
|
+
* The adapter requires the following `authData` fields:
|
|
43
|
+
* - **Secure Authentication**: `code`, `redirect_uri`.
|
|
44
|
+
* - **Insecure Authentication (Not Recommended)**: `id`, `access_token`.
|
|
45
|
+
*
|
|
46
|
+
* ## Auth Payloads
|
|
47
|
+
* ### Secure Authentication Payload
|
|
48
|
+
* ```json
|
|
49
|
+
* {
|
|
50
|
+
* "gpgames": {
|
|
51
|
+
* "code": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
|
52
|
+
* "redirect_uri": "https://example.com/callback"
|
|
53
|
+
* }
|
|
54
|
+
* }
|
|
55
|
+
* ```
|
|
56
|
+
*
|
|
57
|
+
* ### Insecure Authentication Payload (Not Recommended)
|
|
58
|
+
* ```json
|
|
59
|
+
* {
|
|
60
|
+
* "gpgames": {
|
|
61
|
+
* "id": "123456789",
|
|
62
|
+
* "access_token": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"
|
|
63
|
+
* }
|
|
64
|
+
* }
|
|
65
|
+
* ```
|
|
66
|
+
*
|
|
67
|
+
* ## Notes
|
|
68
|
+
* - `enableInsecureAuth` is **not recommended** and may be removed in future versions. Use secure authentication with `code` and `redirect_uri`.
|
|
69
|
+
* - Secure authentication exchanges the `code` provided by the client for an access token using Google Play Games Services' OAuth API.
|
|
70
|
+
*
|
|
71
|
+
* @see {@link https://developers.google.com/games/services/console/enabling Google Play Games Services Authentication Documentation}
|
|
72
|
+
*/
|
|
73
|
+
|
|
74
|
+
class GooglePlayGamesServicesAdapter extends _BaseCodeAuthAdapter.default {
|
|
75
|
+
constructor() {
|
|
76
|
+
super("gpgames");
|
|
77
|
+
}
|
|
78
|
+
async getAccessTokenFromCode(authData) {
|
|
79
|
+
const tokenUrl = 'https://oauth2.googleapis.com/token';
|
|
80
|
+
const response = await fetch(tokenUrl, {
|
|
81
|
+
method: 'POST',
|
|
82
|
+
headers: {
|
|
83
|
+
'Content-Type': 'application/json',
|
|
84
|
+
Accept: 'application/json'
|
|
85
|
+
},
|
|
86
|
+
body: JSON.stringify({
|
|
87
|
+
client_id: this.clientId,
|
|
88
|
+
client_secret: this.clientSecret,
|
|
89
|
+
code: authData.code,
|
|
90
|
+
redirect_uri: authData.redirectUri,
|
|
91
|
+
grant_type: 'authorization_code'
|
|
92
|
+
})
|
|
93
|
+
});
|
|
94
|
+
if (!response.ok) {
|
|
95
|
+
throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to exchange code for token: ${response.statusText}`);
|
|
96
|
+
}
|
|
97
|
+
const data = await response.json();
|
|
98
|
+
if (data.error) {
|
|
99
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, data.error_description || data.error);
|
|
100
|
+
}
|
|
101
|
+
return data.access_token;
|
|
102
|
+
}
|
|
103
|
+
async getUserFromAccessToken(accessToken, authData) {
|
|
104
|
+
const userApiUrl = `https://www.googleapis.com/games/v1/players/${authData.id}`;
|
|
105
|
+
const response = await fetch(userApiUrl, {
|
|
106
|
+
method: 'GET',
|
|
107
|
+
headers: {
|
|
108
|
+
Authorization: `Bearer ${accessToken}`,
|
|
109
|
+
Accept: 'application/json'
|
|
110
|
+
}
|
|
111
|
+
});
|
|
112
|
+
if (!response.ok) {
|
|
113
|
+
throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to fetch Google Play Games Services user: ${response.statusText}`);
|
|
114
|
+
}
|
|
115
|
+
const userData = await response.json();
|
|
116
|
+
if (!userData.playerId || userData.playerId !== authData.id) {
|
|
117
|
+
throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Invalid Google Play Games Services user data received.');
|
|
118
|
+
}
|
|
119
|
+
return {
|
|
120
|
+
id: userData.playerId
|
|
121
|
+
};
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
var _default = exports.default = new GooglePlayGamesServicesAdapter();
|
|
125
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_BaseCodeAuthAdapter","_interopRequireDefault","require","e","__esModule","default","GooglePlayGamesServicesAdapter","BaseCodeAuthAdapter","constructor","getAccessTokenFromCode","authData","tokenUrl","response","fetch","method","headers","Accept","body","JSON","stringify","client_id","clientId","client_secret","clientSecret","code","redirect_uri","redirectUri","grant_type","ok","Parse","Error","VALIDATION_ERROR","statusText","data","json","error","OBJECT_NOT_FOUND","error_description","access_token","getUserFromAccessToken","accessToken","userApiUrl","id","Authorization","userData","playerId","_default","exports"],"sources":["../../../src/Adapters/Auth/gpgames.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Google Play Games Services.\n *\n * @class GooglePlayGamesServicesAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.clientId - Your Google Play Games Services App Client ID. Required for secure authentication.\n * @param {string} options.clientSecret - Your Google Play Games Services 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 Google Play Games Services authentication, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *     \"gpgames\": {\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 *     \"gpgames\": {\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 *   \"gpgames\": {\n *     \"code\": \"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\",\n *     \"redirect_uri\": \"https://example.com/callback\"\n *   }\n * }\n * ```\n *\n * ### Insecure Authentication Payload (Not Recommended)\n * ```json\n * {\n *   \"gpgames\": {\n *     \"id\": \"123456789\",\n *     \"access_token\": \"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx\"\n *   }\n * }\n * ```\n *\n * ## Notes\n * - `enableInsecureAuth` is **not recommended** and may be removed in future versions. Use secure authentication with `code` and `redirect_uri`.\n * - Secure authentication exchanges the `code` provided by the client for an access token using Google Play Games Services' OAuth API.\n *\n * @see {@link https://developers.google.com/games/services/console/enabling Google Play Games Services Authentication Documentation}\n */\n\nimport BaseCodeAuthAdapter from './BaseCodeAuthAdapter';\nclass GooglePlayGamesServicesAdapter extends BaseCodeAuthAdapter {\n  constructor() {\n    super(\"gpgames\");\n  }\n\n  async getAccessTokenFromCode(authData) {\n    const tokenUrl = 'https://oauth2.googleapis.com/token';\n    const response = await fetch(tokenUrl, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        Accept: 'application/json',\n      },\n      body: JSON.stringify({\n        client_id: this.clientId,\n        client_secret: this.clientSecret,\n        code: authData.code,\n        redirect_uri: authData.redirectUri,\n        grant_type: 'authorization_code',\n      }),\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(\n        Parse.Error.VALIDATION_ERROR,\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, authData) {\n    const userApiUrl = `https://www.googleapis.com/games/v1/players/${authData.id}`;\n    const response = await fetch(userApiUrl, {\n      method: 'GET',\n      headers: {\n        Authorization: `Bearer ${accessToken}`,\n        Accept: 'application/json',\n      },\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(\n        Parse.Error.VALIDATION_ERROR,\n        `Failed to fetch Google Play Games Services user: ${response.statusText}`\n      );\n    }\n\n    const userData = await response.json();\n    if (!userData.playerId || userData.playerId !== authData.id) {\n      throw new Parse.Error(\n        Parse.Error.VALIDATION_ERROR,\n        'Invalid Google Play Games Services user data received.'\n      );\n    }\n\n    return {\n      id: userData.playerId\n    };\n  }\n\n}\n\nexport default new GooglePlayGamesServicesAdapter();\n"],"mappings":"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","ignoreList":[]}
|