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,57 +1,184 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
3
|
+
/**
|
|
4
|
+
* Parse Server authentication adapter for Facebook.
|
|
5
|
+
*
|
|
6
|
+
* @class FacebookAdapter
|
|
7
|
+
* @param {Object} options - The adapter configuration options.
|
|
8
|
+
* @param {string} options.appSecret - Your Facebook App Secret. Required for secure authentication.
|
|
9
|
+
* @param {string[]} options.appIds - An array of Facebook App IDs. Required for validating the app.
|
|
10
|
+
*
|
|
11
|
+
* @description
|
|
12
|
+
* ## Parse Server Configuration
|
|
13
|
+
* To configure Parse Server for Facebook authentication, use the following structure:
|
|
14
|
+
* ```json
|
|
15
|
+
* {
|
|
16
|
+
* "auth": {
|
|
17
|
+
* "facebook": {
|
|
18
|
+
* "appSecret": "your-app-secret",
|
|
19
|
+
* "appIds": ["your-app-id"]
|
|
20
|
+
* }
|
|
21
|
+
* }
|
|
22
|
+
* }
|
|
23
|
+
* ```
|
|
24
|
+
*
|
|
25
|
+
* The adapter supports the following authentication methods:
|
|
26
|
+
* - **Standard Login**: Requires `id` and `access_token`.
|
|
27
|
+
* - **Limited Login**: Requires `id` and `token`.
|
|
28
|
+
*
|
|
29
|
+
* ## Auth Payloads
|
|
30
|
+
* ### Standard Login Payload
|
|
31
|
+
* ```json
|
|
32
|
+
* {
|
|
33
|
+
* "facebook": {
|
|
34
|
+
* "id": "1234567",
|
|
35
|
+
* "access_token": "abc123def456ghi789"
|
|
36
|
+
* }
|
|
37
|
+
* }
|
|
38
|
+
* ```
|
|
39
|
+
*
|
|
40
|
+
* ### Limited Login Payload
|
|
41
|
+
* ```json
|
|
42
|
+
* {
|
|
43
|
+
* "facebook": {
|
|
44
|
+
* "id": "1234567",
|
|
45
|
+
* "token": "xxxxx.yyyyy.zzzzz"
|
|
46
|
+
* }
|
|
47
|
+
* }
|
|
48
|
+
* ```
|
|
49
|
+
*
|
|
50
|
+
* ## Notes
|
|
51
|
+
* - **Standard Login**: Use `id` and `access_token` for full functionality.
|
|
52
|
+
* - **Limited Login**: Use `id` and `token` (JWT) when tracking is opted out (e.g., via Apple's App Tracking Transparency).
|
|
53
|
+
* - Supported Parse Server versions:
|
|
54
|
+
* - `>= 6.5.6 < 7`
|
|
55
|
+
* - `>= 7.0.1`
|
|
56
|
+
*
|
|
57
|
+
* Secure authentication is recommended to ensure proper data protection and compliance with Facebook's guidelines.
|
|
58
|
+
*
|
|
59
|
+
* @see {@link https://developers.facebook.com/docs/facebook-login/limited-login/ Facebook Limited Login}
|
|
60
|
+
* @see {@link https://developers.facebook.com/docs/facebook-login/facebook-login-for-business/ Facebook Login for Business}
|
|
61
|
+
*/
|
|
6
62
|
|
|
7
|
-
//
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
63
|
+
// Helper functions for accessing the Facebook Graph API.
|
|
64
|
+
const Parse = require('parse/node').Parse;
|
|
65
|
+
const crypto = require('crypto');
|
|
66
|
+
const jwksClient = require('jwks-rsa');
|
|
67
|
+
const jwt = require('jsonwebtoken');
|
|
68
|
+
const httpsRequest = require('./httpsRequest');
|
|
69
|
+
const authUtils = require('./utils');
|
|
70
|
+
const TOKEN_ISSUER = 'https://www.facebook.com';
|
|
71
|
+
function getAppSecretPath(authData, options = {}) {
|
|
72
|
+
const appSecret = options.appSecret;
|
|
73
|
+
if (!appSecret) {
|
|
74
|
+
return '';
|
|
75
|
+
}
|
|
76
|
+
const appsecret_proof = crypto.createHmac('sha256', appSecret).update(authData.access_token).digest('hex');
|
|
77
|
+
return `&appsecret_proof=${appsecret_proof}`;
|
|
78
|
+
}
|
|
79
|
+
function validateGraphToken(authData, options) {
|
|
80
|
+
return graphRequest('me?fields=id&access_token=' + authData.access_token + getAppSecretPath(authData, options)).then(data => {
|
|
81
|
+
if (data && data.id == authData.id || process.env.TESTING && authData.id === 'test') {
|
|
11
82
|
return;
|
|
12
83
|
}
|
|
13
84
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');
|
|
14
85
|
});
|
|
15
86
|
}
|
|
16
|
-
|
|
17
|
-
// Returns a promise that fulfills iff this app id is valid.
|
|
18
|
-
function validateAppId(appIds, authData) {
|
|
87
|
+
async function validateGraphAppId(appIds, authData, options) {
|
|
19
88
|
var access_token = authData.access_token;
|
|
89
|
+
if (process.env.TESTING && access_token === 'test') {
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
if (!Array.isArray(appIds)) {
|
|
93
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'appIds must be an array.');
|
|
94
|
+
}
|
|
20
95
|
if (!appIds.length) {
|
|
21
96
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is not configured.');
|
|
22
97
|
}
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
return;
|
|
26
|
-
}
|
|
98
|
+
const data = await graphRequest(`app?access_token=${access_token}${getAppSecretPath(authData, options)}`);
|
|
99
|
+
if (!data || !appIds.includes(data.id)) {
|
|
27
100
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
const getFacebookKeyByKeyId = async (keyId, cacheMaxEntries, cacheMaxAge) => {
|
|
104
|
+
const client = jwksClient({
|
|
105
|
+
jwksUri: `${TOKEN_ISSUER}/.well-known/oauth/openid/jwks/`,
|
|
106
|
+
cache: true,
|
|
107
|
+
cacheMaxEntries,
|
|
108
|
+
cacheMaxAge
|
|
28
109
|
});
|
|
110
|
+
let key;
|
|
111
|
+
try {
|
|
112
|
+
key = await authUtils.getSigningKey(client, keyId);
|
|
113
|
+
} catch {
|
|
114
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Unable to find matching key for Key ID: ${keyId}`);
|
|
115
|
+
}
|
|
116
|
+
return key;
|
|
117
|
+
};
|
|
118
|
+
const verifyIdToken = async ({
|
|
119
|
+
token,
|
|
120
|
+
id
|
|
121
|
+
}, {
|
|
122
|
+
clientId,
|
|
123
|
+
cacheMaxEntries,
|
|
124
|
+
cacheMaxAge
|
|
125
|
+
}) => {
|
|
126
|
+
if (!token) {
|
|
127
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'id token is invalid for this user.');
|
|
128
|
+
}
|
|
129
|
+
const {
|
|
130
|
+
kid: keyId,
|
|
131
|
+
alg: algorithm
|
|
132
|
+
} = authUtils.getHeaderFromToken(token);
|
|
133
|
+
const ONE_HOUR_IN_MS = 3600000;
|
|
134
|
+
let jwtClaims;
|
|
135
|
+
cacheMaxAge = cacheMaxAge || ONE_HOUR_IN_MS;
|
|
136
|
+
cacheMaxEntries = cacheMaxEntries || 5;
|
|
137
|
+
const facebookKey = await getFacebookKeyByKeyId(keyId, cacheMaxEntries, cacheMaxAge);
|
|
138
|
+
const signingKey = facebookKey.publicKey || facebookKey.rsaPublicKey;
|
|
139
|
+
try {
|
|
140
|
+
jwtClaims = jwt.verify(token, signingKey, {
|
|
141
|
+
algorithms: algorithm,
|
|
142
|
+
// the audience can be checked against a string, a regular expression or a list of strings and/or regular expressions.
|
|
143
|
+
audience: clientId
|
|
144
|
+
});
|
|
145
|
+
} catch (exception) {
|
|
146
|
+
const message = exception.message;
|
|
147
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);
|
|
148
|
+
}
|
|
149
|
+
if (jwtClaims.iss !== TOKEN_ISSUER) {
|
|
150
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `id token not issued by correct OpenID provider - expected: ${TOKEN_ISSUER} | from: ${jwtClaims.iss}`);
|
|
151
|
+
}
|
|
152
|
+
if (jwtClaims.sub !== id) {
|
|
153
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'auth data is invalid for this user.');
|
|
154
|
+
}
|
|
155
|
+
return jwtClaims;
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
// Returns a promise that fulfills iff this user id is valid.
|
|
159
|
+
function validateAuthData(authData, options) {
|
|
160
|
+
if (authData.token) {
|
|
161
|
+
return verifyIdToken(authData, options);
|
|
162
|
+
} else {
|
|
163
|
+
return validateGraphToken(authData, options);
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
// Returns a promise that fulfills iff this app id is valid.
|
|
168
|
+
function validateAppId(appIds, authData, options) {
|
|
169
|
+
if (authData.token) {
|
|
170
|
+
return Promise.resolve();
|
|
171
|
+
} else {
|
|
172
|
+
return validateGraphAppId(appIds, authData, options);
|
|
173
|
+
}
|
|
29
174
|
}
|
|
30
175
|
|
|
31
176
|
// A promisey wrapper for FB graph requests.
|
|
32
177
|
function graphRequest(path) {
|
|
33
|
-
return
|
|
34
|
-
https.get('https://graph.facebook.com/' + path, function (res) {
|
|
35
|
-
var data = '';
|
|
36
|
-
res.on('data', function (chunk) {
|
|
37
|
-
data += chunk;
|
|
38
|
-
});
|
|
39
|
-
res.on('end', function () {
|
|
40
|
-
try {
|
|
41
|
-
data = JSON.parse(data);
|
|
42
|
-
} catch (e) {
|
|
43
|
-
return reject(e);
|
|
44
|
-
}
|
|
45
|
-
resolve(data);
|
|
46
|
-
});
|
|
47
|
-
}).on('error', function () {
|
|
48
|
-
reject('Failed to validate this access token with Facebook.');
|
|
49
|
-
});
|
|
50
|
-
});
|
|
178
|
+
return httpsRequest.get('https://graph.facebook.com/' + path);
|
|
51
179
|
}
|
|
52
|
-
|
|
53
180
|
module.exports = {
|
|
54
181
|
validateAppId: validateAppId,
|
|
55
182
|
validateAuthData: validateAuthData
|
|
56
183
|
};
|
|
57
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BZGFwdGVycy9BdXRoL2ZhY2Vib29rLmpzIl0sIm5hbWVzIjpbImh0dHBzIiwicmVxdWlyZSIsIlBhcnNlIiwidmFsaWRhdGVBdXRoRGF0YSIsImF1dGhEYXRhIiwiZ3JhcGhSZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwidGhlbiIsImRhdGEiLCJpZCIsIkVycm9yIiwiT0JKRUNUX05PVF9GT1VORCIsInZhbGlkYXRlQXBwSWQiLCJhcHBJZHMiLCJsZW5ndGgiLCJpbmRleE9mIiwicGF0aCIsIlByb21pc2UiLCJyZXNvbHZlIiwicmVqZWN0IiwiZ2V0IiwicmVzIiwib24iLCJjaHVuayIsIkpTT04iLCJwYXJzZSIsImUiLCJtb2R1bGUiLCJleHBvcnRzIl0sIm1hcHBpbmdzIjoiOztBQUFBO0FBQ0EsSUFBSUEsUUFBUUMsUUFBUSxPQUFSLENBQVo7QUFDQSxJQUFJQyxRQUFRRCxRQUFRLFlBQVIsRUFBc0JDLEtBQWxDOztBQUVBO0FBQ0EsU0FBU0MsZ0JBQVQsQ0FBMEJDLFFBQTFCLEVBQW9DO0FBQ2xDLFNBQU9DLGFBQWEsK0JBQStCRCxTQUFTRSxZQUFyRCxFQUNKQyxJQURJLENBQ0VDLElBQUQsSUFBVTtBQUNkLFFBQUlBLFFBQVFBLEtBQUtDLEVBQUwsSUFBV0wsU0FBU0ssRUFBaEMsRUFBb0M7QUFDbEM7QUFDRDtBQUNELFVBQU0sSUFBSVAsTUFBTVEsS0FBVixDQUNKUixNQUFNUSxLQUFOLENBQVlDLGdCQURSLEVBRUoseUNBRkksQ0FBTjtBQUdELEdBUkksQ0FBUDtBQVNEOztBQUVEO0FBQ0EsU0FBU0MsYUFBVCxDQUF1QkMsTUFBdkIsRUFBK0JULFFBQS9CLEVBQXlDO0FBQ3ZDLE1BQUlFLGVBQWVGLFNBQVNFLFlBQTVCO0FBQ0EsTUFBSSxDQUFDTyxPQUFPQyxNQUFaLEVBQW9CO0FBQ2xCLFVBQU0sSUFBSVosTUFBTVEsS0FBVixDQUNKUixNQUFNUSxLQUFOLENBQVlDLGdCQURSLEVBRUosa0NBRkksQ0FBTjtBQUdEO0FBQ0QsU0FBT04sYUFBYSxzQkFBc0JDLFlBQW5DLEVBQ0pDLElBREksQ0FDRUMsSUFBRCxJQUFVO0FBQ2QsUUFBSUEsUUFBUUssT0FBT0UsT0FBUCxDQUFlUCxLQUFLQyxFQUFwQixLQUEyQixDQUFDLENBQXhDLEVBQTJDO0FBQ3pDO0FBQ0Q7QUFDRCxVQUFNLElBQUlQLE1BQU1RLEtBQVYsQ0FDSlIsTUFBTVEsS0FBTixDQUFZQyxnQkFEUixFQUVKLHlDQUZJLENBQU47QUFHRCxHQVJJLENBQVA7QUFTRDs7QUFFRDtBQUNBLFNBQVNOLFlBQVQsQ0FBc0JXLElBQXRCLEVBQTRCO0FBQzFCLFNBQU8sSUFBSUMsT0FBSixDQUFZLFVBQVNDLE9BQVQsRUFBa0JDLE1BQWxCLEVBQTBCO0FBQzNDbkIsVUFBTW9CLEdBQU4sQ0FBVSxnQ0FBZ0NKLElBQTFDLEVBQWdELFVBQVNLLEdBQVQsRUFBYztBQUM1RCxVQUFJYixPQUFPLEVBQVg7QUFDQWEsVUFBSUMsRUFBSixDQUFPLE1BQVAsRUFBZSxVQUFTQyxLQUFULEVBQWdCO0FBQzdCZixnQkFBUWUsS0FBUjtBQUNELE9BRkQ7QUFHQUYsVUFBSUMsRUFBSixDQUFPLEtBQVAsRUFBYyxZQUFXO0FBQ3ZCLFlBQUk7QUFDRmQsaUJBQU9nQixLQUFLQyxLQUFMLENBQVdqQixJQUFYLENBQVA7QUFDRCxTQUZELENBRUUsT0FBTWtCLENBQU4sRUFBUztBQUNULGlCQUFPUCxPQUFPTyxDQUFQLENBQVA7QUFDRDtBQUNEUixnQkFBUVYsSUFBUjtBQUNELE9BUEQ7QUFRRCxLQWJELEVBYUdjLEVBYkgsQ0FhTSxPQWJOLEVBYWUsWUFBVztBQUN4QkgsYUFBTyxxREFBUDtBQUNELEtBZkQ7QUFnQkQsR0FqQk0sQ0FBUDtBQWtCRDs7QUFFRFEsT0FBT0MsT0FBUCxHQUFpQjtBQUNmaEIsaUJBQWVBLGFBREE7QUFFZlQsb0JBQWtCQTtBQUZILENBQWpCIiwiZmlsZSI6ImZhY2Vib29rLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLy8gSGVscGVyIGZ1bmN0aW9ucyBmb3IgYWNjZXNzaW5nIHRoZSBGYWNlYm9vayBHcmFwaCBBUEkuXG52YXIgaHR0cHMgPSByZXF1aXJlKCdodHRwcycpO1xudmFyIFBhcnNlID0gcmVxdWlyZSgncGFyc2Uvbm9kZScpLlBhcnNlO1xuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIHVzZXIgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUF1dGhEYXRhKGF1dGhEYXRhKSB7XG4gIHJldHVybiBncmFwaFJlcXVlc3QoJ21lP2ZpZWxkcz1pZCZhY2Nlc3NfdG9rZW49JyArIGF1dGhEYXRhLmFjY2Vzc190b2tlbilcbiAgICAudGhlbigoZGF0YSkgPT4ge1xuICAgICAgaWYgKGRhdGEgJiYgZGF0YS5pZCA9PSBhdXRoRGF0YS5pZCkge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG4gICAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoXG4gICAgICAgIFBhcnNlLkVycm9yLk9CSkVDVF9OT1RfRk9VTkQsXG4gICAgICAgICdGYWNlYm9vayBhdXRoIGlzIGludmFsaWQgZm9yIHRoaXMgdXNlci4nKTtcbiAgICB9KTtcbn1cblxuLy8gUmV0dXJucyBhIHByb21pc2UgdGhhdCBmdWxmaWxscyBpZmYgdGhpcyBhcHAgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUFwcElkKGFwcElkcywgYXV0aERhdGEpIHtcbiAgdmFyIGFjY2Vzc190b2tlbiA9IGF1dGhEYXRhLmFjY2Vzc190b2tlbjtcbiAgaWYgKCFhcHBJZHMubGVuZ3RoKSB7XG4gICAgdGhyb3cgbmV3IFBhcnNlLkVycm9yKFxuICAgICAgUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCxcbiAgICAgICdGYWNlYm9vayBhdXRoIGlzIG5vdCBjb25maWd1cmVkLicpO1xuICB9XG4gIHJldHVybiBncmFwaFJlcXVlc3QoJ2FwcD9hY2Nlc3NfdG9rZW49JyArIGFjY2Vzc190b2tlbilcbiAgICAudGhlbigoZGF0YSkgPT4ge1xuICAgICAgaWYgKGRhdGEgJiYgYXBwSWRzLmluZGV4T2YoZGF0YS5pZCkgIT0gLTEpIHtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuICAgICAgdGhyb3cgbmV3IFBhcnNlLkVycm9yKFxuICAgICAgICBQYXJzZS5FcnJvci5PQkpFQ1RfTk9UX0ZPVU5ELFxuICAgICAgICAnRmFjZWJvb2sgYXV0aCBpcyBpbnZhbGlkIGZvciB0aGlzIHVzZXIuJyk7XG4gICAgfSk7XG59XG5cbi8vIEEgcHJvbWlzZXkgd3JhcHBlciBmb3IgRkIgZ3JhcGggcmVxdWVzdHMuXG5mdW5jdGlvbiBncmFwaFJlcXVlc3QocGF0aCkge1xuICByZXR1cm4gbmV3IFByb21pc2UoZnVuY3Rpb24ocmVzb2x2ZSwgcmVqZWN0KSB7XG4gICAgaHR0cHMuZ2V0KCdodHRwczovL2dyYXBoLmZhY2Vib29rLmNvbS8nICsgcGF0aCwgZnVuY3Rpb24ocmVzKSB7XG4gICAgICB2YXIgZGF0YSA9ICcnO1xuICAgICAgcmVzLm9uKCdkYXRhJywgZnVuY3Rpb24oY2h1bmspIHtcbiAgICAgICAgZGF0YSArPSBjaHVuaztcbiAgICAgIH0pO1xuICAgICAgcmVzLm9uKCdlbmQnLCBmdW5jdGlvbigpIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBkYXRhID0gSlNPTi5wYXJzZShkYXRhKTtcbiAgICAgICAgfSBjYXRjaChlKSB7XG4gICAgICAgICAgcmV0dXJuIHJlamVjdChlKTtcbiAgICAgICAgfVxuICAgICAgICByZXNvbHZlKGRhdGEpO1xuICAgICAgfSk7XG4gICAgfSkub24oJ2Vycm9yJywgZnVuY3Rpb24oKSB7XG4gICAgICByZWplY3QoJ0ZhaWxlZCB0byB2YWxpZGF0ZSB0aGlzIGFjY2VzcyB0b2tlbiB3aXRoIEZhY2Vib29rLicpO1xuICAgIH0pO1xuICB9KTtcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gIHZhbGlkYXRlQXBwSWQ6IHZhbGlkYXRlQXBwSWQsXG4gIHZhbGlkYXRlQXV0aERhdGE6IHZhbGlkYXRlQXV0aERhdGFcbn07XG4iXX0=
|
|
184
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","crypto","jwksClient","jwt","httpsRequest","authUtils","TOKEN_ISSUER","getAppSecretPath","authData","options","appSecret","appsecret_proof","createHmac","update","access_token","digest","validateGraphToken","graphRequest","then","data","id","process","env","TESTING","Error","OBJECT_NOT_FOUND","validateGraphAppId","appIds","Array","isArray","length","includes","getFacebookKeyByKeyId","keyId","cacheMaxEntries","cacheMaxAge","client","jwksUri","cache","key","getSigningKey","verifyIdToken","token","clientId","kid","alg","algorithm","getHeaderFromToken","ONE_HOUR_IN_MS","jwtClaims","facebookKey","signingKey","publicKey","rsaPublicKey","verify","algorithms","audience","exception","message","iss","sub","validateAuthData","validateAppId","Promise","resolve","path","get","module","exports"],"sources":["../../../src/Adapters/Auth/facebook.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Facebook.\n *\n * @class FacebookAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.appSecret - Your Facebook App Secret. Required for secure authentication.\n * @param {string[]} options.appIds - An array of Facebook App IDs. Required for validating the app.\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for Facebook authentication, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *     \"facebook\": {\n *       \"appSecret\": \"your-app-secret\",\n *       \"appIds\": [\"your-app-id\"]\n *     }\n *   }\n * }\n * ```\n *\n * The adapter supports the following authentication methods:\n * - **Standard Login**: Requires `id` and `access_token`.\n * - **Limited Login**: Requires `id` and `token`.\n *\n * ## Auth Payloads\n * ### Standard Login Payload\n * ```json\n * {\n *   \"facebook\": {\n *     \"id\": \"1234567\",\n *     \"access_token\": \"abc123def456ghi789\"\n *   }\n * }\n * ```\n *\n * ### Limited Login Payload\n * ```json\n * {\n *   \"facebook\": {\n *     \"id\": \"1234567\",\n *     \"token\": \"xxxxx.yyyyy.zzzzz\"\n *   }\n * }\n * ```\n *\n * ## Notes\n * - **Standard Login**: Use `id` and `access_token` for full functionality.\n * - **Limited Login**: Use `id` and `token` (JWT) when tracking is opted out (e.g., via Apple's App Tracking Transparency).\n * - Supported Parse Server versions:\n *   - `>= 6.5.6 < 7`\n *   - `>= 7.0.1`\n *\n * Secure authentication is recommended to ensure proper data protection and compliance with Facebook's guidelines.\n *\n * @see {@link https://developers.facebook.com/docs/facebook-login/limited-login/ Facebook Limited Login}\n * @see {@link https://developers.facebook.com/docs/facebook-login/facebook-login-for-business/ Facebook Login for Business}\n */\n\n// Helper functions for accessing the Facebook Graph API.\nconst Parse = require('parse/node').Parse;\nconst crypto = require('crypto');\nconst jwksClient = require('jwks-rsa');\nconst jwt = require('jsonwebtoken');\nconst httpsRequest = require('./httpsRequest');\nconst authUtils = require('./utils');\n\nconst TOKEN_ISSUER = 'https://www.facebook.com';\n\nfunction getAppSecretPath(authData, options = {}) {\n  const appSecret = options.appSecret;\n  if (!appSecret) {\n    return '';\n  }\n  const appsecret_proof = crypto\n    .createHmac('sha256', appSecret)\n    .update(authData.access_token)\n    .digest('hex');\n\n  return `&appsecret_proof=${appsecret_proof}`;\n}\n\nfunction validateGraphToken(authData, options) {\n  return graphRequest(\n    'me?fields=id&access_token=' + authData.access_token + getAppSecretPath(authData, options)\n  ).then(data => {\n    if ((data && data.id == authData.id) || (process.env.TESTING && authData.id === 'test')) {\n      return;\n    }\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');\n  });\n}\n\nasync function validateGraphAppId(appIds, authData, options) {\n  var access_token = authData.access_token;\n  if (process.env.TESTING && access_token === 'test') {\n    return;\n  }\n  if (!Array.isArray(appIds)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'appIds must be an array.');\n  }\n  if (!appIds.length) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is not configured.');\n  }\n  const data = await graphRequest(\n    `app?access_token=${access_token}${getAppSecretPath(authData, options)}`\n  );\n  if (!data || !appIds.includes(data.id)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');\n  }\n}\n\nconst getFacebookKeyByKeyId = async (keyId, cacheMaxEntries, cacheMaxAge) => {\n  const client = jwksClient({\n    jwksUri: `${TOKEN_ISSUER}/.well-known/oauth/openid/jwks/`,\n    cache: true,\n    cacheMaxEntries,\n    cacheMaxAge,\n  });\n\n  let key;\n  try {\n    key = await authUtils.getSigningKey(client, keyId);\n  } catch {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Unable to find matching key for Key ID: ${keyId}`\n    );\n  }\n  return key;\n};\n\nconst verifyIdToken = async ({ token, id }, { clientId, cacheMaxEntries, cacheMaxAge }) => {\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  const ONE_HOUR_IN_MS = 3600000;\n  let jwtClaims;\n\n  cacheMaxAge = cacheMaxAge || ONE_HOUR_IN_MS;\n  cacheMaxEntries = cacheMaxEntries || 5;\n\n  const facebookKey = await getFacebookKeyByKeyId(keyId, cacheMaxEntries, cacheMaxAge);\n  const signingKey = facebookKey.publicKey || facebookKey.rsaPublicKey;\n\n  try {\n    jwtClaims = jwt.verify(token, signingKey, {\n      algorithms: algorithm,\n      // the audience can be checked against a string, a regular expression or a list of strings and/or regular expressions.\n      audience: clientId,\n    });\n  } catch (exception) {\n    const message = exception.message;\n\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);\n  }\n\n  if (jwtClaims.iss !== TOKEN_ISSUER) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `id token not issued by correct OpenID provider - expected: ${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  return jwtClaims;\n};\n\n// Returns a promise that fulfills iff this user id is valid.\nfunction validateAuthData(authData, options) {\n  if (authData.token) {\n    return verifyIdToken(authData, options);\n  } else {\n    return validateGraphToken(authData, options);\n  }\n}\n\n// Returns a promise that fulfills iff this app id is valid.\nfunction validateAppId(appIds, authData, options) {\n  if (authData.token) {\n    return Promise.resolve();\n  } else {\n    return validateGraphAppId(appIds, authData, options);\n  }\n}\n\n// A promisey wrapper for FB graph requests.\nfunction graphRequest(path) {\n  return httpsRequest.get('https://graph.facebook.com/' + path);\n}\n\nmodule.exports = {\n  validateAppId: validateAppId,\n  validateAuthData: validateAuthData,\n};\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.default = void 0;
|
|
7
|
+
var _crypto = _interopRequireDefault(require("crypto"));
|
|
8
|
+
var _nodeForge = require("node-forge");
|
|
9
|
+
var _AuthAdapter = _interopRequireDefault(require("./AuthAdapter"));
|
|
10
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
11
|
+
/**
|
|
12
|
+
* Parse Server authentication adapter for Apple Game Center.
|
|
13
|
+
*
|
|
14
|
+
* @class AppleGameCenterAdapter
|
|
15
|
+
* @param {Object} options - Configuration options for the adapter.
|
|
16
|
+
* @param {string} options.bundleId - Your Apple Game Center bundle ID. Required for secure authentication.
|
|
17
|
+
* @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).
|
|
18
|
+
*
|
|
19
|
+
* @param {Object} authData - The authentication data provided by the client.
|
|
20
|
+
* @param {string} authData.id - The user ID obtained from Apple Game Center.
|
|
21
|
+
* @param {string} authData.publicKeyUrl - The public key URL obtained from Apple Game Center.
|
|
22
|
+
* @param {string} authData.timestamp - The timestamp obtained from Apple Game Center.
|
|
23
|
+
* @param {string} authData.signature - The signature obtained from Apple Game Center.
|
|
24
|
+
* @param {string} authData.salt - The salt obtained from Apple Game Center.
|
|
25
|
+
* @param {string} [authData.bundleId] - **[DEPRECATED]** The bundle ID obtained from Apple Game Center (required for insecure authentication).
|
|
26
|
+
*
|
|
27
|
+
* @description
|
|
28
|
+
* ## Parse Server Configuration
|
|
29
|
+
* The following `authData` fields are required:
|
|
30
|
+
* `id`, `publicKeyUrl`, `timestamp`, `signature`, and `salt`. These fields are validated against the configured `bundleId` for additional security.
|
|
31
|
+
*
|
|
32
|
+
* To configure Parse Server for Apple Game Center authentication, use the following structure:
|
|
33
|
+
* ```json
|
|
34
|
+
* {
|
|
35
|
+
* "auth": {
|
|
36
|
+
* "gcenter": {
|
|
37
|
+
* "bundleId": "com.valid.app"
|
|
38
|
+
* }
|
|
39
|
+
* }
|
|
40
|
+
* ```
|
|
41
|
+
*
|
|
42
|
+
* ## Insecure Authentication (Not Recommended)
|
|
43
|
+
* The following `authData` fields are required for insecure authentication:
|
|
44
|
+
* `id`, `publicKeyUrl`, `timestamp`, `signature`, `salt`, and `bundleId` (**[DEPRECATED]**). This flow is insecure and poses potential security risks.
|
|
45
|
+
*
|
|
46
|
+
* To configure Parse Server for insecure authentication, use the following structure:
|
|
47
|
+
* ```json
|
|
48
|
+
* {
|
|
49
|
+
* "auth": {
|
|
50
|
+
* "gcenter": {
|
|
51
|
+
* "enableInsecureAuth": true
|
|
52
|
+
* }
|
|
53
|
+
* }
|
|
54
|
+
* ```
|
|
55
|
+
*
|
|
56
|
+
* ### Deprecation Notice
|
|
57
|
+
* The `enableInsecureAuth` option and `authData.bundleId` parameter are deprecated and may be removed in future releases. Use secure authentication with the `bundleId` configured in the `options` object instead.
|
|
58
|
+
*
|
|
59
|
+
*
|
|
60
|
+
* @example <caption>Secure Authentication Example</caption>
|
|
61
|
+
* // Example authData for secure authentication:
|
|
62
|
+
* const authData = {
|
|
63
|
+
* gcenter: {
|
|
64
|
+
* id: "1234567",
|
|
65
|
+
* publicKeyUrl: "https://valid.apple.com/public/timeout.cer",
|
|
66
|
+
* timestamp: 1460981421303,
|
|
67
|
+
* salt: "saltST==",
|
|
68
|
+
* signature: "PoDwf39DCN464B49jJCU0d9Y0J"
|
|
69
|
+
* }
|
|
70
|
+
* };
|
|
71
|
+
*
|
|
72
|
+
* @example <caption>Insecure Authentication Example (Not Recommended)</caption>
|
|
73
|
+
* // Example authData for insecure authentication:
|
|
74
|
+
* const authData = {
|
|
75
|
+
* gcenter: {
|
|
76
|
+
* id: "1234567",
|
|
77
|
+
* publicKeyUrl: "https://valid.apple.com/public/timeout.cer",
|
|
78
|
+
* timestamp: 1460981421303,
|
|
79
|
+
* salt: "saltST==",
|
|
80
|
+
* signature: "PoDwf39DCN464B49jJCU0d9Y0J",
|
|
81
|
+
* bundleId: "com.valid.app" // Deprecated.
|
|
82
|
+
* }
|
|
83
|
+
* };
|
|
84
|
+
*
|
|
85
|
+
* @see {@link https://developer.apple.com/documentation/gamekit/gklocalplayer/3516283-fetchitems Apple Game Center Documentation}
|
|
86
|
+
*/
|
|
87
|
+
/* global BigInt */
|
|
88
|
+
|
|
89
|
+
class GameCenterAuth extends _AuthAdapter.default {
|
|
90
|
+
constructor() {
|
|
91
|
+
super();
|
|
92
|
+
this.ca = {
|
|
93
|
+
cert: null,
|
|
94
|
+
url: null
|
|
95
|
+
};
|
|
96
|
+
this.cache = {};
|
|
97
|
+
this.bundleId = '';
|
|
98
|
+
}
|
|
99
|
+
validateOptions(options) {
|
|
100
|
+
if (!options) {
|
|
101
|
+
throw new Error('Game center auth options are required.');
|
|
102
|
+
}
|
|
103
|
+
if (!this.loadingPromise) {
|
|
104
|
+
this.loadingPromise = this.loadCertificate(options);
|
|
105
|
+
}
|
|
106
|
+
this.enableInsecureAuth = options.enableInsecureAuth;
|
|
107
|
+
this.bundleId = options.bundleId;
|
|
108
|
+
if (!this.enableInsecureAuth && !this.bundleId) {
|
|
109
|
+
throw new Error('bundleId is required for secure auth.');
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
async loadCertificate(options) {
|
|
113
|
+
const rootCertificateUrl = options.rootCertificateUrl || 'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';
|
|
114
|
+
if (this.ca.url === rootCertificateUrl) {
|
|
115
|
+
return rootCertificateUrl;
|
|
116
|
+
}
|
|
117
|
+
const {
|
|
118
|
+
certificate,
|
|
119
|
+
headers
|
|
120
|
+
} = await this.fetchCertificate(rootCertificateUrl);
|
|
121
|
+
if (headers.get('content-type') !== 'application/x-pem-file' || !headers.get('content-length') || parseInt(headers.get('content-length'), 10) > 10000) {
|
|
122
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid rootCertificateURL.');
|
|
123
|
+
}
|
|
124
|
+
this.ca.cert = _nodeForge.pki.certificateFromPem(certificate);
|
|
125
|
+
this.ca.url = rootCertificateUrl;
|
|
126
|
+
return rootCertificateUrl;
|
|
127
|
+
}
|
|
128
|
+
verifyPublicKeyUrl(publicKeyUrl) {
|
|
129
|
+
const regex = /^https:\/\/(?:[-_A-Za-z0-9]+\.){0,}apple\.com\/.*\.cer$/;
|
|
130
|
+
return regex.test(publicKeyUrl);
|
|
131
|
+
}
|
|
132
|
+
async fetchCertificate(url) {
|
|
133
|
+
const response = await fetch(url);
|
|
134
|
+
if (!response.ok) {
|
|
135
|
+
throw new Error(`Failed to fetch certificate: ${url}`);
|
|
136
|
+
}
|
|
137
|
+
const contentType = response.headers.get('content-type');
|
|
138
|
+
const isPem = contentType?.includes('application/x-pem-file');
|
|
139
|
+
if (isPem) {
|
|
140
|
+
const certificate = await response.text();
|
|
141
|
+
return {
|
|
142
|
+
certificate,
|
|
143
|
+
headers: response.headers
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
const data = await response.arrayBuffer();
|
|
147
|
+
const binaryData = Buffer.from(data);
|
|
148
|
+
const asn1Cert = _nodeForge.asn1.fromDer(binaryData.toString('binary'));
|
|
149
|
+
const forgeCert = _nodeForge.pki.certificateFromAsn1(asn1Cert);
|
|
150
|
+
const certificate = _nodeForge.pki.certificateToPem(forgeCert);
|
|
151
|
+
return {
|
|
152
|
+
certificate,
|
|
153
|
+
headers: response.headers
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
async getAppleCertificate(publicKeyUrl) {
|
|
157
|
+
if (!this.verifyPublicKeyUrl(publicKeyUrl)) {
|
|
158
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
159
|
+
}
|
|
160
|
+
if (this.cache[publicKeyUrl]) {
|
|
161
|
+
return this.cache[publicKeyUrl];
|
|
162
|
+
}
|
|
163
|
+
const {
|
|
164
|
+
certificate,
|
|
165
|
+
headers
|
|
166
|
+
} = await this.fetchCertificate(publicKeyUrl);
|
|
167
|
+
const cacheControl = headers.get('cache-control');
|
|
168
|
+
const expire = cacheControl?.match(/max-age=([0-9]+)/);
|
|
169
|
+
this.verifyPublicKeyIssuer(certificate, publicKeyUrl);
|
|
170
|
+
if (expire) {
|
|
171
|
+
this.cache[publicKeyUrl] = certificate;
|
|
172
|
+
setTimeout(() => delete this.cache[publicKeyUrl], parseInt(expire[1], 10) * 1000);
|
|
173
|
+
}
|
|
174
|
+
return certificate;
|
|
175
|
+
}
|
|
176
|
+
verifyPublicKeyIssuer(cert, publicKeyUrl) {
|
|
177
|
+
const publicKeyCert = _nodeForge.pki.certificateFromPem(cert);
|
|
178
|
+
if (!this.ca.cert) {
|
|
179
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Root certificate is invalid or missing.');
|
|
180
|
+
}
|
|
181
|
+
if (!this.ca.cert.verify(publicKeyCert)) {
|
|
182
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
verifySignature(publicKey, authData) {
|
|
186
|
+
const bundleId = this.bundleId || this.enableInsecureAuth && authData.bundleId;
|
|
187
|
+
const verifier = _crypto.default.createVerify('sha256');
|
|
188
|
+
verifier.update(Buffer.from(authData.id, 'utf8'));
|
|
189
|
+
verifier.update(Buffer.from(bundleId, 'utf8'));
|
|
190
|
+
verifier.update(this.convertTimestampToBigEndian(authData.timestamp));
|
|
191
|
+
verifier.update(Buffer.from(authData.salt, 'base64'));
|
|
192
|
+
if (!verifier.verify(publicKey, authData.signature, 'base64')) {
|
|
193
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid signature.');
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
async validateAuthData(authData) {
|
|
197
|
+
const requiredKeys = ['id', 'publicKeyUrl', 'timestamp', 'signature', 'salt'];
|
|
198
|
+
if (this.enableInsecureAuth) {
|
|
199
|
+
requiredKeys.push('bundleId');
|
|
200
|
+
}
|
|
201
|
+
for (const key of requiredKeys) {
|
|
202
|
+
if (!authData[key]) {
|
|
203
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `AuthData ${key} is missing.`);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
await this.loadingPromise;
|
|
207
|
+
const publicKey = await this.getAppleCertificate(authData.publicKeyUrl);
|
|
208
|
+
this.verifySignature(publicKey, authData);
|
|
209
|
+
}
|
|
210
|
+
convertTimestampToBigEndian(timestamp) {
|
|
211
|
+
const buffer = Buffer.alloc(8);
|
|
212
|
+
buffer.writeBigUInt64BE(BigInt(timestamp));
|
|
213
|
+
return buffer;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
var _default = exports.default = new GameCenterAuth();
|
|
217
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_crypto","_interopRequireDefault","require","_nodeForge","_AuthAdapter","e","__esModule","default","GameCenterAuth","AuthAdapter","constructor","ca","cert","url","cache","bundleId","validateOptions","options","Error","loadingPromise","loadCertificate","enableInsecureAuth","rootCertificateUrl","certificate","headers","fetchCertificate","get","parseInt","Parse","OBJECT_NOT_FOUND","pki","certificateFromPem","verifyPublicKeyUrl","publicKeyUrl","regex","test","response","fetch","ok","contentType","isPem","includes","text","data","arrayBuffer","binaryData","Buffer","from","asn1Cert","asn1","fromDer","toString","forgeCert","certificateFromAsn1","certificateToPem","getAppleCertificate","cacheControl","expire","match","verifyPublicKeyIssuer","setTimeout","publicKeyCert","verify","verifySignature","publicKey","authData","verifier","crypto","createVerify","update","id","convertTimestampToBigEndian","timestamp","salt","signature","validateAuthData","requiredKeys","push","key","buffer","alloc","writeBigUInt64BE","BigInt","_default","exports"],"sources":["../../../src/Adapters/Auth/gcenter.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Apple Game Center.\n *\n * @class AppleGameCenterAdapter\n * @param {Object} options - Configuration options for the adapter.\n * @param {string} options.bundleId - Your Apple Game Center bundle ID. 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.id - The user ID obtained from Apple Game Center.\n * @param {string} authData.publicKeyUrl - The public key URL obtained from Apple Game Center.\n * @param {string} authData.timestamp - The timestamp obtained from Apple Game Center.\n * @param {string} authData.signature - The signature obtained from Apple Game Center.\n * @param {string} authData.salt - The salt obtained from Apple Game Center.\n * @param {string} [authData.bundleId] - **[DEPRECATED]** The bundle ID obtained from Apple Game Center (required for insecure authentication).\n *\n * @description\n * ## Parse Server Configuration\n * The following `authData` fields are required:\n * `id`, `publicKeyUrl`, `timestamp`, `signature`, and `salt`. These fields are validated against the configured `bundleId` for additional security.\n *\n * To configure Parse Server for Apple Game Center authentication, use the following structure:\n * ```json\n * {\n *  \"auth\": {\n *    \"gcenter\": {\n *     \"bundleId\": \"com.valid.app\"\n *  }\n * }\n * ```\n *\n * ## Insecure Authentication (Not Recommended)\n * The following `authData` fields are required for insecure authentication:\n * `id`, `publicKeyUrl`, `timestamp`, `signature`, `salt`, and `bundleId` (**[DEPRECATED]**). This flow is insecure and poses potential security risks.\n *\n * To configure Parse Server for insecure authentication, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *    \"gcenter\": {\n *      \"enableInsecureAuth\": true\n *   }\n * }\n * ```\n *\n * ### Deprecation Notice\n * The `enableInsecureAuth` option and `authData.bundleId` parameter are deprecated and may be removed in future releases. Use secure authentication with the `bundleId` configured in the `options` object instead.\n *\n *\n * @example <caption>Secure Authentication Example</caption>\n * // Example authData for secure authentication:\n * const authData = {\n *   gcenter: {\n *     id: \"1234567\",\n *     publicKeyUrl: \"https://valid.apple.com/public/timeout.cer\",\n *     timestamp: 1460981421303,\n *     salt: \"saltST==\",\n *     signature: \"PoDwf39DCN464B49jJCU0d9Y0J\"\n *   }\n * };\n *\n * @example <caption>Insecure Authentication Example (Not Recommended)</caption>\n * // Example authData for insecure authentication:\n * const authData = {\n *   gcenter: {\n *     id: \"1234567\",\n *     publicKeyUrl: \"https://valid.apple.com/public/timeout.cer\",\n *     timestamp: 1460981421303,\n *     salt: \"saltST==\",\n *     signature: \"PoDwf39DCN464B49jJCU0d9Y0J\",\n *     bundleId: \"com.valid.app\" // Deprecated.\n *   }\n * };\n *\n * @see {@link https://developer.apple.com/documentation/gamekit/gklocalplayer/3516283-fetchitems Apple Game Center Documentation}\n */\n/* global BigInt */\n\nimport crypto from 'crypto';\nimport { asn1, pki } from 'node-forge';\nimport AuthAdapter from './AuthAdapter';\nclass GameCenterAuth extends AuthAdapter {\n  constructor() {\n    super();\n    this.ca = { cert: null, url: null };\n    this.cache = {};\n    this.bundleId = '';\n  }\n\n  validateOptions(options) {\n    if (!options) {\n      throw new Error('Game center auth options are required.');\n    }\n\n    if (!this.loadingPromise) {\n      this.loadingPromise = this.loadCertificate(options);\n    }\n\n    this.enableInsecureAuth = options.enableInsecureAuth;\n    this.bundleId = options.bundleId;\n\n    if (!this.enableInsecureAuth && !this.bundleId) {\n      throw new Error('bundleId is required for secure auth.');\n    }\n  }\n\n  async loadCertificate(options) {\n    const rootCertificateUrl =\n      options.rootCertificateUrl ||\n      'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';\n\n    if (this.ca.url === rootCertificateUrl) {\n      return rootCertificateUrl;\n    }\n\n    const { certificate, headers } = await this.fetchCertificate(rootCertificateUrl);\n\n    if (\n      headers.get('content-type') !== 'application/x-pem-file' ||\n      !headers.get('content-length') ||\n      parseInt(headers.get('content-length'), 10) > 10000\n    ) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid rootCertificateURL.');\n    }\n\n    this.ca.cert = pki.certificateFromPem(certificate);\n    this.ca.url = rootCertificateUrl;\n\n    return rootCertificateUrl;\n  }\n\n  verifyPublicKeyUrl(publicKeyUrl) {\n    const regex = /^https:\\/\\/(?:[-_A-Za-z0-9]+\\.){0,}apple\\.com\\/.*\\.cer$/;\n    return regex.test(publicKeyUrl);\n  }\n\n  async fetchCertificate(url) {\n    const response = await fetch(url);\n    if (!response.ok) {\n      throw new Error(`Failed to fetch certificate: ${url}`);\n    }\n\n    const contentType = response.headers.get('content-type');\n    const isPem = contentType?.includes('application/x-pem-file');\n\n    if (isPem) {\n      const certificate = await response.text();\n      return { certificate, headers: response.headers };\n    }\n\n    const data = await response.arrayBuffer();\n    const binaryData = Buffer.from(data);\n\n    const asn1Cert = asn1.fromDer(binaryData.toString('binary'));\n    const forgeCert = pki.certificateFromAsn1(asn1Cert);\n    const certificate = pki.certificateToPem(forgeCert);\n\n    return { certificate, headers: response.headers };\n  }\n\n  async getAppleCertificate(publicKeyUrl) {\n    if (!this.verifyPublicKeyUrl(publicKeyUrl)) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);\n    }\n\n    if (this.cache[publicKeyUrl]) {\n      return this.cache[publicKeyUrl];\n    }\n\n    const { certificate, headers } = await this.fetchCertificate(publicKeyUrl);\n    const cacheControl = headers.get('cache-control');\n    const expire = cacheControl?.match(/max-age=([0-9]+)/);\n\n    this.verifyPublicKeyIssuer(certificate, publicKeyUrl);\n\n    if (expire) {\n      this.cache[publicKeyUrl] = certificate;\n      setTimeout(() => delete this.cache[publicKeyUrl], parseInt(expire[1], 10) * 1000);\n    }\n\n    return certificate;\n  }\n\n  verifyPublicKeyIssuer(cert, publicKeyUrl) {\n    const publicKeyCert = pki.certificateFromPem(cert);\n\n    if (!this.ca.cert) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        'Root certificate is invalid or missing.'\n      );\n    }\n\n    if (!this.ca.cert.verify(publicKeyCert)) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);\n    }\n  }\n\n  verifySignature(publicKey, authData) {\n    const bundleId = this.bundleId || (this.enableInsecureAuth && authData.bundleId);\n\n    const verifier = crypto.createVerify('sha256');\n    verifier.update(Buffer.from(authData.id, 'utf8'));\n    verifier.update(Buffer.from(bundleId, 'utf8'));\n    verifier.update(this.convertTimestampToBigEndian(authData.timestamp));\n    verifier.update(Buffer.from(authData.salt, 'base64'));\n\n    if (!verifier.verify(publicKey, authData.signature, 'base64')) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid signature.');\n    }\n  }\n\n  async validateAuthData(authData) {\n\n    const requiredKeys = ['id', 'publicKeyUrl', 'timestamp', 'signature', 'salt'];\n    if (this.enableInsecureAuth) {\n      requiredKeys.push('bundleId');\n    }\n\n    for (const key of requiredKeys) {\n      if (!authData[key]) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `AuthData ${key} is missing.`);\n      }\n    }\n\n    await this.loadingPromise;\n\n    const publicKey = await this.getAppleCertificate(authData.publicKeyUrl);\n    this.verifySignature(publicKey, authData);\n  }\n\n  convertTimestampToBigEndian(timestamp) {\n    const buffer = Buffer.alloc(8);\n    buffer.writeBigUInt64BE(BigInt(timestamp));\n    return buffer;\n  }\n}\n\nexport default new GameCenterAuth();\n"],"mappings":"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","ignoreList":[]}
|