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,9 +1,53 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Parse Server authentication adapter for Janrain Capture API.
|
|
5
|
+
*
|
|
6
|
+
* @class JanrainCapture
|
|
7
|
+
* @param {Object} options - The adapter configuration options.
|
|
8
|
+
* @param {String} options.janrain_capture_host - The Janrain Capture API host.
|
|
9
|
+
*
|
|
10
|
+
* @param {Object} authData - The authentication data provided by the client.
|
|
11
|
+
* @param {String} authData.id - The Janrain Capture user ID.
|
|
12
|
+
* @param {String} authData.access_token - The Janrain Capture access token.
|
|
13
|
+
*
|
|
14
|
+
* @description
|
|
15
|
+
* ## Parse Server Configuration
|
|
16
|
+
* To configure Parse Server for Janrain Capture authentication, use the following structure:
|
|
17
|
+
* ```json
|
|
18
|
+
* {
|
|
19
|
+
* "auth": {
|
|
20
|
+
* "janrain": {
|
|
21
|
+
* "janrain_capture_host": "your-janrain-capture-host"
|
|
22
|
+
* }
|
|
23
|
+
* }
|
|
24
|
+
* }
|
|
25
|
+
* ```
|
|
26
|
+
*
|
|
27
|
+
* The adapter requires the following `authData` fields:
|
|
28
|
+
* - `id`: The Janrain Capture user ID.
|
|
29
|
+
* - `access_token`: An authorized Janrain Capture access token for the user.
|
|
30
|
+
*
|
|
31
|
+
* ## Auth Payload Example
|
|
32
|
+
* ```json
|
|
33
|
+
* {
|
|
34
|
+
* "janrain": {
|
|
35
|
+
* "id": "user's Janrain Capture ID as a string",
|
|
36
|
+
* "access_token": "an authorized Janrain Capture access token for the user"
|
|
37
|
+
* }
|
|
38
|
+
* }
|
|
39
|
+
* ```
|
|
40
|
+
*
|
|
41
|
+
* ## Notes
|
|
42
|
+
* Parse Server validates the provided `authData` using the Janrain Capture API.
|
|
43
|
+
*
|
|
44
|
+
* @see {@link https://docs.janrain.com/api/registration/entity/#entity Janrain Capture API Documentation}
|
|
45
|
+
*/
|
|
2
46
|
|
|
3
47
|
// Helper functions for accessing the Janrain Capture API.
|
|
4
|
-
var https = require('https');
|
|
5
48
|
var Parse = require('parse/node').Parse;
|
|
6
49
|
var querystring = require('querystring');
|
|
50
|
+
const httpsRequest = require('./httpsRequest');
|
|
7
51
|
|
|
8
52
|
// Returns a promise that fulfills iff this user id is valid.
|
|
9
53
|
function validateAuthData(authData, options) {
|
|
@@ -25,32 +69,17 @@ function validateAppId() {
|
|
|
25
69
|
|
|
26
70
|
// A promisey wrapper for api requests
|
|
27
71
|
function request(host, access_token) {
|
|
28
|
-
|
|
29
72
|
var query_string_data = querystring.stringify({
|
|
30
|
-
|
|
31
|
-
|
|
73
|
+
access_token: access_token,
|
|
74
|
+
attribute_name: 'uuid' // we only need to pull the uuid for this access token to make sure it matches
|
|
32
75
|
});
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
host: host,
|
|
37
|
-
path: '/entity?' + query_string_data
|
|
38
|
-
}, function (res) {
|
|
39
|
-
var data = '';
|
|
40
|
-
res.on('data', function (chunk) {
|
|
41
|
-
data += chunk;
|
|
42
|
-
});
|
|
43
|
-
res.on('end', function () {
|
|
44
|
-
resolve(JSON.parse(data));
|
|
45
|
-
});
|
|
46
|
-
}).on('error', function () {
|
|
47
|
-
reject('Failed to validate this access token with Janrain capture.');
|
|
48
|
-
});
|
|
76
|
+
return httpsRequest.get({
|
|
77
|
+
host: host,
|
|
78
|
+
path: '/entity?' + query_string_data
|
|
49
79
|
});
|
|
50
80
|
}
|
|
51
|
-
|
|
52
81
|
module.exports = {
|
|
53
82
|
validateAppId: validateAppId,
|
|
54
83
|
validateAuthData: validateAuthData
|
|
55
84
|
};
|
|
56
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
85
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJxdWVyeXN0cmluZyIsImh0dHBzUmVxdWVzdCIsInZhbGlkYXRlQXV0aERhdGEiLCJhdXRoRGF0YSIsIm9wdGlvbnMiLCJyZXF1ZXN0IiwiamFucmFpbl9jYXB0dXJlX2hvc3QiLCJhY2Nlc3NfdG9rZW4iLCJ0aGVuIiwiZGF0YSIsInN0YXQiLCJyZXN1bHQiLCJpZCIsIkVycm9yIiwiT0JKRUNUX05PVF9GT1VORCIsInZhbGlkYXRlQXBwSWQiLCJQcm9taXNlIiwicmVzb2x2ZSIsImhvc3QiLCJxdWVyeV9zdHJpbmdfZGF0YSIsInN0cmluZ2lmeSIsImF0dHJpYnV0ZV9uYW1lIiwiZ2V0IiwicGF0aCIsIm1vZHVsZSIsImV4cG9ydHMiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvQWRhcHRlcnMvQXV0aC9qYW5yYWluY2FwdHVyZS5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFBhcnNlIFNlcnZlciBhdXRoZW50aWNhdGlvbiBhZGFwdGVyIGZvciBKYW5yYWluIENhcHR1cmUgQVBJLlxuICpcbiAqIEBjbGFzcyBKYW5yYWluQ2FwdHVyZVxuICogQHBhcmFtIHtPYmplY3R9IG9wdGlvbnMgLSBUaGUgYWRhcHRlciBjb25maWd1cmF0aW9uIG9wdGlvbnMuXG4gKiBAcGFyYW0ge1N0cmluZ30gb3B0aW9ucy5qYW5yYWluX2NhcHR1cmVfaG9zdCAtIFRoZSBKYW5yYWluIENhcHR1cmUgQVBJIGhvc3QuXG4gKlxuICogQHBhcmFtIHtPYmplY3R9IGF1dGhEYXRhIC0gVGhlIGF1dGhlbnRpY2F0aW9uIGRhdGEgcHJvdmlkZWQgYnkgdGhlIGNsaWVudC5cbiAqIEBwYXJhbSB7U3RyaW5nfSBhdXRoRGF0YS5pZCAtIFRoZSBKYW5yYWluIENhcHR1cmUgdXNlciBJRC5cbiAqIEBwYXJhbSB7U3RyaW5nfSBhdXRoRGF0YS5hY2Nlc3NfdG9rZW4gLSBUaGUgSmFucmFpbiBDYXB0dXJlIGFjY2VzcyB0b2tlbi5cbiAqXG4gKiBAZGVzY3JpcHRpb25cbiAqICMjIFBhcnNlIFNlcnZlciBDb25maWd1cmF0aW9uXG4gKiBUbyBjb25maWd1cmUgUGFyc2UgU2VydmVyIGZvciBKYW5yYWluIENhcHR1cmUgYXV0aGVudGljYXRpb24sIHVzZSB0aGUgZm9sbG93aW5nIHN0cnVjdHVyZTpcbiAqIGBgYGpzb25cbiAqIHtcbiAqICAgXCJhdXRoXCI6IHtcbiAqICAgICBcImphbnJhaW5cIjoge1xuICogICAgICAgXCJqYW5yYWluX2NhcHR1cmVfaG9zdFwiOiBcInlvdXItamFucmFpbi1jYXB0dXJlLWhvc3RcIlxuICogICAgIH1cbiAqICAgfVxuICogfVxuICogYGBgXG4gKlxuICogVGhlIGFkYXB0ZXIgcmVxdWlyZXMgdGhlIGZvbGxvd2luZyBgYXV0aERhdGFgIGZpZWxkczpcbiAqIC0gYGlkYDogVGhlIEphbnJhaW4gQ2FwdHVyZSB1c2VyIElELlxuICogLSBgYWNjZXNzX3Rva2VuYDogQW4gYXV0aG9yaXplZCBKYW5yYWluIENhcHR1cmUgYWNjZXNzIHRva2VuIGZvciB0aGUgdXNlci5cbiAqXG4gKiAjIyBBdXRoIFBheWxvYWQgRXhhbXBsZVxuICogYGBganNvblxuICoge1xuICogICBcImphbnJhaW5cIjoge1xuICogICAgIFwiaWRcIjogXCJ1c2VyJ3MgSmFucmFpbiBDYXB0dXJlIElEIGFzIGEgc3RyaW5nXCIsXG4gKiAgICAgXCJhY2Nlc3NfdG9rZW5cIjogXCJhbiBhdXRob3JpemVkIEphbnJhaW4gQ2FwdHVyZSBhY2Nlc3MgdG9rZW4gZm9yIHRoZSB1c2VyXCJcbiAqICAgfVxuICogfVxuICogYGBgXG4gKlxuICogIyMgTm90ZXNcbiAqIFBhcnNlIFNlcnZlciB2YWxpZGF0ZXMgdGhlIHByb3ZpZGVkIGBhdXRoRGF0YWAgdXNpbmcgdGhlIEphbnJhaW4gQ2FwdHVyZSBBUEkuXG4gKlxuICogQHNlZSB7QGxpbmsgaHR0cHM6Ly9kb2NzLmphbnJhaW4uY29tL2FwaS9yZWdpc3RyYXRpb24vZW50aXR5LyNlbnRpdHkgSmFucmFpbiBDYXB0dXJlIEFQSSBEb2N1bWVudGF0aW9ufVxuICovXG5cblxuLy8gSGVscGVyIGZ1bmN0aW9ucyBmb3IgYWNjZXNzaW5nIHRoZSBKYW5yYWluIENhcHR1cmUgQVBJLlxudmFyIFBhcnNlID0gcmVxdWlyZSgncGFyc2Uvbm9kZScpLlBhcnNlO1xudmFyIHF1ZXJ5c3RyaW5nID0gcmVxdWlyZSgncXVlcnlzdHJpbmcnKTtcbmNvbnN0IGh0dHBzUmVxdWVzdCA9IHJlcXVpcmUoJy4vaHR0cHNSZXF1ZXN0Jyk7XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEsIG9wdGlvbnMpIHtcbiAgcmV0dXJuIHJlcXVlc3Qob3B0aW9ucy5qYW5yYWluX2NhcHR1cmVfaG9zdCwgYXV0aERhdGEuYWNjZXNzX3Rva2VuKS50aGVuKGRhdGEgPT4ge1xuICAgIC8vc3VjY2Vzc2Z1bCByZXNwb25zZSB3aWxsIGhhdmUgYSBcInN0YXRcIiAoc3RhdHVzKSBvZiAnb2snIGFuZCBhIHJlc3VsdCBub2RlIHRoYXQgc3RvcmVzIHRoZSB1dWlkLCBiZWNhdXNlIHRoYXQncyBhbGwgd2UgYXNrZWQgZm9yXG4gICAgLy9zZWU6IGh0dHBzOi8vZG9jcy5qYW5yYWluLmNvbS9hcGkvcmVnaXN0cmF0aW9uL2VudGl0eS8jZW50aXR5XG4gICAgaWYgKGRhdGEgJiYgZGF0YS5zdGF0ID09ICdvaycgJiYgZGF0YS5yZXN1bHQgPT0gYXV0aERhdGEuaWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhyb3cgbmV3IFBhcnNlLkVycm9yKFxuICAgICAgUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCxcbiAgICAgICdKYW5yYWluIGNhcHR1cmUgYXV0aCBpcyBpbnZhbGlkIGZvciB0aGlzIHVzZXIuJ1xuICAgICk7XG4gIH0pO1xufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIC8vbm8tb3BcbiAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpO1xufVxuXG4vLyBBIHByb21pc2V5IHdyYXBwZXIgZm9yIGFwaSByZXF1ZXN0c1xuZnVuY3Rpb24gcmVxdWVzdChob3N0LCBhY2Nlc3NfdG9rZW4pIHtcbiAgdmFyIHF1ZXJ5X3N0cmluZ19kYXRhID0gcXVlcnlzdHJpbmcuc3RyaW5naWZ5KHtcbiAgICBhY2Nlc3NfdG9rZW46IGFjY2Vzc190b2tlbixcbiAgICBhdHRyaWJ1dGVfbmFtZTogJ3V1aWQnLCAvLyB3ZSBvbmx5IG5lZWQgdG8gcHVsbCB0aGUgdXVpZCBmb3IgdGhpcyBhY2Nlc3MgdG9rZW4gdG8gbWFrZSBzdXJlIGl0IG1hdGNoZXNcbiAgfSk7XG5cbiAgcmV0dXJuIGh0dHBzUmVxdWVzdC5nZXQoeyBob3N0OiBob3N0LCBwYXRoOiAnL2VudGl0eT8nICsgcXVlcnlfc3RyaW5nX2RhdGEgfSk7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICB2YWxpZGF0ZUFwcElkOiB2YWxpZGF0ZUFwcElkLFxuICB2YWxpZGF0ZUF1dGhEYXRhOiB2YWxpZGF0ZUF1dGhEYXRhLFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFHQTtBQUNBLElBQUlBLEtBQUssR0FBR0MsT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDRCxLQUFLO0FBQ3ZDLElBQUlFLFdBQVcsR0FBR0QsT0FBTyxDQUFDLGFBQWEsQ0FBQztBQUN4QyxNQUFNRSxZQUFZLEdBQUdGLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQzs7QUFFOUM7QUFDQSxTQUFTRyxnQkFBZ0JBLENBQUNDLFFBQVEsRUFBRUMsT0FBTyxFQUFFO0VBQzNDLE9BQU9DLE9BQU8sQ0FBQ0QsT0FBTyxDQUFDRSxvQkFBb0IsRUFBRUgsUUFBUSxDQUFDSSxZQUFZLENBQUMsQ0FBQ0MsSUFBSSxDQUFDQyxJQUFJLElBQUk7SUFDL0U7SUFDQTtJQUNBLElBQUlBLElBQUksSUFBSUEsSUFBSSxDQUFDQyxJQUFJLElBQUksSUFBSSxJQUFJRCxJQUFJLENBQUNFLE1BQU0sSUFBSVIsUUFBUSxDQUFDUyxFQUFFLEVBQUU7TUFDM0Q7SUFDRjtJQUNBLE1BQU0sSUFBSWQsS0FBSyxDQUFDZSxLQUFLLENBQ25CZixLQUFLLENBQUNlLEtBQUssQ0FBQ0MsZ0JBQWdCLEVBQzVCLGdEQUNGLENBQUM7RUFDSCxDQUFDLENBQUM7QUFDSjs7QUFFQTtBQUNBLFNBQVNDLGFBQWFBLENBQUEsRUFBRztFQUN2QjtFQUNBLE9BQU9DLE9BQU8sQ0FBQ0MsT0FBTyxDQUFDLENBQUM7QUFDMUI7O0FBRUE7QUFDQSxTQUFTWixPQUFPQSxDQUFDYSxJQUFJLEVBQUVYLFlBQVksRUFBRTtFQUNuQyxJQUFJWSxpQkFBaUIsR0FBR25CLFdBQVcsQ0FBQ29CLFNBQVMsQ0FBQztJQUM1Q2IsWUFBWSxFQUFFQSxZQUFZO0lBQzFCYyxjQUFjLEVBQUUsTUFBTSxDQUFFO0VBQzFCLENBQUMsQ0FBQztFQUVGLE9BQU9wQixZQUFZLENBQUNxQixHQUFHLENBQUM7SUFBRUosSUFBSSxFQUFFQSxJQUFJO0lBQUVLLElBQUksRUFBRSxVQUFVLEdBQUdKO0VBQWtCLENBQUMsQ0FBQztBQUMvRTtBQUVBSyxNQUFNLENBQUNDLE9BQU8sR0FBRztFQUNmVixhQUFhLEVBQUVBLGFBQWE7RUFDNUJiLGdCQUFnQixFQUFFQTtBQUNwQixDQUFDIiwiaWdub3JlTGlzdCI6W119
|
|
@@ -1,13 +1,22 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
|
+
var _Config = _interopRequireDefault(require("../../Config"));
|
|
4
|
+
var _Deprecator = _interopRequireDefault(require("../../Deprecator/Deprecator"));
|
|
5
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
3
6
|
// Helper functions for accessing the Janrain Engage API.
|
|
4
|
-
var
|
|
7
|
+
var httpsRequest = require('./httpsRequest');
|
|
5
8
|
var Parse = require('parse/node').Parse;
|
|
6
9
|
var querystring = require('querystring');
|
|
7
|
-
|
|
8
10
|
// Returns a promise that fulfills iff this user id is valid.
|
|
9
11
|
function validateAuthData(authData, options) {
|
|
10
|
-
|
|
12
|
+
const config = _Config.default.get(Parse.applicationId);
|
|
13
|
+
_Deprecator.default.logRuntimeDeprecation({
|
|
14
|
+
usage: 'janrainengage adapter'
|
|
15
|
+
});
|
|
16
|
+
if (!config?.auth?.janrainengage?.enableInsecureAuth || !config.enableInsecureAuthAdapters) {
|
|
17
|
+
throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'janrainengage adapter only works with enableInsecureAuth: true');
|
|
18
|
+
}
|
|
19
|
+
return apiRequest(options.api_key, authData.auth_token).then(data => {
|
|
11
20
|
//successful response will have a "stat" (status) of 'ok' and a profile node with an identifier
|
|
12
21
|
//see: http://developers.janrain.com/overview/social-login/identity-providers/user-profile-data/#normalized-user-profile-data
|
|
13
22
|
if (data && data.stat == 'ok' && data.profile.identifier == authData.id) {
|
|
@@ -24,14 +33,12 @@ function validateAppId() {
|
|
|
24
33
|
}
|
|
25
34
|
|
|
26
35
|
// A promisey wrapper for api requests
|
|
27
|
-
function
|
|
28
|
-
|
|
36
|
+
function apiRequest(api_key, auth_token) {
|
|
29
37
|
var post_data = querystring.stringify({
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
38
|
+
token: auth_token,
|
|
39
|
+
apiKey: api_key,
|
|
40
|
+
format: 'json'
|
|
33
41
|
});
|
|
34
|
-
|
|
35
42
|
var post_options = {
|
|
36
43
|
host: 'rpxnow.com',
|
|
37
44
|
path: '/api/v2/auth_info',
|
|
@@ -41,34 +48,10 @@ function request(api_key, auth_token) {
|
|
|
41
48
|
'Content-Length': post_data.length
|
|
42
49
|
}
|
|
43
50
|
};
|
|
44
|
-
|
|
45
|
-
return new Promise(function (resolve, reject) {
|
|
46
|
-
// Create the post request.
|
|
47
|
-
var post_req = https.request(post_options, function (res) {
|
|
48
|
-
var data = '';
|
|
49
|
-
res.setEncoding('utf8');
|
|
50
|
-
// Append data as we receive it from the Janrain engage server.
|
|
51
|
-
res.on('data', function (d) {
|
|
52
|
-
data += d;
|
|
53
|
-
});
|
|
54
|
-
// Once we have all the data, we can parse it and return the data we want.
|
|
55
|
-
res.on('end', function () {
|
|
56
|
-
try {
|
|
57
|
-
data = JSON.parse(data);
|
|
58
|
-
} catch (e) {
|
|
59
|
-
return reject(e);
|
|
60
|
-
}
|
|
61
|
-
resolve(data);
|
|
62
|
-
});
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
post_req.write(post_data);
|
|
66
|
-
post_req.end();
|
|
67
|
-
});
|
|
51
|
+
return httpsRequest.request(post_options, post_data);
|
|
68
52
|
}
|
|
69
|
-
|
|
70
53
|
module.exports = {
|
|
71
54
|
validateAppId: validateAppId,
|
|
72
55
|
validateAuthData: validateAuthData
|
|
73
56
|
};
|
|
74
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
57
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfQ29uZmlnIiwiX2ludGVyb3BSZXF1aXJlRGVmYXVsdCIsInJlcXVpcmUiLCJfRGVwcmVjYXRvciIsImUiLCJfX2VzTW9kdWxlIiwiZGVmYXVsdCIsImh0dHBzUmVxdWVzdCIsIlBhcnNlIiwicXVlcnlzdHJpbmciLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJvcHRpb25zIiwiY29uZmlnIiwiQ29uZmlnIiwiZ2V0IiwiYXBwbGljYXRpb25JZCIsIkRlcHJlY2F0b3IiLCJsb2dSdW50aW1lRGVwcmVjYXRpb24iLCJ1c2FnZSIsImF1dGgiLCJqYW5yYWluZW5nYWdlIiwiZW5hYmxlSW5zZWN1cmVBdXRoIiwiZW5hYmxlSW5zZWN1cmVBdXRoQWRhcHRlcnMiLCJFcnJvciIsIklOVEVSTkFMX1NFUlZFUl9FUlJPUiIsImFwaVJlcXVlc3QiLCJhcGlfa2V5IiwiYXV0aF90b2tlbiIsInRoZW4iLCJkYXRhIiwic3RhdCIsInByb2ZpbGUiLCJpZGVudGlmaWVyIiwiaWQiLCJPQkpFQ1RfTk9UX0ZPVU5EIiwidmFsaWRhdGVBcHBJZCIsIlByb21pc2UiLCJyZXNvbHZlIiwicG9zdF9kYXRhIiwic3RyaW5naWZ5IiwidG9rZW4iLCJhcGlLZXkiLCJmb3JtYXQiLCJwb3N0X29wdGlvbnMiLCJob3N0IiwicGF0aCIsIm1ldGhvZCIsImhlYWRlcnMiLCJsZW5ndGgiLCJyZXF1ZXN0IiwibW9kdWxlIiwiZXhwb3J0cyJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BZGFwdGVycy9BdXRoL2phbnJhaW5lbmdhZ2UuanMiXSwic291cmNlc0NvbnRlbnQiOlsiLy8gSGVscGVyIGZ1bmN0aW9ucyBmb3IgYWNjZXNzaW5nIHRoZSBKYW5yYWluIEVuZ2FnZSBBUEkuXG52YXIgaHR0cHNSZXF1ZXN0ID0gcmVxdWlyZSgnLi9odHRwc1JlcXVlc3QnKTtcbnZhciBQYXJzZSA9IHJlcXVpcmUoJ3BhcnNlL25vZGUnKS5QYXJzZTtcbnZhciBxdWVyeXN0cmluZyA9IHJlcXVpcmUoJ3F1ZXJ5c3RyaW5nJyk7XG5pbXBvcnQgQ29uZmlnIGZyb20gJy4uLy4uL0NvbmZpZyc7XG5pbXBvcnQgRGVwcmVjYXRvciBmcm9tICcuLi8uLi9EZXByZWNhdG9yL0RlcHJlY2F0b3InO1xuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIHVzZXIgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUF1dGhEYXRhKGF1dGhEYXRhLCBvcHRpb25zKSB7XG4gIGNvbnN0IGNvbmZpZyA9IENvbmZpZy5nZXQoUGFyc2UuYXBwbGljYXRpb25JZCk7XG5cbiAgRGVwcmVjYXRvci5sb2dSdW50aW1lRGVwcmVjYXRpb24oeyB1c2FnZTogJ2phbnJhaW5lbmdhZ2UgYWRhcHRlcicgfSk7XG4gIGlmICghY29uZmlnPy5hdXRoPy5qYW5yYWluZW5nYWdlPy5lbmFibGVJbnNlY3VyZUF1dGggfHwgIWNvbmZpZy5lbmFibGVJbnNlY3VyZUF1dGhBZGFwdGVycykge1xuICAgIHRocm93IG5ldyBQYXJzZS5FcnJvcihQYXJzZS5FcnJvci5JTlRFUk5BTF9TRVJWRVJfRVJST1IsICdqYW5yYWluZW5nYWdlIGFkYXB0ZXIgb25seSB3b3JrcyB3aXRoIGVuYWJsZUluc2VjdXJlQXV0aDogdHJ1ZScpO1xuICB9XG5cbiAgcmV0dXJuIGFwaVJlcXVlc3Qob3B0aW9ucy5hcGlfa2V5LCBhdXRoRGF0YS5hdXRoX3Rva2VuKS50aGVuKGRhdGEgPT4ge1xuICAgIC8vc3VjY2Vzc2Z1bCByZXNwb25zZSB3aWxsIGhhdmUgYSBcInN0YXRcIiAoc3RhdHVzKSBvZiAnb2snIGFuZCBhIHByb2ZpbGUgbm9kZSB3aXRoIGFuIGlkZW50aWZpZXJcbiAgICAvL3NlZTogaHR0cDovL2RldmVsb3BlcnMuamFucmFpbi5jb20vb3ZlcnZpZXcvc29jaWFsLWxvZ2luL2lkZW50aXR5LXByb3ZpZGVycy91c2VyLXByb2ZpbGUtZGF0YS8jbm9ybWFsaXplZC11c2VyLXByb2ZpbGUtZGF0YVxuICAgIGlmIChkYXRhICYmIGRhdGEuc3RhdCA9PSAnb2snICYmIGRhdGEucHJvZmlsZS5pZGVudGlmaWVyID09IGF1dGhEYXRhLmlkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRocm93IG5ldyBQYXJzZS5FcnJvcihcbiAgICAgIFBhcnNlLkVycm9yLk9CSkVDVF9OT1RfRk9VTkQsXG4gICAgICAnSmFucmFpbiBlbmdhZ2UgYXV0aCBpcyBpbnZhbGlkIGZvciB0aGlzIHVzZXIuJ1xuICAgICk7XG4gIH0pO1xufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIC8vbm8tb3BcbiAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpO1xufVxuXG4vLyBBIHByb21pc2V5IHdyYXBwZXIgZm9yIGFwaSByZXF1ZXN0c1xuZnVuY3Rpb24gYXBpUmVxdWVzdChhcGlfa2V5LCBhdXRoX3Rva2VuKSB7XG4gIHZhciBwb3N0X2RhdGEgPSBxdWVyeXN0cmluZy5zdHJpbmdpZnkoe1xuICAgIHRva2VuOiBhdXRoX3Rva2VuLFxuICAgIGFwaUtleTogYXBpX2tleSxcbiAgICBmb3JtYXQ6ICdqc29uJyxcbiAgfSk7XG5cbiAgdmFyIHBvc3Rfb3B0aW9ucyA9IHtcbiAgICBob3N0OiAncnB4bm93LmNvbScsXG4gICAgcGF0aDogJy9hcGkvdjIvYXV0aF9pbmZvJyxcbiAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICBoZWFkZXJzOiB7XG4gICAgICAnQ29udGVudC1UeXBlJzogJ2FwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZCcsXG4gICAgICAnQ29udGVudC1MZW5ndGgnOiBwb3N0X2RhdGEubGVuZ3RoLFxuICAgIH0sXG4gIH07XG5cbiAgcmV0dXJuIGh0dHBzUmVxdWVzdC5yZXF1ZXN0KHBvc3Rfb3B0aW9ucywgcG9zdF9kYXRhKTtcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gIHZhbGlkYXRlQXBwSWQ6IHZhbGlkYXRlQXBwSWQsXG4gIHZhbGlkYXRlQXV0aERhdGE6IHZhbGlkYXRlQXV0aERhdGEsXG59O1xuIl0sIm1hcHBpbmdzIjoiOztBQUlBLElBQUFBLE9BQUEsR0FBQUMsc0JBQUEsQ0FBQUMsT0FBQTtBQUNBLElBQUFDLFdBQUEsR0FBQUYsc0JBQUEsQ0FBQUMsT0FBQTtBQUFxRCxTQUFBRCx1QkFBQUcsQ0FBQSxXQUFBQSxDQUFBLElBQUFBLENBQUEsQ0FBQUMsVUFBQSxHQUFBRCxDQUFBLEtBQUFFLE9BQUEsRUFBQUYsQ0FBQTtBQUxyRDtBQUNBLElBQUlHLFlBQVksR0FBR0wsT0FBTyxDQUFDLGdCQUFnQixDQUFDO0FBQzVDLElBQUlNLEtBQUssR0FBR04sT0FBTyxDQUFDLFlBQVksQ0FBQyxDQUFDTSxLQUFLO0FBQ3ZDLElBQUlDLFdBQVcsR0FBR1AsT0FBTyxDQUFDLGFBQWEsQ0FBQztBQUl4QztBQUNBLFNBQVNRLGdCQUFnQkEsQ0FBQ0MsUUFBUSxFQUFFQyxPQUFPLEVBQUU7RUFDM0MsTUFBTUMsTUFBTSxHQUFHQyxlQUFNLENBQUNDLEdBQUcsQ0FBQ1AsS0FBSyxDQUFDUSxhQUFhLENBQUM7RUFFOUNDLG1CQUFVLENBQUNDLHFCQUFxQixDQUFDO0lBQUVDLEtBQUssRUFBRTtFQUF3QixDQUFDLENBQUM7RUFDcEUsSUFBSSxDQUFDTixNQUFNLEVBQUVPLElBQUksRUFBRUMsYUFBYSxFQUFFQyxrQkFBa0IsSUFBSSxDQUFDVCxNQUFNLENBQUNVLDBCQUEwQixFQUFFO0lBQzFGLE1BQU0sSUFBSWYsS0FBSyxDQUFDZ0IsS0FBSyxDQUFDaEIsS0FBSyxDQUFDZ0IsS0FBSyxDQUFDQyxxQkFBcUIsRUFBRSxnRUFBZ0UsQ0FBQztFQUM1SDtFQUVBLE9BQU9DLFVBQVUsQ0FBQ2QsT0FBTyxDQUFDZSxPQUFPLEVBQUVoQixRQUFRLENBQUNpQixVQUFVLENBQUMsQ0FBQ0MsSUFBSSxDQUFDQyxJQUFJLElBQUk7SUFDbkU7SUFDQTtJQUNBLElBQUlBLElBQUksSUFBSUEsSUFBSSxDQUFDQyxJQUFJLElBQUksSUFBSSxJQUFJRCxJQUFJLENBQUNFLE9BQU8sQ0FBQ0MsVUFBVSxJQUFJdEIsUUFBUSxDQUFDdUIsRUFBRSxFQUFFO01BQ3ZFO0lBQ0Y7SUFDQSxNQUFNLElBQUkxQixLQUFLLENBQUNnQixLQUFLLENBQ25CaEIsS0FBSyxDQUFDZ0IsS0FBSyxDQUFDVyxnQkFBZ0IsRUFDNUIsK0NBQ0YsQ0FBQztFQUNILENBQUMsQ0FBQztBQUNKOztBQUVBO0FBQ0EsU0FBU0MsYUFBYUEsQ0FBQSxFQUFHO0VBQ3ZCO0VBQ0EsT0FBT0MsT0FBTyxDQUFDQyxPQUFPLENBQUMsQ0FBQztBQUMxQjs7QUFFQTtBQUNBLFNBQVNaLFVBQVVBLENBQUNDLE9BQU8sRUFBRUMsVUFBVSxFQUFFO0VBQ3ZDLElBQUlXLFNBQVMsR0FBRzlCLFdBQVcsQ0FBQytCLFNBQVMsQ0FBQztJQUNwQ0MsS0FBSyxFQUFFYixVQUFVO0lBQ2pCYyxNQUFNLEVBQUVmLE9BQU87SUFDZmdCLE1BQU0sRUFBRTtFQUNWLENBQUMsQ0FBQztFQUVGLElBQUlDLFlBQVksR0FBRztJQUNqQkMsSUFBSSxFQUFFLFlBQVk7SUFDbEJDLElBQUksRUFBRSxtQkFBbUI7SUFDekJDLE1BQU0sRUFBRSxNQUFNO0lBQ2RDLE9BQU8sRUFBRTtNQUNQLGNBQWMsRUFBRSxtQ0FBbUM7TUFDbkQsZ0JBQWdCLEVBQUVULFNBQVMsQ0FBQ1U7SUFDOUI7RUFDRixDQUFDO0VBRUQsT0FBTzFDLFlBQVksQ0FBQzJDLE9BQU8sQ0FBQ04sWUFBWSxFQUFFTCxTQUFTLENBQUM7QUFDdEQ7QUFFQVksTUFBTSxDQUFDQyxPQUFPLEdBQUc7RUFDZmhCLGFBQWEsRUFBRUEsYUFBYTtFQUM1QjFCLGdCQUFnQixFQUFFQTtBQUNwQixDQUFDIiwiaWdub3JlTGlzdCI6W119
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Parse Server authentication adapter for Keycloak.
|
|
5
|
+
*
|
|
6
|
+
* @class KeycloakAdapter
|
|
7
|
+
* @param {Object} options - The adapter configuration options.
|
|
8
|
+
* @param {Object} options.config - The Keycloak configuration object, typically loaded from a JSON file.
|
|
9
|
+
* @param {String} options.config.auth-server-url - The Keycloak authentication server URL.
|
|
10
|
+
* @param {String} options.config.realm - The Keycloak realm name.
|
|
11
|
+
* @param {String} options.config.client-id - The Keycloak client ID.
|
|
12
|
+
*
|
|
13
|
+
* @param {Object} authData - The authentication data provided by the client.
|
|
14
|
+
* @param {String} authData.access_token - The Keycloak access token retrieved during client authentication.
|
|
15
|
+
* @param {String} authData.id - The user ID retrieved from Keycloak during client authentication.
|
|
16
|
+
* @param {Array} [authData.roles] - The roles assigned to the user in Keycloak (optional).
|
|
17
|
+
* @param {Array} [authData.groups] - The groups assigned to the user in Keycloak (optional).
|
|
18
|
+
*
|
|
19
|
+
* @description
|
|
20
|
+
* ## Parse Server Configuration
|
|
21
|
+
* To configure Parse Server for Keycloak authentication, use the following structure:
|
|
22
|
+
* ```javascript
|
|
23
|
+
* {
|
|
24
|
+
* "auth": {
|
|
25
|
+
* "keycloak": {
|
|
26
|
+
* "config": require('./auth/keycloak.json')
|
|
27
|
+
* }
|
|
28
|
+
* }
|
|
29
|
+
* }
|
|
30
|
+
* ```
|
|
31
|
+
* Ensure the `keycloak.json` configuration file is generated from Keycloak's setup guide and includes:
|
|
32
|
+
* - `auth-server-url`: The Keycloak authentication server URL.
|
|
33
|
+
* - `realm`: The Keycloak realm name.
|
|
34
|
+
* - `client-id`: The Keycloak client ID.
|
|
35
|
+
*
|
|
36
|
+
* ## Auth Data
|
|
37
|
+
* The adapter requires the following `authData` fields:
|
|
38
|
+
* - `access_token`: The Keycloak access token retrieved during client authentication.
|
|
39
|
+
* - `id`: The user ID retrieved from Keycloak during client authentication.
|
|
40
|
+
* - `roles` (optional): The roles assigned to the user in Keycloak.
|
|
41
|
+
* - `groups` (optional): The groups assigned to the user in Keycloak.
|
|
42
|
+
*
|
|
43
|
+
* ## Auth Payload Example
|
|
44
|
+
* ### Example Auth Data
|
|
45
|
+
* ```json
|
|
46
|
+
* {
|
|
47
|
+
* "keycloak": {
|
|
48
|
+
* "access_token": "an authorized Keycloak access token for the user",
|
|
49
|
+
* "id": "user's Keycloak ID as a string",
|
|
50
|
+
* "roles": ["admin", "user"],
|
|
51
|
+
* "groups": ["group1", "group2"]
|
|
52
|
+
* }
|
|
53
|
+
* }
|
|
54
|
+
* ```
|
|
55
|
+
*
|
|
56
|
+
* ## Notes
|
|
57
|
+
* - Parse Server validates the provided `authData` by making a `userinfo` call to Keycloak and ensures the attributes match those returned by Keycloak.
|
|
58
|
+
*
|
|
59
|
+
* ## Keycloak Configuration
|
|
60
|
+
* To configure Keycloak, copy the JSON configuration file generated from Keycloak's setup guide:
|
|
61
|
+
* - [Keycloak Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/index.html#_javascript_adapter)
|
|
62
|
+
*
|
|
63
|
+
* Place the configuration file on your server, for example:
|
|
64
|
+
* - `auth/keycloak.json`
|
|
65
|
+
*
|
|
66
|
+
* For more information on Keycloak authentication, see:
|
|
67
|
+
* - [Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/)
|
|
68
|
+
* - [Server Administration Documentation](https://www.keycloak.org/docs/latest/server_admin/)
|
|
69
|
+
*/
|
|
70
|
+
|
|
71
|
+
const {
|
|
72
|
+
Parse
|
|
73
|
+
} = require('parse/node');
|
|
74
|
+
const httpsRequest = require('./httpsRequest');
|
|
75
|
+
const arraysEqual = (_arr1, _arr2) => {
|
|
76
|
+
if (!Array.isArray(_arr1) || !Array.isArray(_arr2) || _arr1.length !== _arr2.length) {
|
|
77
|
+
return false;
|
|
78
|
+
}
|
|
79
|
+
var arr1 = _arr1.concat().sort();
|
|
80
|
+
var arr2 = _arr2.concat().sort();
|
|
81
|
+
for (var i = 0; i < arr1.length; i++) {
|
|
82
|
+
if (arr1[i] !== arr2[i]) {
|
|
83
|
+
return false;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
return true;
|
|
87
|
+
};
|
|
88
|
+
const handleAuth = async ({
|
|
89
|
+
access_token,
|
|
90
|
+
id,
|
|
91
|
+
roles,
|
|
92
|
+
groups
|
|
93
|
+
} = {}, {
|
|
94
|
+
config
|
|
95
|
+
} = {}) => {
|
|
96
|
+
if (!(access_token && id)) {
|
|
97
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing access token and/or User id');
|
|
98
|
+
}
|
|
99
|
+
if (!config || !(config['auth-server-url'] && config['realm'])) {
|
|
100
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing keycloak configuration');
|
|
101
|
+
}
|
|
102
|
+
try {
|
|
103
|
+
const response = await httpsRequest.get({
|
|
104
|
+
host: config['auth-server-url'],
|
|
105
|
+
path: `/realms/${config['realm']}/protocol/openid-connect/userinfo`,
|
|
106
|
+
headers: {
|
|
107
|
+
Authorization: 'Bearer ' + access_token
|
|
108
|
+
}
|
|
109
|
+
});
|
|
110
|
+
if (response && response.data && response.data.sub == id && arraysEqual(response.data.roles, roles) && arraysEqual(response.data.groups, groups)) {
|
|
111
|
+
return;
|
|
112
|
+
}
|
|
113
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid authentication');
|
|
114
|
+
} catch (e) {
|
|
115
|
+
if (e instanceof Parse.Error) {
|
|
116
|
+
throw e;
|
|
117
|
+
}
|
|
118
|
+
const error = JSON.parse(e.text);
|
|
119
|
+
if (error.error_description) {
|
|
120
|
+
throw new Parse.Error(Parse.Error.HOSTING_ERROR, error.error_description);
|
|
121
|
+
} else {
|
|
122
|
+
throw new Parse.Error(Parse.Error.HOSTING_ERROR, 'Could not connect to the authentication server');
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
/*
|
|
128
|
+
@param {Object} authData: the client provided authData
|
|
129
|
+
@param {string} authData.access_token: the access_token retrieved from client authentication in Keycloak
|
|
130
|
+
@param {string} authData.id: the id retrieved from client authentication in Keycloak
|
|
131
|
+
@param {Array} authData.roles: the roles retrieved from client authentication in Keycloak
|
|
132
|
+
@param {Array} authData.groups: the groups retrieved from client authentication in Keycloak
|
|
133
|
+
@param {Object} options: additional options
|
|
134
|
+
@param {Object} options.config: the config object passed during Parse Server instantiation
|
|
135
|
+
*/
|
|
136
|
+
function validateAuthData(authData, options = {}) {
|
|
137
|
+
return handleAuth(authData, options);
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// Returns a promise that fulfills if this app id is valid.
|
|
141
|
+
function validateAppId() {
|
|
142
|
+
return Promise.resolve();
|
|
143
|
+
}
|
|
144
|
+
module.exports = {
|
|
145
|
+
validateAppId,
|
|
146
|
+
validateAuthData
|
|
147
|
+
};
|
|
148
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","httpsRequest","arraysEqual","_arr1","_arr2","Array","isArray","length","arr1","concat","sort","arr2","i","handleAuth","access_token","id","roles","groups","config","Error","OBJECT_NOT_FOUND","response","get","host","path","headers","Authorization","data","sub","e","error","JSON","parse","text","error_description","HOSTING_ERROR","validateAuthData","authData","options","validateAppId","Promise","resolve","module","exports"],"sources":["../../../src/Adapters/Auth/keycloak.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Keycloak.\n *\n * @class KeycloakAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {Object} options.config - The Keycloak configuration object, typically loaded from a JSON file.\n * @param {String} options.config.auth-server-url - The Keycloak authentication server URL.\n * @param {String} options.config.realm - The Keycloak realm name.\n * @param {String} options.config.client-id - The Keycloak client ID.\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {String} authData.access_token - The Keycloak access token retrieved during client authentication.\n * @param {String} authData.id - The user ID retrieved from Keycloak during client authentication.\n * @param {Array} [authData.roles] - The roles assigned to the user in Keycloak (optional).\n * @param {Array} [authData.groups] - The groups assigned to the user in Keycloak (optional).\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for Keycloak authentication, use the following structure:\n * ```javascript\n * {\n *   \"auth\": {\n *     \"keycloak\": {\n *       \"config\": require('./auth/keycloak.json')\n *     }\n *   }\n * }\n * ```\n * Ensure the `keycloak.json` configuration file is generated from Keycloak's setup guide and includes:\n * - `auth-server-url`: The Keycloak authentication server URL.\n * - `realm`: The Keycloak realm name.\n * - `client-id`: The Keycloak client ID.\n *\n * ## Auth Data\n * The adapter requires the following `authData` fields:\n * - `access_token`: The Keycloak access token retrieved during client authentication.\n * - `id`: The user ID retrieved from Keycloak during client authentication.\n * - `roles` (optional): The roles assigned to the user in Keycloak.\n * - `groups` (optional): The groups assigned to the user in Keycloak.\n *\n * ## Auth Payload Example\n * ### Example Auth Data\n * ```json\n * {\n *   \"keycloak\": {\n *     \"access_token\": \"an authorized Keycloak access token for the user\",\n *     \"id\": \"user's Keycloak ID as a string\",\n *     \"roles\": [\"admin\", \"user\"],\n *     \"groups\": [\"group1\", \"group2\"]\n *   }\n * }\n * ```\n *\n * ## Notes\n * - Parse Server validates the provided `authData` by making a `userinfo` call to Keycloak and ensures the attributes match those returned by Keycloak.\n *\n * ## Keycloak Configuration\n * To configure Keycloak, copy the JSON configuration file generated from Keycloak's setup guide:\n * - [Keycloak Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/index.html#_javascript_adapter)\n *\n * Place the configuration file on your server, for example:\n * - `auth/keycloak.json`\n *\n * For more information on Keycloak authentication, see:\n * - [Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/)\n * - [Server Administration Documentation](https://www.keycloak.org/docs/latest/server_admin/)\n */\n\nconst { Parse } = require('parse/node');\nconst httpsRequest = require('./httpsRequest');\n\nconst arraysEqual = (_arr1, _arr2) => {\n  if (!Array.isArray(_arr1) || !Array.isArray(_arr2) || _arr1.length !== _arr2.length) { return false; }\n\n  var arr1 = _arr1.concat().sort();\n  var arr2 = _arr2.concat().sort();\n\n  for (var i = 0; i < arr1.length; i++) {\n    if (arr1[i] !== arr2[i]) { return false; }\n  }\n\n  return true;\n};\n\nconst handleAuth = async ({ access_token, id, roles, groups } = {}, { config } = {}) => {\n  if (!(access_token && id)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing access token and/or User id');\n  }\n  if (!config || !(config['auth-server-url'] && config['realm'])) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing keycloak configuration');\n  }\n  try {\n    const response = await httpsRequest.get({\n      host: config['auth-server-url'],\n      path: `/realms/${config['realm']}/protocol/openid-connect/userinfo`,\n      headers: {\n        Authorization: 'Bearer ' + access_token,\n      },\n    });\n    if (\n      response &&\n      response.data &&\n      response.data.sub == id &&\n      arraysEqual(response.data.roles, roles) &&\n      arraysEqual(response.data.groups, groups)\n    ) {\n      return;\n    }\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid authentication');\n  } catch (e) {\n    if (e instanceof Parse.Error) {\n      throw e;\n    }\n    const error = JSON.parse(e.text);\n    if (error.error_description) {\n      throw new Parse.Error(Parse.Error.HOSTING_ERROR, error.error_description);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.HOSTING_ERROR,\n        'Could not connect to the authentication server'\n      );\n    }\n  }\n};\n\n/*\n  @param {Object} authData: the client provided authData\n  @param {string} authData.access_token: the access_token retrieved from client authentication in Keycloak\n  @param {string} authData.id: the id retrieved from client authentication in Keycloak\n  @param {Array}  authData.roles: the roles retrieved from client authentication in Keycloak\n  @param {Array}  authData.groups: the groups retrieved from client authentication in Keycloak\n  @param {Object} options: additional options\n  @param {Object} options.config: the config object passed during Parse Server instantiation\n*/\nfunction validateAuthData(authData, options = {}) {\n  return handleAuth(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,\n  validateAuthData,\n};\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Parse Server authentication adapter for LDAP.
|
|
5
|
+
*
|
|
6
|
+
* @class LDAP
|
|
7
|
+
* @param {Object} options - The adapter configuration options.
|
|
8
|
+
* @param {String} options.url - The LDAP server URL. Must start with `ldap://` or `ldaps://`.
|
|
9
|
+
* @param {String} options.suffix - The LDAP suffix for user distinguished names (DN).
|
|
10
|
+
* @param {String} [options.dn] - The distinguished name (DN) template for user authentication. Replace `{{id}}` with the username.
|
|
11
|
+
* @param {Object} [options.tlsOptions] - Options for LDAPS TLS connections.
|
|
12
|
+
* @param {String} [options.groupCn] - The common name (CN) of the group to verify user membership.
|
|
13
|
+
* @param {String} [options.groupFilter] - The LDAP search filter for groups, with `{{id}}` replaced by the username.
|
|
14
|
+
*
|
|
15
|
+
* @param {Object} authData - The authentication data provided by the client.
|
|
16
|
+
* @param {String} authData.id - The user's LDAP username.
|
|
17
|
+
* @param {String} authData.password - The user's LDAP password.
|
|
18
|
+
*
|
|
19
|
+
* @description
|
|
20
|
+
* ## Parse Server Configuration
|
|
21
|
+
* To configure Parse Server for LDAP authentication, use the following structure:
|
|
22
|
+
* ```javascript
|
|
23
|
+
* {
|
|
24
|
+
* auth: {
|
|
25
|
+
* ldap: {
|
|
26
|
+
* url: 'ldaps://ldap.example.com',
|
|
27
|
+
* suffix: 'ou=users,dc=example,dc=com',
|
|
28
|
+
* groupCn: 'admins',
|
|
29
|
+
* groupFilter: '(memberUid={{id}})',
|
|
30
|
+
* tlsOptions: {
|
|
31
|
+
* rejectUnauthorized: false
|
|
32
|
+
* }
|
|
33
|
+
* }
|
|
34
|
+
* }
|
|
35
|
+
* }
|
|
36
|
+
* ```
|
|
37
|
+
*
|
|
38
|
+
* ## Authentication Process
|
|
39
|
+
* 1. Validates the provided `authData` using an LDAP bind operation.
|
|
40
|
+
* 2. Optionally, verifies that the user belongs to a specific group by performing an LDAP search using the provided `groupCn` or `groupFilter`.
|
|
41
|
+
*
|
|
42
|
+
* ## Auth Payload
|
|
43
|
+
* The adapter requires the following `authData` fields:
|
|
44
|
+
* - `id`: The user's LDAP username.
|
|
45
|
+
* - `password`: The user's LDAP password.
|
|
46
|
+
*
|
|
47
|
+
* ### Example Auth Payload
|
|
48
|
+
* ```json
|
|
49
|
+
* {
|
|
50
|
+
* "ldap": {
|
|
51
|
+
* "id": "jdoe",
|
|
52
|
+
* "password": "password123"
|
|
53
|
+
* }
|
|
54
|
+
* }
|
|
55
|
+
* ```
|
|
56
|
+
*
|
|
57
|
+
* @example <caption>Configuration Example</caption>
|
|
58
|
+
* // Example Parse Server configuration:
|
|
59
|
+
* const config = {
|
|
60
|
+
* auth: {
|
|
61
|
+
* ldap: {
|
|
62
|
+
* url: 'ldaps://ldap.example.com',
|
|
63
|
+
* suffix: 'ou=users,dc=example,dc=com',
|
|
64
|
+
* groupCn: 'admins',
|
|
65
|
+
* groupFilter: '(memberUid={{id}})',
|
|
66
|
+
* tlsOptions: {
|
|
67
|
+
* rejectUnauthorized: false
|
|
68
|
+
* }
|
|
69
|
+
* }
|
|
70
|
+
* }
|
|
71
|
+
* };
|
|
72
|
+
*
|
|
73
|
+
* @see {@link https://ldap.com/ LDAP Basics}
|
|
74
|
+
* @see {@link https://ldap.com/ldap-filters/ LDAP Filters}
|
|
75
|
+
*/
|
|
76
|
+
|
|
77
|
+
const ldapjs = require('ldapjs');
|
|
78
|
+
const Parse = require('parse/node').Parse;
|
|
79
|
+
function validateAuthData(authData, options) {
|
|
80
|
+
if (!optionsAreValid(options)) {
|
|
81
|
+
return new Promise((_, reject) => {
|
|
82
|
+
reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP auth configuration missing'));
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
const clientOptions = options.url.startsWith('ldaps://') ? {
|
|
86
|
+
url: options.url,
|
|
87
|
+
tlsOptions: options.tlsOptions
|
|
88
|
+
} : {
|
|
89
|
+
url: options.url
|
|
90
|
+
};
|
|
91
|
+
const client = ldapjs.createClient(clientOptions);
|
|
92
|
+
const userCn = typeof options.dn === 'string' ? options.dn.replace('{{id}}', authData.id) : `uid=${authData.id},${options.suffix}`;
|
|
93
|
+
return new Promise((resolve, reject) => {
|
|
94
|
+
client.bind(userCn, authData.password, ldapError => {
|
|
95
|
+
delete authData.password;
|
|
96
|
+
if (ldapError) {
|
|
97
|
+
let error;
|
|
98
|
+
switch (ldapError.code) {
|
|
99
|
+
case 49:
|
|
100
|
+
error = new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LDAP: Wrong username or password');
|
|
101
|
+
break;
|
|
102
|
+
case 'DEPTH_ZERO_SELF_SIGNED_CERT':
|
|
103
|
+
error = new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LDAPS: Certificate mismatch');
|
|
104
|
+
break;
|
|
105
|
+
default:
|
|
106
|
+
error = new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LDAP: Somthing went wrong (' + ldapError.code + ')');
|
|
107
|
+
}
|
|
108
|
+
reject(error);
|
|
109
|
+
client.destroy(ldapError);
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
if (typeof options.groupCn === 'string' && typeof options.groupFilter === 'string') {
|
|
113
|
+
searchForGroup(client, options, authData.id, resolve, reject);
|
|
114
|
+
} else {
|
|
115
|
+
client.unbind();
|
|
116
|
+
client.destroy();
|
|
117
|
+
resolve();
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
function optionsAreValid(options) {
|
|
123
|
+
return typeof options === 'object' && typeof options.suffix === 'string' && typeof options.url === 'string' && (options.url.startsWith('ldap://') || options.url.startsWith('ldaps://') && typeof options.tlsOptions === 'object');
|
|
124
|
+
}
|
|
125
|
+
function searchForGroup(client, options, id, resolve, reject) {
|
|
126
|
+
const filter = options.groupFilter.replace(/{{id}}/gi, id);
|
|
127
|
+
const opts = {
|
|
128
|
+
scope: 'sub',
|
|
129
|
+
filter: filter
|
|
130
|
+
};
|
|
131
|
+
let found = false;
|
|
132
|
+
client.search(options.suffix, opts, (searchError, res) => {
|
|
133
|
+
if (searchError) {
|
|
134
|
+
client.unbind();
|
|
135
|
+
client.destroy();
|
|
136
|
+
return reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP group search failed'));
|
|
137
|
+
}
|
|
138
|
+
res.on('searchEntry', entry => {
|
|
139
|
+
if (entry.pojo.attributes.find(obj => obj.type === 'cn').values.includes(options.groupCn)) {
|
|
140
|
+
found = true;
|
|
141
|
+
client.unbind();
|
|
142
|
+
client.destroy();
|
|
143
|
+
return resolve();
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
res.on('end', () => {
|
|
147
|
+
if (!found) {
|
|
148
|
+
client.unbind();
|
|
149
|
+
client.destroy();
|
|
150
|
+
return reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP: User not in group'));
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
res.on('error', () => {
|
|
154
|
+
client.unbind();
|
|
155
|
+
client.destroy();
|
|
156
|
+
return reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP group search failed'));
|
|
157
|
+
});
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
function validateAppId() {
|
|
161
|
+
return Promise.resolve();
|
|
162
|
+
}
|
|
163
|
+
module.exports = {
|
|
164
|
+
validateAppId,
|
|
165
|
+
validateAuthData
|
|
166
|
+
};
|
|
167
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ldapjs","require","Parse","validateAuthData","authData","options","optionsAreValid","Promise","_","reject","Error","INTERNAL_SERVER_ERROR","clientOptions","url","startsWith","tlsOptions","client","createClient","userCn","dn","replace","id","suffix","resolve","bind","password","ldapError","error","code","OBJECT_NOT_FOUND","destroy","groupCn","groupFilter","searchForGroup","unbind","filter","opts","scope","found","search","searchError","res","on","entry","pojo","attributes","find","obj","type","values","includes","validateAppId","module","exports"],"sources":["../../../src/Adapters/Auth/ldap.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for LDAP.\n *\n * @class LDAP\n * @param {Object} options - The adapter configuration options.\n * @param {String} options.url - The LDAP server URL. Must start with `ldap://` or `ldaps://`.\n * @param {String} options.suffix - The LDAP suffix for user distinguished names (DN).\n * @param {String} [options.dn] - The distinguished name (DN) template for user authentication. Replace `{{id}}` with the username.\n * @param {Object} [options.tlsOptions] - Options for LDAPS TLS connections.\n * @param {String} [options.groupCn] - The common name (CN) of the group to verify user membership.\n * @param {String} [options.groupFilter] - The LDAP search filter for groups, with `{{id}}` replaced by the username.\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {String} authData.id - The user's LDAP username.\n * @param {String} authData.password - The user's LDAP password.\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for LDAP authentication, use the following structure:\n * ```javascript\n * {\n *   auth: {\n *     ldap: {\n *       url: 'ldaps://ldap.example.com',\n *       suffix: 'ou=users,dc=example,dc=com',\n *       groupCn: 'admins',\n *       groupFilter: '(memberUid={{id}})',\n *       tlsOptions: {\n *         rejectUnauthorized: false\n *       }\n *     }\n *   }\n * }\n * ```\n *\n * ## Authentication Process\n * 1. Validates the provided `authData` using an LDAP bind operation.\n * 2. Optionally, verifies that the user belongs to a specific group by performing an LDAP search using the provided `groupCn` or `groupFilter`.\n *\n * ## Auth Payload\n * The adapter requires the following `authData` fields:\n * - `id`: The user's LDAP username.\n * - `password`: The user's LDAP password.\n *\n * ### Example Auth Payload\n * ```json\n * {\n *   \"ldap\": {\n *     \"id\": \"jdoe\",\n *     \"password\": \"password123\"\n *   }\n * }\n * ```\n *\n * @example <caption>Configuration Example</caption>\n * // Example Parse Server configuration:\n * const config = {\n *   auth: {\n *     ldap: {\n *       url: 'ldaps://ldap.example.com',\n *       suffix: 'ou=users,dc=example,dc=com',\n *       groupCn: 'admins',\n *       groupFilter: '(memberUid={{id}})',\n *       tlsOptions: {\n *         rejectUnauthorized: false\n *       }\n *     }\n *   }\n * };\n *\n * @see {@link https://ldap.com/ LDAP Basics}\n * @see {@link https://ldap.com/ldap-filters/ LDAP Filters}\n */\n\n\nconst ldapjs = require('ldapjs');\nconst Parse = require('parse/node').Parse;\n\nfunction validateAuthData(authData, options) {\n  if (!optionsAreValid(options)) {\n    return new Promise((_, reject) => {\n      reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP auth configuration missing'));\n    });\n  }\n  const clientOptions = options.url.startsWith('ldaps://')\n    ? { url: options.url, tlsOptions: options.tlsOptions }\n    : { url: options.url };\n\n  const client = ldapjs.createClient(clientOptions);\n  const userCn =\n    typeof options.dn === 'string'\n      ? options.dn.replace('{{id}}', authData.id)\n      : `uid=${authData.id},${options.suffix}`;\n\n  return new Promise((resolve, reject) => {\n    client.bind(userCn, authData.password, ldapError => {\n      delete authData.password;\n      if (ldapError) {\n        let error;\n        switch (ldapError.code) {\n          case 49:\n            error = new Parse.Error(\n              Parse.Error.OBJECT_NOT_FOUND,\n              'LDAP: Wrong username or password'\n            );\n            break;\n          case 'DEPTH_ZERO_SELF_SIGNED_CERT':\n            error = new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'LDAPS: Certificate mismatch');\n            break;\n          default:\n            error = new Parse.Error(\n              Parse.Error.OBJECT_NOT_FOUND,\n              'LDAP: Somthing went wrong (' + ldapError.code + ')'\n            );\n        }\n        reject(error);\n        client.destroy(ldapError);\n        return;\n      }\n\n      if (typeof options.groupCn === 'string' && typeof options.groupFilter === 'string') {\n        searchForGroup(client, options, authData.id, resolve, reject);\n      } else {\n        client.unbind();\n        client.destroy();\n        resolve();\n      }\n    });\n  });\n}\n\nfunction optionsAreValid(options) {\n  return (\n    typeof options === 'object' &&\n    typeof options.suffix === 'string' &&\n    typeof options.url === 'string' &&\n    (options.url.startsWith('ldap://') ||\n      (options.url.startsWith('ldaps://') && typeof options.tlsOptions === 'object'))\n  );\n}\n\nfunction searchForGroup(client, options, id, resolve, reject) {\n  const filter = options.groupFilter.replace(/{{id}}/gi, id);\n  const opts = {\n    scope: 'sub',\n    filter: filter,\n  };\n  let found = false;\n  client.search(options.suffix, opts, (searchError, res) => {\n    if (searchError) {\n      client.unbind();\n      client.destroy();\n      return reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP group search failed'));\n    }\n    res.on('searchEntry', entry => {\n      if (entry.pojo.attributes.find(obj => obj.type === 'cn').values.includes(options.groupCn)) {\n        found = true;\n        client.unbind();\n        client.destroy();\n        return resolve();\n      }\n    });\n    res.on('end', () => {\n      if (!found) {\n        client.unbind();\n        client.destroy();\n        return reject(\n          new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP: User not in group')\n        );\n      }\n    });\n    res.on('error', () => {\n      client.unbind();\n      client.destroy();\n      return reject(new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'LDAP group search failed'));\n    });\n  });\n}\n\nfunction validateAppId() {\n  return Promise.resolve();\n}\n\nmodule.exports = {\n  validateAppId,\n  validateAuthData,\n};\n"],"mappings":"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","ignoreList":[]}
|