parse-server 2.8.4 → 8.6.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +167 -25
- package/NOTICE +10 -0
- package/README.md +929 -278
- package/lib/AccountLockout.js +47 -30
- package/lib/Adapters/AdapterLoader.js +21 -6
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
- package/lib/Adapters/Auth/AuthAdapter.js +116 -13
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
- package/lib/Adapters/Auth/OAuth1Client.js +27 -46
- package/lib/Adapters/Auth/apple.js +123 -0
- package/lib/Adapters/Auth/facebook.js +162 -35
- package/lib/Adapters/Auth/gcenter.js +217 -0
- package/lib/Adapters/Auth/github.js +118 -48
- package/lib/Adapters/Auth/google.js +160 -51
- package/lib/Adapters/Auth/gpgames.js +125 -0
- package/lib/Adapters/Auth/httpsRequest.js +6 -7
- package/lib/Adapters/Auth/index.js +170 -62
- package/lib/Adapters/Auth/instagram.js +114 -40
- package/lib/Adapters/Auth/janraincapture.js +52 -23
- package/lib/Adapters/Auth/janrainengage.js +19 -36
- package/lib/Adapters/Auth/keycloak.js +148 -0
- package/lib/Adapters/Auth/ldap.js +167 -0
- package/lib/Adapters/Auth/line.js +125 -0
- package/lib/Adapters/Auth/linkedin.js +111 -55
- package/lib/Adapters/Auth/meetup.js +24 -34
- package/lib/Adapters/Auth/mfa.js +324 -0
- package/lib/Adapters/Auth/microsoft.js +111 -0
- package/lib/Adapters/Auth/oauth2.js +97 -162
- package/lib/Adapters/Auth/phantauth.js +53 -0
- package/lib/Adapters/Auth/qq.js +108 -49
- package/lib/Adapters/Auth/spotify.js +107 -55
- package/lib/Adapters/Auth/twitter.js +188 -48
- package/lib/Adapters/Auth/utils.js +28 -0
- package/lib/Adapters/Auth/vkontakte.js +26 -39
- package/lib/Adapters/Auth/wechat.js +106 -44
- package/lib/Adapters/Auth/weibo.js +132 -58
- package/lib/Adapters/Cache/CacheAdapter.js +13 -8
- package/lib/Adapters/Cache/InMemoryCache.js +3 -13
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
- package/lib/Adapters/Cache/LRUCache.js +13 -27
- package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
- package/lib/Adapters/Cache/SchemaCache.js +25 -0
- package/lib/Adapters/Email/MailAdapter.js +10 -8
- package/lib/Adapters/Files/FilesAdapter.js +83 -25
- package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
- package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
- package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
- package/lib/Adapters/Logger/WinstonLogger.js +69 -88
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
- package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
- package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
- package/lib/Adapters/Push/PushAdapter.js +14 -7
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
- package/lib/Auth.js +488 -125
- package/lib/ClientSDK.js +2 -6
- package/lib/Config.js +525 -94
- package/lib/Controllers/AdaptableController.js +5 -25
- package/lib/Controllers/AnalyticsController.js +22 -23
- package/lib/Controllers/CacheController.js +10 -31
- package/lib/Controllers/DatabaseController.js +767 -313
- package/lib/Controllers/FilesController.js +49 -54
- package/lib/Controllers/HooksController.js +80 -84
- package/lib/Controllers/LiveQueryController.js +35 -22
- package/lib/Controllers/LoggerController.js +22 -58
- package/lib/Controllers/ParseGraphQLController.js +293 -0
- package/lib/Controllers/PushController.js +58 -49
- package/lib/Controllers/SchemaController.js +916 -422
- package/lib/Controllers/UserController.js +265 -180
- package/lib/Controllers/index.js +90 -125
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +30 -0
- package/lib/Deprecator/Deprecator.js +127 -0
- package/lib/Error.js +48 -0
- package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
- package/lib/GraphQL/ParseGraphQLServer.js +214 -0
- package/lib/GraphQL/helpers/objectsMutations.js +30 -0
- package/lib/GraphQL/helpers/objectsQueries.js +246 -0
- package/lib/GraphQL/loaders/configMutations.js +87 -0
- package/lib/GraphQL/loaders/configQueries.js +79 -0
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
- package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
- package/lib/GraphQL/loaders/filesMutations.js +107 -0
- package/lib/GraphQL/loaders/functionsMutations.js +78 -0
- package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
- package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
- package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
- package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
- package/lib/GraphQL/loaders/schemaMutations.js +162 -0
- package/lib/GraphQL/loaders/schemaQueries.js +81 -0
- package/lib/GraphQL/loaders/schemaTypes.js +341 -0
- package/lib/GraphQL/loaders/usersMutations.js +433 -0
- package/lib/GraphQL/loaders/usersQueries.js +90 -0
- package/lib/GraphQL/parseGraphQLUtils.js +63 -0
- package/lib/GraphQL/transformers/className.js +14 -0
- package/lib/GraphQL/transformers/constraintType.js +53 -0
- package/lib/GraphQL/transformers/inputType.js +51 -0
- package/lib/GraphQL/transformers/mutation.js +274 -0
- package/lib/GraphQL/transformers/outputType.js +51 -0
- package/lib/GraphQL/transformers/query.js +237 -0
- package/lib/GraphQL/transformers/schemaFields.js +99 -0
- package/lib/KeyPromiseQueue.js +48 -0
- package/lib/LiveQuery/Client.js +25 -33
- package/lib/LiveQuery/Id.js +2 -5
- package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
- package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
- package/lib/LiveQuery/ParsePubSub.js +7 -16
- package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
- package/lib/LiveQuery/QueryTools.js +76 -15
- package/lib/LiveQuery/RequestSchema.js +111 -97
- package/lib/LiveQuery/SessionTokenCache.js +23 -36
- package/lib/LiveQuery/Subscription.js +8 -17
- package/lib/LiveQuery/equalObjects.js +2 -3
- package/lib/Options/Definitions.js +1355 -382
- package/lib/Options/docs.js +301 -62
- package/lib/Options/index.js +11 -1
- package/lib/Options/parsers.js +14 -10
- package/lib/Page.js +44 -0
- package/lib/ParseMessageQueue.js +6 -13
- package/lib/ParseServer.js +474 -235
- package/lib/ParseServerRESTController.js +102 -40
- package/lib/PromiseRouter.js +39 -50
- package/lib/Push/PushQueue.js +24 -30
- package/lib/Push/PushWorker.js +32 -56
- package/lib/Push/utils.js +22 -35
- package/lib/RestQuery.js +361 -139
- package/lib/RestWrite.js +713 -344
- package/lib/Routers/AggregateRouter.js +97 -71
- package/lib/Routers/AnalyticsRouter.js +8 -14
- package/lib/Routers/AudiencesRouter.js +16 -35
- package/lib/Routers/ClassesRouter.js +86 -72
- package/lib/Routers/CloudCodeRouter.js +28 -37
- package/lib/Routers/FeaturesRouter.js +22 -25
- package/lib/Routers/FilesRouter.js +266 -171
- package/lib/Routers/FunctionsRouter.js +87 -103
- package/lib/Routers/GlobalConfigRouter.js +94 -33
- package/lib/Routers/GraphQLRouter.js +41 -0
- package/lib/Routers/HooksRouter.js +43 -47
- package/lib/Routers/IAPValidationRouter.js +57 -70
- package/lib/Routers/InstallationsRouter.js +17 -25
- package/lib/Routers/LogsRouter.js +10 -25
- package/lib/Routers/PagesRouter.js +647 -0
- package/lib/Routers/PublicAPIRouter.js +104 -112
- package/lib/Routers/PurgeRouter.js +19 -29
- package/lib/Routers/PushRouter.js +14 -28
- package/lib/Routers/RolesRouter.js +7 -14
- package/lib/Routers/SchemasRouter.js +63 -42
- package/lib/Routers/SecurityRouter.js +34 -0
- package/lib/Routers/SessionsRouter.js +25 -38
- package/lib/Routers/UsersRouter.js +463 -190
- package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
- package/lib/SchemaMigrations/Migrations.js +30 -0
- package/lib/Security/Check.js +109 -0
- package/lib/Security/CheckGroup.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
- package/lib/Security/CheckGroups/CheckGroups.js +21 -0
- package/lib/Security/CheckRunner.js +213 -0
- package/lib/SharedRest.js +29 -0
- package/lib/StatusHandler.js +96 -93
- package/lib/TestUtils.js +70 -14
- package/lib/Utils.js +468 -0
- package/lib/batch.js +74 -40
- package/lib/cache.js +8 -8
- package/lib/cli/definitions/parse-live-query-server.js +4 -3
- package/lib/cli/definitions/parse-server.js +4 -3
- package/lib/cli/parse-live-query-server.js +9 -17
- package/lib/cli/parse-server.js +49 -47
- package/lib/cli/utils/commander.js +20 -29
- package/lib/cli/utils/runner.js +31 -32
- package/lib/cloud-code/Parse.Cloud.js +711 -36
- package/lib/cloud-code/Parse.Server.js +21 -0
- package/lib/cryptoUtils.js +6 -11
- package/lib/defaults.js +21 -15
- package/lib/deprecated.js +1 -1
- package/lib/index.js +78 -67
- package/lib/logger.js +12 -20
- package/lib/middlewares.js +484 -160
- package/lib/password.js +10 -6
- package/lib/request.js +175 -0
- package/lib/requiredParameter.js +4 -3
- package/lib/rest.js +157 -82
- package/lib/triggers.js +627 -185
- package/lib/vendor/README.md +3 -3
- package/lib/vendor/mongodbUrl.js +224 -137
- package/package.json +135 -57
- package/postinstall.js +38 -50
- package/public_html/invalid_verification_link.html +3 -3
- package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
- package/types/@types/deepcopy/index.d.ts +5 -0
- package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
- package/types/Options/index.d.ts +301 -0
- package/types/ParseServer.d.ts +65 -0
- package/types/eslint.config.mjs +30 -0
- package/types/index.d.ts +21 -0
- package/types/logger.d.ts +2 -0
- package/types/tests.ts +44 -0
- package/types/tsconfig.json +24 -0
- package/CHANGELOG.md +0 -1246
- package/PATENTS +0 -37
- package/bin/dev +0 -37
- package/lib/.DS_Store +0 -0
- package/lib/Adapters/Auth/common.js +0 -2
- package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
- package/lib/Controllers/SchemaCache.js +0 -97
- package/lib/LiveQuery/.DS_Store +0 -0
- package/lib/cli/utils/parsers.js +0 -77
- package/lib/cloud-code/.DS_Store +0 -0
- package/lib/cloud-code/HTTPResponse.js +0 -57
- package/lib/cloud-code/Untitled-1 +0 -123
- package/lib/cloud-code/httpRequest.js +0 -102
- package/lib/cloud-code/team.html +0 -123
- package/lib/graphql/ParseClass.js +0 -234
- package/lib/graphql/Schema.js +0 -197
- package/lib/graphql/index.js +0 -1
- package/lib/graphql/types/ACL.js +0 -35
- package/lib/graphql/types/Date.js +0 -25
- package/lib/graphql/types/File.js +0 -24
- package/lib/graphql/types/GeoPoint.js +0 -35
- package/lib/graphql/types/JSONObject.js +0 -30
- package/lib/graphql/types/NumberInput.js +0 -43
- package/lib/graphql/types/NumberQuery.js +0 -42
- package/lib/graphql/types/Pointer.js +0 -35
- package/lib/graphql/types/QueryConstraint.js +0 -61
- package/lib/graphql/types/StringQuery.js +0 -39
- package/lib/graphql/types/index.js +0 -110
|
@@ -1,55 +1,28 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.MongoStorageAdapter =
|
|
7
|
-
|
|
8
|
-
var
|
|
9
|
-
|
|
10
|
-
var
|
|
11
|
-
|
|
12
|
-
var
|
|
13
|
-
|
|
14
|
-
var
|
|
15
|
-
|
|
16
|
-
var
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
var _MongoTransform = require('./MongoTransform');
|
|
21
|
-
|
|
22
|
-
var _node = require('parse/node');
|
|
23
|
-
|
|
24
|
-
var _node2 = _interopRequireDefault(_node);
|
|
25
|
-
|
|
26
|
-
var _lodash = require('lodash');
|
|
27
|
-
|
|
28
|
-
var _lodash2 = _interopRequireDefault(_lodash);
|
|
29
|
-
|
|
30
|
-
var _defaults = require('../../../defaults');
|
|
31
|
-
|
|
32
|
-
var _defaults2 = _interopRequireDefault(_defaults);
|
|
33
|
-
|
|
34
|
-
var _logger = require('../../../logger');
|
|
35
|
-
|
|
36
|
-
var _logger2 = _interopRequireDefault(_logger);
|
|
37
|
-
|
|
38
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
39
|
-
|
|
40
|
-
function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; }
|
|
6
|
+
exports.default = exports.MongoStorageAdapter = void 0;
|
|
7
|
+
var _mongodbUrl = require("../../../vendor/mongodbUrl");
|
|
8
|
+
var _StorageAdapter = require("../StorageAdapter");
|
|
9
|
+
var _MongoCollection = _interopRequireDefault(require("./MongoCollection"));
|
|
10
|
+
var _MongoSchemaCollection = _interopRequireDefault(require("./MongoSchemaCollection"));
|
|
11
|
+
var _MongoTransform = require("./MongoTransform");
|
|
12
|
+
var _node = _interopRequireDefault(require("parse/node"));
|
|
13
|
+
var _lodash = _interopRequireDefault(require("lodash"));
|
|
14
|
+
var _defaults = _interopRequireWildcard(require("../../../defaults"));
|
|
15
|
+
var _logger = _interopRequireDefault(require("../../../logger"));
|
|
16
|
+
var _Utils = _interopRequireDefault(require("../../../Utils"));
|
|
17
|
+
function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function (e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != typeof e && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (const t in e) "default" !== t && {}.hasOwnProperty.call(e, t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, t)) && (i.get || i.set) ? o(f, t, i) : f[t] = e[t]); return f; })(e, t); }
|
|
18
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
41
19
|
// -disable-next
|
|
42
|
-
|
|
43
20
|
// -disable-next
|
|
44
|
-
|
|
45
|
-
|
|
46
21
|
// -disable-next
|
|
47
22
|
const mongodb = require('mongodb');
|
|
48
23
|
const MongoClient = mongodb.MongoClient;
|
|
49
24
|
const ReadPreference = mongodb.ReadPreference;
|
|
50
|
-
|
|
51
25
|
const MongoSchemaCollectionName = '_SCHEMA';
|
|
52
|
-
|
|
53
26
|
const storageAdapterAllCollections = mongoAdapter => {
|
|
54
27
|
return mongoAdapter.connect().then(() => mongoAdapter.database.collections()).then(collections => {
|
|
55
28
|
return collections.filter(collection => {
|
|
@@ -62,13 +35,11 @@ const storageAdapterAllCollections = mongoAdapter => {
|
|
|
62
35
|
});
|
|
63
36
|
});
|
|
64
37
|
};
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
38
|
+
const convertParseSchemaToMongoSchema = ({
|
|
39
|
+
...schema
|
|
40
|
+
}) => {
|
|
69
41
|
delete schema.fields._rperm;
|
|
70
42
|
delete schema.fields._wperm;
|
|
71
|
-
|
|
72
43
|
if (schema.className === '_User') {
|
|
73
44
|
// Legacy mongo adapter knows about the difference between password and _hashed_password.
|
|
74
45
|
// Future database adapters will only know about _hashed_password.
|
|
@@ -76,7 +47,6 @@ const convertParseSchemaToMongoSchema = (_ref) => {
|
|
|
76
47
|
// to add _hashed_password back ever.
|
|
77
48
|
delete schema.fields._hashed_password;
|
|
78
49
|
}
|
|
79
|
-
|
|
80
50
|
return schema;
|
|
81
51
|
};
|
|
82
52
|
|
|
@@ -90,11 +60,22 @@ const mongoSchemaFromFieldsAndClassNameAndCLP = (fields, className, classLevelPe
|
|
|
90
60
|
createdAt: 'string',
|
|
91
61
|
_metadata: undefined
|
|
92
62
|
};
|
|
93
|
-
|
|
94
63
|
for (const fieldName in fields) {
|
|
95
|
-
|
|
64
|
+
const {
|
|
65
|
+
type,
|
|
66
|
+
targetClass,
|
|
67
|
+
...fieldOptions
|
|
68
|
+
} = fields[fieldName];
|
|
69
|
+
mongoObject[fieldName] = _MongoSchemaCollection.default.parseFieldTypeToMongoFieldType({
|
|
70
|
+
type,
|
|
71
|
+
targetClass
|
|
72
|
+
});
|
|
73
|
+
if (fieldOptions && Object.keys(fieldOptions).length > 0) {
|
|
74
|
+
mongoObject._metadata = mongoObject._metadata || {};
|
|
75
|
+
mongoObject._metadata.fields_options = mongoObject._metadata.fields_options || {};
|
|
76
|
+
mongoObject._metadata.fields_options[fieldName] = fieldOptions;
|
|
77
|
+
}
|
|
96
78
|
}
|
|
97
|
-
|
|
98
79
|
if (typeof classLevelPermissions !== 'undefined') {
|
|
99
80
|
mongoObject._metadata = mongoObject._metadata || {};
|
|
100
81
|
if (!classLevelPermissions) {
|
|
@@ -103,39 +84,61 @@ const mongoSchemaFromFieldsAndClassNameAndCLP = (fields, className, classLevelPe
|
|
|
103
84
|
mongoObject._metadata.class_permissions = classLevelPermissions;
|
|
104
85
|
}
|
|
105
86
|
}
|
|
106
|
-
|
|
107
87
|
if (indexes && typeof indexes === 'object' && Object.keys(indexes).length > 0) {
|
|
108
88
|
mongoObject._metadata = mongoObject._metadata || {};
|
|
109
89
|
mongoObject._metadata.indexes = indexes;
|
|
110
90
|
}
|
|
111
|
-
|
|
112
91
|
if (!mongoObject._metadata) {
|
|
113
92
|
// cleanup the unused _metadata
|
|
114
93
|
delete mongoObject._metadata;
|
|
115
94
|
}
|
|
116
|
-
|
|
117
95
|
return mongoObject;
|
|
118
96
|
};
|
|
119
|
-
|
|
97
|
+
function validateExplainValue(explain) {
|
|
98
|
+
if (explain) {
|
|
99
|
+
// The list of allowed explain values is from node-mongodb-native/lib/explain.js
|
|
100
|
+
const explainAllowedValues = ['queryPlanner', 'queryPlannerExtended', 'executionStats', 'allPlansExecution', false, true];
|
|
101
|
+
if (!explainAllowedValues.includes(explain)) {
|
|
102
|
+
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, 'Invalid value for explain');
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
120
106
|
class MongoStorageAdapter {
|
|
121
107
|
// Private
|
|
108
|
+
|
|
109
|
+
// Public
|
|
110
|
+
|
|
122
111
|
constructor({
|
|
123
|
-
uri =
|
|
112
|
+
uri = _defaults.default.DefaultMongoURI,
|
|
124
113
|
collectionPrefix = '',
|
|
125
114
|
mongoOptions = {}
|
|
126
115
|
}) {
|
|
127
116
|
this._uri = uri;
|
|
128
117
|
this._collectionPrefix = collectionPrefix;
|
|
129
|
-
this.
|
|
118
|
+
this._onchange = () => {};
|
|
130
119
|
|
|
131
120
|
// MaxTimeMS is not a global MongoDB client option, it is applied per operation.
|
|
132
121
|
this._maxTimeMS = mongoOptions.maxTimeMS;
|
|
133
122
|
this.canSortOnJoinTables = true;
|
|
134
|
-
|
|
123
|
+
this.enableSchemaHooks = !!mongoOptions.enableSchemaHooks;
|
|
124
|
+
this.schemaCacheTtl = mongoOptions.schemaCacheTtl;
|
|
125
|
+
this.disableIndexFieldValidation = !!mongoOptions.disableIndexFieldValidation;
|
|
126
|
+
this._logClientEvents = mongoOptions.logClientEvents;
|
|
127
|
+
|
|
128
|
+
// Create a copy of mongoOptions and remove Parse Server-specific options that should not
|
|
129
|
+
// be passed to MongoDB client. Note: We only delete from this._mongoOptions, not from the
|
|
130
|
+
// original mongoOptions object, because other components (like DatabaseController) need
|
|
131
|
+
// access to these options.
|
|
132
|
+
this._mongoOptions = {
|
|
133
|
+
...mongoOptions
|
|
134
|
+
};
|
|
135
|
+
for (const key of _defaults.ParseServerDatabaseOptions) {
|
|
136
|
+
delete this._mongoOptions[key];
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
watch(callback) {
|
|
140
|
+
this._onchange = callback;
|
|
135
141
|
}
|
|
136
|
-
// Public
|
|
137
|
-
|
|
138
|
-
|
|
139
142
|
connect() {
|
|
140
143
|
if (this.connectionPromise) {
|
|
141
144
|
return this.connectionPromise;
|
|
@@ -144,7 +147,6 @@ class MongoStorageAdapter {
|
|
|
144
147
|
// parsing and re-formatting causes the auth value (if there) to get URI
|
|
145
148
|
// encoded
|
|
146
149
|
const encodedUri = (0, _mongodbUrl.format)((0, _mongodbUrl.parse)(this._uri));
|
|
147
|
-
|
|
148
150
|
this.connectionPromise = MongoClient.connect(encodedUri, this._mongoOptions).then(client => {
|
|
149
151
|
// Starting mongoDB 3.0, the MongoClient.connect don't return a DB anymore but a client
|
|
150
152
|
// Fortunately, we can get back the options and use them to select the proper DB.
|
|
@@ -155,78 +157,110 @@ class MongoStorageAdapter {
|
|
|
155
157
|
delete this.connectionPromise;
|
|
156
158
|
return;
|
|
157
159
|
}
|
|
158
|
-
|
|
160
|
+
client.on('error', () => {
|
|
159
161
|
delete this.connectionPromise;
|
|
160
162
|
});
|
|
161
|
-
|
|
163
|
+
client.on('close', () => {
|
|
162
164
|
delete this.connectionPromise;
|
|
163
165
|
});
|
|
166
|
+
|
|
167
|
+
// Set up client event logging if configured
|
|
168
|
+
if (this._logClientEvents && Array.isArray(this._logClientEvents)) {
|
|
169
|
+
this._logClientEvents.forEach(eventConfig => {
|
|
170
|
+
client.on(eventConfig.name, event => {
|
|
171
|
+
let logData = {};
|
|
172
|
+
if (!eventConfig.keys || eventConfig.keys.length === 0) {
|
|
173
|
+
logData = event;
|
|
174
|
+
} else {
|
|
175
|
+
eventConfig.keys.forEach(keyPath => {
|
|
176
|
+
logData[keyPath] = _lodash.default.get(event, keyPath);
|
|
177
|
+
});
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// Validate log level exists, fallback to 'info'
|
|
181
|
+
const logLevel = typeof _logger.default[eventConfig.logLevel] === 'function' ? eventConfig.logLevel : 'info';
|
|
182
|
+
|
|
183
|
+
// Safe JSON serialization with Map/Set and circular reference support
|
|
184
|
+
const logMessage = `MongoDB client event ${eventConfig.name}: ${JSON.stringify(logData, _Utils.default.getCircularReplacer())}`;
|
|
185
|
+
_logger.default[logLevel](logMessage);
|
|
186
|
+
});
|
|
187
|
+
});
|
|
188
|
+
}
|
|
164
189
|
this.client = client;
|
|
165
190
|
this.database = database;
|
|
166
191
|
}).catch(err => {
|
|
167
192
|
delete this.connectionPromise;
|
|
168
193
|
return Promise.reject(err);
|
|
169
194
|
});
|
|
170
|
-
|
|
171
195
|
return this.connectionPromise;
|
|
172
196
|
}
|
|
173
|
-
|
|
174
197
|
handleError(error) {
|
|
175
198
|
if (error && error.code === 13) {
|
|
176
199
|
// Unauthorized error
|
|
177
200
|
delete this.client;
|
|
178
201
|
delete this.database;
|
|
179
202
|
delete this.connectionPromise;
|
|
180
|
-
|
|
203
|
+
_logger.default.error('Received unauthorized error', {
|
|
204
|
+
error: error
|
|
205
|
+
});
|
|
181
206
|
}
|
|
182
207
|
throw error;
|
|
183
208
|
}
|
|
184
|
-
|
|
185
|
-
handleShutdown() {
|
|
209
|
+
async handleShutdown() {
|
|
186
210
|
if (!this.client) {
|
|
187
211
|
return;
|
|
188
212
|
}
|
|
189
|
-
this.client.close(false);
|
|
213
|
+
await this.client.close(false);
|
|
214
|
+
delete this.connectionPromise;
|
|
190
215
|
}
|
|
191
|
-
|
|
192
216
|
_adaptiveCollection(name) {
|
|
193
|
-
return this.connect().then(() => this.database.collection(this._collectionPrefix + name)).then(rawCollection => new
|
|
217
|
+
return this.connect().then(() => this.database.collection(this._collectionPrefix + name)).then(rawCollection => new _MongoCollection.default(rawCollection)).catch(err => this.handleError(err));
|
|
194
218
|
}
|
|
195
|
-
|
|
196
219
|
_schemaCollection() {
|
|
197
|
-
return this.connect().then(() => this._adaptiveCollection(MongoSchemaCollectionName)).then(collection =>
|
|
220
|
+
return this.connect().then(() => this._adaptiveCollection(MongoSchemaCollectionName)).then(collection => {
|
|
221
|
+
if (!this._stream && this.enableSchemaHooks) {
|
|
222
|
+
this._stream = collection._mongoCollection.watch();
|
|
223
|
+
this._stream.on('change', () => this._onchange());
|
|
224
|
+
}
|
|
225
|
+
return new _MongoSchemaCollection.default(collection);
|
|
226
|
+
});
|
|
198
227
|
}
|
|
199
|
-
|
|
200
228
|
classExists(name) {
|
|
201
229
|
return this.connect().then(() => {
|
|
202
|
-
return this.database.listCollections({
|
|
230
|
+
return this.database.listCollections({
|
|
231
|
+
name: this._collectionPrefix + name
|
|
232
|
+
}).toArray();
|
|
203
233
|
}).then(collections => {
|
|
204
234
|
return collections.length > 0;
|
|
205
235
|
}).catch(err => this.handleError(err));
|
|
206
236
|
}
|
|
207
|
-
|
|
208
237
|
setClassLevelPermissions(className, CLPs) {
|
|
209
238
|
return this._schemaCollection().then(schemaCollection => schemaCollection.updateSchema(className, {
|
|
210
|
-
$set: {
|
|
239
|
+
$set: {
|
|
240
|
+
'_metadata.class_permissions': CLPs
|
|
241
|
+
}
|
|
211
242
|
})).catch(err => this.handleError(err));
|
|
212
243
|
}
|
|
213
|
-
|
|
214
244
|
setIndexesWithSchemaFormat(className, submittedIndexes, existingIndexes = {}, fields) {
|
|
215
245
|
if (submittedIndexes === undefined) {
|
|
216
246
|
return Promise.resolve();
|
|
217
247
|
}
|
|
218
248
|
if (Object.keys(existingIndexes).length === 0) {
|
|
219
|
-
existingIndexes = {
|
|
249
|
+
existingIndexes = {
|
|
250
|
+
_id_: {
|
|
251
|
+
_id: 1
|
|
252
|
+
}
|
|
253
|
+
};
|
|
220
254
|
}
|
|
221
255
|
const deletePromises = [];
|
|
222
256
|
const insertedIndexes = [];
|
|
223
257
|
Object.keys(submittedIndexes).forEach(name => {
|
|
224
258
|
const field = submittedIndexes[name];
|
|
225
259
|
if (existingIndexes[name] && field.__op !== 'Delete') {
|
|
226
|
-
throw new
|
|
260
|
+
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, `Index ${name} exists, cannot update.`);
|
|
227
261
|
}
|
|
228
262
|
if (!existingIndexes[name] && field.__op === 'Delete') {
|
|
229
|
-
throw new
|
|
263
|
+
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, `Index ${name} does not exist, cannot delete.`);
|
|
230
264
|
}
|
|
231
265
|
if (field.__op === 'Delete') {
|
|
232
266
|
const promise = this.dropIndex(className, name);
|
|
@@ -234,8 +268,8 @@ class MongoStorageAdapter {
|
|
|
234
268
|
delete existingIndexes[name];
|
|
235
269
|
} else {
|
|
236
270
|
Object.keys(field).forEach(key => {
|
|
237
|
-
if (!
|
|
238
|
-
throw new
|
|
271
|
+
if (!this.disableIndexFieldValidation && !Object.prototype.hasOwnProperty.call(fields, key.indexOf('_p_') === 0 ? key.replace('_p_', '') : key)) {
|
|
272
|
+
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, `Field ${key} does not exist, cannot add index.`);
|
|
239
273
|
}
|
|
240
274
|
});
|
|
241
275
|
existingIndexes[name] = field;
|
|
@@ -250,10 +284,11 @@ class MongoStorageAdapter {
|
|
|
250
284
|
insertPromise = this.createIndexes(className, insertedIndexes);
|
|
251
285
|
}
|
|
252
286
|
return Promise.all(deletePromises).then(() => insertPromise).then(() => this._schemaCollection()).then(schemaCollection => schemaCollection.updateSchema(className, {
|
|
253
|
-
$set: {
|
|
287
|
+
$set: {
|
|
288
|
+
'_metadata.indexes': existingIndexes
|
|
289
|
+
}
|
|
254
290
|
})).catch(err => this.handleError(err));
|
|
255
291
|
}
|
|
256
|
-
|
|
257
292
|
setIndexesFromMongo(className) {
|
|
258
293
|
return this.getIndexes(className).then(indexes => {
|
|
259
294
|
indexes = indexes.reduce((obj, index) => {
|
|
@@ -268,21 +303,25 @@ class MongoStorageAdapter {
|
|
|
268
303
|
return obj;
|
|
269
304
|
}, {});
|
|
270
305
|
return this._schemaCollection().then(schemaCollection => schemaCollection.updateSchema(className, {
|
|
271
|
-
$set: {
|
|
306
|
+
$set: {
|
|
307
|
+
'_metadata.indexes': indexes
|
|
308
|
+
}
|
|
272
309
|
}));
|
|
273
310
|
}).catch(err => this.handleError(err)).catch(() => {
|
|
274
311
|
// Ignore if collection not found
|
|
275
312
|
return Promise.resolve();
|
|
276
313
|
});
|
|
277
314
|
}
|
|
278
|
-
|
|
279
315
|
createClass(className, schema) {
|
|
280
316
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
281
317
|
const mongoObject = mongoSchemaFromFieldsAndClassNameAndCLP(schema.fields, className, schema.classLevelPermissions, schema.indexes);
|
|
282
318
|
mongoObject._id = className;
|
|
283
319
|
return this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields).then(() => this._schemaCollection()).then(schemaCollection => schemaCollection.insertSchema(mongoObject)).catch(err => this.handleError(err));
|
|
284
320
|
}
|
|
285
|
-
|
|
321
|
+
async updateFieldOptions(className, fieldName, type) {
|
|
322
|
+
const schemaCollection = await this._schemaCollection();
|
|
323
|
+
await schemaCollection.updateFieldOptions(className, fieldName, type);
|
|
324
|
+
}
|
|
286
325
|
addFieldIfNotExists(className, fieldName, type) {
|
|
287
326
|
return this._schemaCollection().then(schemaCollection => schemaCollection.addFieldIfNotExists(className, fieldName, type)).then(() => this.createIndexesIfNeeded(className, fieldName, type)).catch(err => this.handleError(err));
|
|
288
327
|
}
|
|
@@ -300,9 +339,8 @@ class MongoStorageAdapter {
|
|
|
300
339
|
// We've dropped the collection, now remove the _SCHEMA document
|
|
301
340
|
.then(() => this._schemaCollection()).then(schemaCollection => schemaCollection.findAndDeleteSchema(className)).catch(err => this.handleError(err));
|
|
302
341
|
}
|
|
303
|
-
|
|
304
342
|
deleteAllClasses(fast) {
|
|
305
|
-
return storageAdapterAllCollections(this).then(collections => Promise.all(collections.map(collection => fast ? collection.
|
|
343
|
+
return storageAdapterAllCollections(this).then(collections => Promise.all(collections.map(collection => fast ? collection.deleteMany({}) : collection.drop())));
|
|
306
344
|
}
|
|
307
345
|
|
|
308
346
|
// Remove the column and all the data. For Relations, the _Join collection is handled
|
|
@@ -333,17 +371,30 @@ class MongoStorageAdapter {
|
|
|
333
371
|
return fieldName;
|
|
334
372
|
}
|
|
335
373
|
});
|
|
336
|
-
const collectionUpdate = {
|
|
374
|
+
const collectionUpdate = {
|
|
375
|
+
$unset: {}
|
|
376
|
+
};
|
|
337
377
|
mongoFormatNames.forEach(name => {
|
|
338
378
|
collectionUpdate['$unset'][name] = null;
|
|
339
379
|
});
|
|
340
|
-
|
|
341
|
-
|
|
380
|
+
const collectionFilter = {
|
|
381
|
+
$or: []
|
|
382
|
+
};
|
|
383
|
+
mongoFormatNames.forEach(name => {
|
|
384
|
+
collectionFilter['$or'].push({
|
|
385
|
+
[name]: {
|
|
386
|
+
$exists: true
|
|
387
|
+
}
|
|
388
|
+
});
|
|
389
|
+
});
|
|
390
|
+
const schemaUpdate = {
|
|
391
|
+
$unset: {}
|
|
392
|
+
};
|
|
342
393
|
fieldNames.forEach(name => {
|
|
343
394
|
schemaUpdate['$unset'][name] = null;
|
|
395
|
+
schemaUpdate['$unset'][`_metadata.fields_options.${name}`] = null;
|
|
344
396
|
});
|
|
345
|
-
|
|
346
|
-
return this._adaptiveCollection(className).then(collection => collection.updateMany({}, collectionUpdate)).then(() => this._schemaCollection()).then(schemaCollection => schemaCollection.updateSchema(className, schemaUpdate)).catch(err => this.handleError(err));
|
|
397
|
+
return this._adaptiveCollection(className).then(collection => collection.updateMany(collectionFilter, collectionUpdate)).then(() => this._schemaCollection()).then(schemaCollection => schemaCollection.updateSchema(className, schemaUpdate)).catch(err => this.handleError(err));
|
|
347
398
|
}
|
|
348
399
|
|
|
349
400
|
// Return a promise for all schemas known to this adapter, in Parse format. In case the
|
|
@@ -363,18 +414,22 @@ class MongoStorageAdapter {
|
|
|
363
414
|
// TODO: As yet not particularly well specified. Creates an object. Maybe shouldn't even need the schema,
|
|
364
415
|
// and should infer from the type. Or maybe does need the schema for validations. Or maybe needs
|
|
365
416
|
// the schema only for the legacy mongo format. We'll figure that out later.
|
|
366
|
-
createObject(className, schema, object) {
|
|
417
|
+
createObject(className, schema, object, transactionalSession) {
|
|
367
418
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
368
419
|
const mongoObject = (0, _MongoTransform.parseObjectToMongoObjectForCreate)(className, object, schema);
|
|
369
|
-
return this._adaptiveCollection(className).then(collection => collection.insertOne(mongoObject)).
|
|
420
|
+
return this._adaptiveCollection(className).then(collection => collection.insertOne(mongoObject, transactionalSession)).then(() => ({
|
|
421
|
+
ops: [mongoObject]
|
|
422
|
+
})).catch(error => {
|
|
370
423
|
if (error.code === 11000) {
|
|
371
424
|
// Duplicate value
|
|
372
|
-
const err = new
|
|
425
|
+
const err = new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');
|
|
373
426
|
err.underlyingError = error;
|
|
374
427
|
if (error.message) {
|
|
375
428
|
const matches = error.message.match(/index:[\sa-zA-Z0-9_\-\.]+\$?([a-zA-Z_-]+)_1/);
|
|
376
429
|
if (matches && Array.isArray(matches)) {
|
|
377
|
-
err.userInfo = {
|
|
430
|
+
err.userInfo = {
|
|
431
|
+
duplicated_field: matches[1]
|
|
432
|
+
};
|
|
378
433
|
}
|
|
379
434
|
}
|
|
380
435
|
throw err;
|
|
@@ -386,61 +441,88 @@ class MongoStorageAdapter {
|
|
|
386
441
|
// Remove all objects that match the given Parse Query.
|
|
387
442
|
// If no objects match, reject with OBJECT_NOT_FOUND. If objects are found and deleted, resolve with undefined.
|
|
388
443
|
// If there is some other error, reject with INTERNAL_SERVER_ERROR.
|
|
389
|
-
deleteObjectsByQuery(className, schema, query) {
|
|
444
|
+
deleteObjectsByQuery(className, schema, query, transactionalSession) {
|
|
390
445
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
391
446
|
return this._adaptiveCollection(className).then(collection => {
|
|
392
447
|
const mongoWhere = (0, _MongoTransform.transformWhere)(className, query, schema);
|
|
393
|
-
return collection.deleteMany(mongoWhere);
|
|
394
|
-
}).catch(err => this.handleError(err)).then(({
|
|
395
|
-
|
|
396
|
-
|
|
448
|
+
return collection.deleteMany(mongoWhere, transactionalSession);
|
|
449
|
+
}).catch(err => this.handleError(err)).then(({
|
|
450
|
+
deletedCount
|
|
451
|
+
}) => {
|
|
452
|
+
if (deletedCount === 0) {
|
|
453
|
+
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'Object not found.');
|
|
397
454
|
}
|
|
398
455
|
return Promise.resolve();
|
|
399
456
|
}, () => {
|
|
400
|
-
throw new
|
|
457
|
+
throw new _node.default.Error(_node.default.Error.INTERNAL_SERVER_ERROR, 'Database adapter error');
|
|
401
458
|
});
|
|
402
459
|
}
|
|
403
460
|
|
|
404
461
|
// Apply the update to all objects that match the given Parse Query.
|
|
405
|
-
updateObjectsByQuery(className, schema, query, update) {
|
|
462
|
+
updateObjectsByQuery(className, schema, query, update, transactionalSession) {
|
|
406
463
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
407
464
|
const mongoUpdate = (0, _MongoTransform.transformUpdate)(className, update, schema);
|
|
408
465
|
const mongoWhere = (0, _MongoTransform.transformWhere)(className, query, schema);
|
|
409
|
-
return this._adaptiveCollection(className).then(collection => collection.updateMany(mongoWhere, mongoUpdate)).catch(err => this.handleError(err));
|
|
466
|
+
return this._adaptiveCollection(className).then(collection => collection.updateMany(mongoWhere, mongoUpdate, transactionalSession)).catch(err => this.handleError(err));
|
|
410
467
|
}
|
|
411
468
|
|
|
412
469
|
// Atomically finds and updates an object based on query.
|
|
413
470
|
// Return value not currently well specified.
|
|
414
|
-
findOneAndUpdate(className, schema, query, update) {
|
|
471
|
+
findOneAndUpdate(className, schema, query, update, transactionalSession) {
|
|
415
472
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
416
473
|
const mongoUpdate = (0, _MongoTransform.transformUpdate)(className, update, schema);
|
|
417
474
|
const mongoWhere = (0, _MongoTransform.transformWhere)(className, query, schema);
|
|
418
|
-
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.
|
|
475
|
+
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.findOneAndUpdate(mongoWhere, mongoUpdate, {
|
|
476
|
+
returnDocument: 'after',
|
|
477
|
+
session: transactionalSession || undefined
|
|
478
|
+
})).then(result => (0, _MongoTransform.mongoObjectToParseObject)(className, result, schema)).catch(error => {
|
|
419
479
|
if (error.code === 11000) {
|
|
420
|
-
throw new
|
|
480
|
+
throw new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');
|
|
421
481
|
}
|
|
422
482
|
throw error;
|
|
423
483
|
}).catch(err => this.handleError(err));
|
|
424
484
|
}
|
|
425
485
|
|
|
426
486
|
// Hopefully we can get rid of this. It's only used for config and hooks.
|
|
427
|
-
upsertOneObject(className, schema, query, update) {
|
|
487
|
+
upsertOneObject(className, schema, query, update, transactionalSession) {
|
|
428
488
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
429
489
|
const mongoUpdate = (0, _MongoTransform.transformUpdate)(className, update, schema);
|
|
430
490
|
const mongoWhere = (0, _MongoTransform.transformWhere)(className, query, schema);
|
|
431
|
-
return this._adaptiveCollection(className).then(collection => collection.upsertOne(mongoWhere, mongoUpdate)).catch(err => this.handleError(err));
|
|
491
|
+
return this._adaptiveCollection(className).then(collection => collection.upsertOne(mongoWhere, mongoUpdate, transactionalSession)).catch(err => this.handleError(err));
|
|
432
492
|
}
|
|
433
493
|
|
|
434
494
|
// Executes a find. Accepts: className, query in Parse format, and { skip, limit, sort }.
|
|
435
|
-
find(className, schema, query, {
|
|
495
|
+
find(className, schema, query, {
|
|
496
|
+
skip,
|
|
497
|
+
limit,
|
|
498
|
+
sort,
|
|
499
|
+
keys,
|
|
500
|
+
readPreference,
|
|
501
|
+
hint,
|
|
502
|
+
caseInsensitive,
|
|
503
|
+
explain,
|
|
504
|
+
comment
|
|
505
|
+
}) {
|
|
506
|
+
validateExplainValue(explain);
|
|
436
507
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
437
508
|
const mongoWhere = (0, _MongoTransform.transformWhere)(className, query, schema);
|
|
438
|
-
const mongoSort =
|
|
439
|
-
const mongoKeys =
|
|
440
|
-
|
|
509
|
+
const mongoSort = _lodash.default.mapKeys(sort, (value, fieldName) => (0, _MongoTransform.transformKey)(className, fieldName, schema));
|
|
510
|
+
const mongoKeys = _lodash.default.reduce(keys, (memo, key) => {
|
|
511
|
+
if (key === 'ACL') {
|
|
512
|
+
memo['_rperm'] = 1;
|
|
513
|
+
memo['_wperm'] = 1;
|
|
514
|
+
} else {
|
|
515
|
+
memo[(0, _MongoTransform.transformKey)(className, key, schema)] = 1;
|
|
516
|
+
}
|
|
441
517
|
return memo;
|
|
442
518
|
}, {});
|
|
443
519
|
|
|
520
|
+
// If we aren't requesting the `_id` field, we need to explicitly opt out
|
|
521
|
+
// of it. Doing so in parse-server is unusual, but it can allow us to
|
|
522
|
+
// optimize some queries with covering indexes.
|
|
523
|
+
if (keys && !mongoKeys._id) {
|
|
524
|
+
mongoKeys._id = 0;
|
|
525
|
+
}
|
|
444
526
|
readPreference = this._parseReadPreference(readPreference);
|
|
445
527
|
return this.createTextIndexesIfNeeded(className, query, schema).then(() => this._adaptiveCollection(className)).then(collection => collection.find(mongoWhere, {
|
|
446
528
|
skip,
|
|
@@ -448,8 +530,49 @@ class MongoStorageAdapter {
|
|
|
448
530
|
sort: mongoSort,
|
|
449
531
|
keys: mongoKeys,
|
|
450
532
|
maxTimeMS: this._maxTimeMS,
|
|
451
|
-
readPreference
|
|
452
|
-
|
|
533
|
+
readPreference,
|
|
534
|
+
hint,
|
|
535
|
+
caseInsensitive,
|
|
536
|
+
explain,
|
|
537
|
+
comment
|
|
538
|
+
})).then(objects => {
|
|
539
|
+
if (explain) {
|
|
540
|
+
return objects;
|
|
541
|
+
}
|
|
542
|
+
return objects.map(object => (0, _MongoTransform.mongoObjectToParseObject)(className, object, schema));
|
|
543
|
+
}).catch(err => this.handleError(err));
|
|
544
|
+
}
|
|
545
|
+
ensureIndex(className, schema, fieldNames, indexName, caseInsensitive = false, options = {}) {
|
|
546
|
+
schema = convertParseSchemaToMongoSchema(schema);
|
|
547
|
+
const indexCreationRequest = {};
|
|
548
|
+
const mongoFieldNames = fieldNames.map(fieldName => (0, _MongoTransform.transformKey)(className, fieldName, schema));
|
|
549
|
+
mongoFieldNames.forEach(fieldName => {
|
|
550
|
+
indexCreationRequest[fieldName] = options.indexType !== undefined ? options.indexType : 1;
|
|
551
|
+
});
|
|
552
|
+
const defaultOptions = {
|
|
553
|
+
background: true,
|
|
554
|
+
sparse: true
|
|
555
|
+
};
|
|
556
|
+
const indexNameOptions = indexName ? {
|
|
557
|
+
name: indexName
|
|
558
|
+
} : {};
|
|
559
|
+
const ttlOptions = options.ttl !== undefined ? {
|
|
560
|
+
expireAfterSeconds: options.ttl
|
|
561
|
+
} : {};
|
|
562
|
+
const sparseOptions = options.sparse !== undefined ? {
|
|
563
|
+
sparse: options.sparse
|
|
564
|
+
} : {};
|
|
565
|
+
const caseInsensitiveOptions = caseInsensitive ? {
|
|
566
|
+
collation: _MongoCollection.default.caseInsensitiveCollation()
|
|
567
|
+
} : {};
|
|
568
|
+
const indexOptions = {
|
|
569
|
+
...defaultOptions,
|
|
570
|
+
...caseInsensitiveOptions,
|
|
571
|
+
...indexNameOptions,
|
|
572
|
+
...ttlOptions,
|
|
573
|
+
...sparseOptions
|
|
574
|
+
};
|
|
575
|
+
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.createIndex(indexCreationRequest, indexOptions)).catch(err => this.handleError(err));
|
|
453
576
|
}
|
|
454
577
|
|
|
455
578
|
// Create a unique index. Unique indexes on nullable fields are not allowed. Since we don't
|
|
@@ -466,7 +589,7 @@ class MongoStorageAdapter {
|
|
|
466
589
|
});
|
|
467
590
|
return this._adaptiveCollection(className).then(collection => collection._ensureSparseUniqueIndexInBackground(indexCreationRequest)).catch(error => {
|
|
468
591
|
if (error.code === 11000) {
|
|
469
|
-
throw new
|
|
592
|
+
throw new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, 'Tried to ensure field uniqueness for a class that already has duplicates.');
|
|
470
593
|
}
|
|
471
594
|
throw error;
|
|
472
595
|
}).catch(err => this.handleError(err));
|
|
@@ -480,34 +603,32 @@ class MongoStorageAdapter {
|
|
|
480
603
|
}
|
|
481
604
|
|
|
482
605
|
// Executes a count.
|
|
483
|
-
count(className, schema, query, readPreference) {
|
|
606
|
+
count(className, schema, query, readPreference, _estimate, hint, comment) {
|
|
484
607
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
485
608
|
readPreference = this._parseReadPreference(readPreference);
|
|
486
|
-
return this._adaptiveCollection(className).then(collection => collection.count((0, _MongoTransform.transformWhere)(className, query, schema), {
|
|
609
|
+
return this._adaptiveCollection(className).then(collection => collection.count((0, _MongoTransform.transformWhere)(className, query, schema, true), {
|
|
487
610
|
maxTimeMS: this._maxTimeMS,
|
|
488
|
-
readPreference
|
|
611
|
+
readPreference,
|
|
612
|
+
hint,
|
|
613
|
+
comment
|
|
489
614
|
})).catch(err => this.handleError(err));
|
|
490
615
|
}
|
|
491
|
-
|
|
492
616
|
distinct(className, schema, query, fieldName) {
|
|
493
617
|
schema = convertParseSchemaToMongoSchema(schema);
|
|
494
618
|
const isPointerField = schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
}
|
|
498
|
-
return this._adaptiveCollection(className).then(collection => collection.distinct(fieldName, (0, _MongoTransform.transformWhere)(className, query, schema))).then(objects => {
|
|
619
|
+
const transformField = (0, _MongoTransform.transformKey)(className, fieldName, schema);
|
|
620
|
+
return this._adaptiveCollection(className).then(collection => collection.distinct(transformField, (0, _MongoTransform.transformWhere)(className, query, schema))).then(objects => {
|
|
499
621
|
objects = objects.filter(obj => obj != null);
|
|
500
622
|
return objects.map(object => {
|
|
501
623
|
if (isPointerField) {
|
|
502
|
-
|
|
503
|
-
return (0, _MongoTransform.transformPointerString)(schema, field, object);
|
|
624
|
+
return (0, _MongoTransform.transformPointerString)(schema, fieldName, object);
|
|
504
625
|
}
|
|
505
626
|
return (0, _MongoTransform.mongoObjectToParseObject)(className, object, schema);
|
|
506
627
|
});
|
|
507
628
|
}).catch(err => this.handleError(err));
|
|
508
629
|
}
|
|
509
|
-
|
|
510
|
-
|
|
630
|
+
aggregate(className, schema, pipeline, readPreference, hint, explain, comment) {
|
|
631
|
+
validateExplainValue(explain);
|
|
511
632
|
let isPointerField = false;
|
|
512
633
|
pipeline = pipeline.map(stage => {
|
|
513
634
|
if (stage.$group) {
|
|
@@ -522,21 +643,25 @@ class MongoStorageAdapter {
|
|
|
522
643
|
if (stage.$project) {
|
|
523
644
|
stage.$project = this._parseAggregateProjectArgs(schema, stage.$project);
|
|
524
645
|
}
|
|
646
|
+
if (stage.$geoNear && stage.$geoNear.query) {
|
|
647
|
+
stage.$geoNear.query = this._parseAggregateArgs(schema, stage.$geoNear.query);
|
|
648
|
+
}
|
|
525
649
|
return stage;
|
|
526
650
|
});
|
|
527
651
|
readPreference = this._parseReadPreference(readPreference);
|
|
528
|
-
return this._adaptiveCollection(className).then(collection => collection.aggregate(pipeline, {
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
652
|
+
return this._adaptiveCollection(className).then(collection => collection.aggregate(pipeline, {
|
|
653
|
+
readPreference,
|
|
654
|
+
maxTimeMS: this._maxTimeMS,
|
|
655
|
+
hint,
|
|
656
|
+
explain,
|
|
657
|
+
comment
|
|
658
|
+
})).then(results => {
|
|
534
659
|
results.forEach(result => {
|
|
535
|
-
if (
|
|
660
|
+
if (Object.prototype.hasOwnProperty.call(result, '_id')) {
|
|
536
661
|
if (isPointerField && result._id) {
|
|
537
662
|
result._id = result._id.split('$')[1];
|
|
538
663
|
}
|
|
539
|
-
if (result._id == null ||
|
|
664
|
+
if (result._id == null || result._id == undefined || ['object', 'string'].includes(typeof result._id) && _lodash.default.isEmpty(result._id)) {
|
|
540
665
|
result._id = null;
|
|
541
666
|
}
|
|
542
667
|
result.objectId = result._id;
|
|
@@ -567,7 +692,9 @@ class MongoStorageAdapter {
|
|
|
567
692
|
// As much as I hate recursion...this seemed like a good fit for it. We're essentially traversing
|
|
568
693
|
// down a tree to find a "leaf node" and checking to see if it needs to be converted.
|
|
569
694
|
_parseAggregateArgs(schema, pipeline) {
|
|
570
|
-
if (
|
|
695
|
+
if (pipeline === null) {
|
|
696
|
+
return null;
|
|
697
|
+
} else if (Array.isArray(pipeline)) {
|
|
571
698
|
return pipeline.map(value => this._parseAggregateArgs(schema, value));
|
|
572
699
|
} else if (typeof pipeline === 'object') {
|
|
573
700
|
const returnValue = {};
|
|
@@ -584,7 +711,6 @@ class MongoStorageAdapter {
|
|
|
584
711
|
} else {
|
|
585
712
|
returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);
|
|
586
713
|
}
|
|
587
|
-
|
|
588
714
|
if (field === 'objectId') {
|
|
589
715
|
returnValue['_id'] = returnValue[field];
|
|
590
716
|
delete returnValue[field];
|
|
@@ -613,7 +739,6 @@ class MongoStorageAdapter {
|
|
|
613
739
|
} else {
|
|
614
740
|
returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);
|
|
615
741
|
}
|
|
616
|
-
|
|
617
742
|
if (field === 'objectId') {
|
|
618
743
|
returnValue['_id'] = returnValue[field];
|
|
619
744
|
delete returnValue[field];
|
|
@@ -655,23 +780,33 @@ class MongoStorageAdapter {
|
|
|
655
780
|
return pipeline;
|
|
656
781
|
}
|
|
657
782
|
|
|
658
|
-
|
|
659
|
-
|
|
660
|
-
|
|
661
|
-
|
|
783
|
+
/**
|
|
784
|
+
* Recursively converts values to Date objects. Since the passed object is part of an aggregation
|
|
785
|
+
* pipeline and can contain various logic operators (like $gt, $lt, etc), this function will
|
|
786
|
+
* traverse the object and convert any strings that can be parsed as dates into Date objects.
|
|
787
|
+
* @param {any} value The value to convert.
|
|
788
|
+
* @returns {any} The original value if not convertible to Date, or a Date object if it is.
|
|
789
|
+
*/
|
|
662
790
|
_convertToDate(value) {
|
|
791
|
+
if (value instanceof Date) {
|
|
792
|
+
return value;
|
|
793
|
+
}
|
|
663
794
|
if (typeof value === 'string') {
|
|
664
|
-
return new Date(value);
|
|
795
|
+
return isNaN(Date.parse(value)) ? value : new Date(value);
|
|
665
796
|
}
|
|
666
|
-
|
|
667
|
-
|
|
668
|
-
|
|
669
|
-
|
|
797
|
+
if (typeof value === 'object') {
|
|
798
|
+
const returnValue = {};
|
|
799
|
+
for (const field in value) {
|
|
800
|
+
returnValue[field] = this._convertToDate(value[field]);
|
|
801
|
+
}
|
|
802
|
+
return returnValue;
|
|
670
803
|
}
|
|
671
|
-
return
|
|
804
|
+
return value;
|
|
672
805
|
}
|
|
673
|
-
|
|
674
806
|
_parseReadPreference(readPreference) {
|
|
807
|
+
if (readPreference) {
|
|
808
|
+
readPreference = readPreference.toUpperCase();
|
|
809
|
+
}
|
|
675
810
|
switch (readPreference) {
|
|
676
811
|
case 'PRIMARY':
|
|
677
812
|
readPreference = ReadPreference.PRIMARY;
|
|
@@ -689,25 +824,23 @@ class MongoStorageAdapter {
|
|
|
689
824
|
readPreference = ReadPreference.NEAREST;
|
|
690
825
|
break;
|
|
691
826
|
case undefined:
|
|
827
|
+
case null:
|
|
828
|
+
case '':
|
|
692
829
|
break;
|
|
693
830
|
default:
|
|
694
|
-
throw new
|
|
831
|
+
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, 'Not supported read preference.');
|
|
695
832
|
}
|
|
696
833
|
return readPreference;
|
|
697
834
|
}
|
|
698
|
-
|
|
699
835
|
performInitialization() {
|
|
700
836
|
return Promise.resolve();
|
|
701
837
|
}
|
|
702
|
-
|
|
703
838
|
createIndex(className, index) {
|
|
704
839
|
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.createIndex(index)).catch(err => this.handleError(err));
|
|
705
840
|
}
|
|
706
|
-
|
|
707
841
|
createIndexes(className, indexes) {
|
|
708
842
|
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.createIndexes(indexes)).catch(err => this.handleError(err));
|
|
709
843
|
}
|
|
710
|
-
|
|
711
844
|
createIndexesIfNeeded(className, fieldName, type) {
|
|
712
845
|
if (type && type.type === 'Polygon') {
|
|
713
846
|
const index = {
|
|
@@ -717,7 +850,6 @@ class MongoStorageAdapter {
|
|
|
717
850
|
}
|
|
718
851
|
return Promise.resolve();
|
|
719
852
|
}
|
|
720
|
-
|
|
721
853
|
createTextIndexesIfNeeded(className, query, schema) {
|
|
722
854
|
for (const fieldName in query) {
|
|
723
855
|
if (!query[fieldName] || !query[fieldName].$text) {
|
|
@@ -726,13 +858,15 @@ class MongoStorageAdapter {
|
|
|
726
858
|
const existingIndexes = schema.indexes;
|
|
727
859
|
for (const key in existingIndexes) {
|
|
728
860
|
const index = existingIndexes[key];
|
|
729
|
-
if (
|
|
861
|
+
if (Object.prototype.hasOwnProperty.call(index, fieldName)) {
|
|
730
862
|
return Promise.resolve();
|
|
731
863
|
}
|
|
732
864
|
}
|
|
733
865
|
const indexName = `${fieldName}_text`;
|
|
734
866
|
const textIndex = {
|
|
735
|
-
[indexName]: {
|
|
867
|
+
[indexName]: {
|
|
868
|
+
[fieldName]: 'text'
|
|
869
|
+
}
|
|
736
870
|
};
|
|
737
871
|
return this.setIndexesWithSchemaFormat(className, textIndex, existingIndexes, schema.fields).catch(error => {
|
|
738
872
|
if (error.code === 85) {
|
|
@@ -744,19 +878,15 @@ class MongoStorageAdapter {
|
|
|
744
878
|
}
|
|
745
879
|
return Promise.resolve();
|
|
746
880
|
}
|
|
747
|
-
|
|
748
881
|
getIndexes(className) {
|
|
749
882
|
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.indexes()).catch(err => this.handleError(err));
|
|
750
883
|
}
|
|
751
|
-
|
|
752
884
|
dropIndex(className, index) {
|
|
753
885
|
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.dropIndex(index)).catch(err => this.handleError(err));
|
|
754
886
|
}
|
|
755
|
-
|
|
756
887
|
dropAllIndexes(className) {
|
|
757
888
|
return this._adaptiveCollection(className).then(collection => collection._mongoCollection.dropIndexes()).catch(err => this.handleError(err));
|
|
758
889
|
}
|
|
759
|
-
|
|
760
890
|
updateSchemaWithIndexes() {
|
|
761
891
|
return this.getAllClasses().then(classes => {
|
|
762
892
|
const promises = classes.map(schema => {
|
|
@@ -765,8 +895,30 @@ class MongoStorageAdapter {
|
|
|
765
895
|
return Promise.all(promises);
|
|
766
896
|
}).catch(err => this.handleError(err));
|
|
767
897
|
}
|
|
898
|
+
createTransactionalSession() {
|
|
899
|
+
const transactionalSection = this.client.startSession();
|
|
900
|
+
transactionalSection.startTransaction();
|
|
901
|
+
return Promise.resolve(transactionalSection);
|
|
902
|
+
}
|
|
903
|
+
commitTransactionalSession(transactionalSection) {
|
|
904
|
+
const commit = retries => {
|
|
905
|
+
return transactionalSection.commitTransaction().catch(error => {
|
|
906
|
+
if (error && error.hasErrorLabel('TransientTransactionError') && retries > 0) {
|
|
907
|
+
return commit(retries - 1);
|
|
908
|
+
}
|
|
909
|
+
throw error;
|
|
910
|
+
}).then(() => {
|
|
911
|
+
transactionalSection.endSession();
|
|
912
|
+
});
|
|
913
|
+
};
|
|
914
|
+
return commit(5);
|
|
915
|
+
}
|
|
916
|
+
abortTransactionalSession(transactionalSection) {
|
|
917
|
+
return transactionalSection.abortTransaction().then(() => {
|
|
918
|
+
transactionalSection.endSession();
|
|
919
|
+
});
|
|
920
|
+
}
|
|
768
921
|
}
|
|
769
|
-
|
|
770
922
|
exports.MongoStorageAdapter = MongoStorageAdapter;
|
|
771
|
-
exports.default = MongoStorageAdapter;
|
|
772
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Adapters/Storage/Mongo/MongoStorageAdapter.js"],"names":["mongodb","require","MongoClient","ReadPreference","MongoSchemaCollectionName","storageAdapterAllCollections","mongoAdapter","connect","then","database","collections","filter","collection","namespace","match","collectionName","indexOf","_collectionPrefix","convertParseSchemaToMongoSchema","schema","fields","_rperm","_wperm","className","_hashed_password","mongoSchemaFromFieldsAndClassNameAndCLP","classLevelPermissions","indexes","mongoObject","_id","objectId","updatedAt","createdAt","_metadata","undefined","fieldName","MongoSchemaCollection","parseFieldTypeToMongoFieldType","class_permissions","Object","keys","length","MongoStorageAdapter","constructor","uri","defaults","DefaultMongoURI","collectionPrefix","mongoOptions","_uri","_mongoOptions","_maxTimeMS","maxTimeMS","canSortOnJoinTables","connectionPromise","encodedUri","client","options","s","db","dbName","on","catch","err","Promise","reject","handleError","error","code","logger","handleShutdown","close","_adaptiveCollection","name","rawCollection","MongoCollection","_schemaCollection","classExists","listCollections","toArray","setClassLevelPermissions","CLPs","schemaCollection","updateSchema","$set","setIndexesWithSchemaFormat","submittedIndexes","existingIndexes","resolve","_id_","deletePromises","insertedIndexes","forEach","field","__op","Parse","Error","INVALID_QUERY","promise","dropIndex","push","key","hasOwnProperty","insertPromise","createIndexes","all","setIndexesFromMongo","getIndexes","reduce","obj","index","_fts","_ftsx","weights","createClass","insertSchema","addFieldIfNotExists","type","createIndexesIfNeeded","deleteClass","drop","message","findAndDeleteSchema","deleteAllClasses","fast","map","remove","deleteFields","fieldNames","mongoFormatNames","collectionUpdate","schemaUpdate","updateMany","getAllClasses","schemasCollection","_fetchAllSchemasFrom_SCHEMA","getClass","_fetchOneSchemaFrom_SCHEMA","createObject","object","insertOne","DUPLICATE_VALUE","underlyingError","matches","Array","isArray","userInfo","duplicated_field","deleteObjectsByQuery","query","mongoWhere","deleteMany","result","n","OBJECT_NOT_FOUND","INTERNAL_SERVER_ERROR","updateObjectsByQuery","update","mongoUpdate","findOneAndUpdate","_mongoCollection","findAndModify","new","value","upsertOneObject","upsertOne","find","skip","limit","sort","readPreference","mongoSort","_","mapKeys","mongoKeys","memo","_parseReadPreference","createTextIndexesIfNeeded","objects","ensureUniqueness","indexCreationRequest","mongoFieldNames","_ensureSparseUniqueIndexInBackground","_rawFind","count","distinct","isPointerField","substring","aggregate","pipeline","stage","$group","_parseAggregateGroupArgs","$match","_parseAggregateArgs","$project","_parseAggregateProjectArgs","results","split","isEmpty","returnValue","targetClass","_convertToDate","Date","PRIMARY","PRIMARY_PREFERRED","SECONDARY","SECONDARY_PREFERRED","NEAREST","performInitialization","createIndex","$text","indexName","textIndex","dropAllIndexes","dropIndexes","updateSchemaWithIndexes","classes","promises"],"mappings":"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","file":"MongoStorageAdapter.js","sourcesContent":["// @flow\nimport MongoCollection       from './MongoCollection';\nimport MongoSchemaCollection from './MongoSchemaCollection';\nimport { StorageAdapter }    from '../StorageAdapter';\nimport type { SchemaType,\n  QueryType,\n  StorageClass,\n  QueryOptions } from '../StorageAdapter';\nimport {\n  parse as parseUrl,\n  format as formatUrl,\n} from '../../../vendor/mongodbUrl';\nimport {\n  parseObjectToMongoObjectForCreate,\n  mongoObjectToParseObject,\n  transformKey,\n  transformWhere,\n  transformUpdate,\n  transformPointerString,\n} from './MongoTransform';\n// @flow-disable-next\nimport Parse                 from 'parse/node';\n// @flow-disable-next\nimport _                     from 'lodash';\nimport defaults              from '../../../defaults';\nimport logger                from '../../../logger';\n\n// @flow-disable-next\nconst mongodb = require('mongodb');\nconst MongoClient = mongodb.MongoClient;\nconst ReadPreference = mongodb.ReadPreference;\n\nconst MongoSchemaCollectionName = '_SCHEMA';\n\nconst storageAdapterAllCollections = mongoAdapter => {\n  return mongoAdapter.connect()\n    .then(() => mongoAdapter.database.collections())\n    .then(collections => {\n      return collections.filter(collection => {\n        if (collection.namespace.match(/\\.system\\./)) {\n          return false;\n        }\n        // TODO: If you have one app with a collection prefix that happens to be a prefix of another\n        // apps prefix, this will go very very badly. We should fix that somehow.\n        return (collection.collectionName.indexOf(mongoAdapter._collectionPrefix) == 0);\n      });\n    });\n}\n\nconst convertParseSchemaToMongoSchema = ({...schema}) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  if (schema.className === '_User') {\n    // Legacy mongo adapter knows about the difference between password and _hashed_password.\n    // Future database adapters will only know about _hashed_password.\n    // Note: Parse Server will bring back password with injectDefaultSchema, so we don't need\n    // to add _hashed_password back ever.\n    delete schema.fields._hashed_password;\n  }\n\n  return schema;\n}\n\n// Returns { code, error } if invalid, or { result }, an object\n// suitable for inserting into _SCHEMA collection, otherwise.\nconst mongoSchemaFromFieldsAndClassNameAndCLP = (fields, className, classLevelPermissions, indexes) => {\n  const mongoObject = {\n    _id: className,\n    objectId: 'string',\n    updatedAt: 'string',\n    createdAt: 'string',\n    _metadata: undefined,\n  };\n\n  for (const fieldName in fields) {\n    mongoObject[fieldName] = MongoSchemaCollection.parseFieldTypeToMongoFieldType(fields[fieldName]);\n  }\n\n  if (typeof classLevelPermissions !== 'undefined') {\n    mongoObject._metadata = mongoObject._metadata || {};\n    if (!classLevelPermissions) {\n      delete mongoObject._metadata.class_permissions;\n    } else {\n      mongoObject._metadata.class_permissions = classLevelPermissions;\n    }\n  }\n\n  if (indexes && typeof indexes === 'object' && Object.keys(indexes).length > 0) {\n    mongoObject._metadata = mongoObject._metadata || {};\n    mongoObject._metadata.indexes = indexes;\n  }\n\n  if (!mongoObject._metadata) { // cleanup the unused _metadata\n    delete mongoObject._metadata;\n  }\n\n  return mongoObject;\n}\n\n\nexport class MongoStorageAdapter implements StorageAdapter {\n  // Private\n  _uri: string;\n  _collectionPrefix: string;\n  _mongoOptions: Object;\n  // Public\n  connectionPromise: Promise<any>;\n  database: any;\n  client: MongoClient;\n  _maxTimeMS: ?number;\n  canSortOnJoinTables: boolean;\n\n  constructor({\n    uri = defaults.DefaultMongoURI,\n    collectionPrefix = '',\n    mongoOptions = {},\n  }: any) {\n    this._uri = uri;\n    this._collectionPrefix = collectionPrefix;\n    this._mongoOptions = mongoOptions;\n\n    // MaxTimeMS is not a global MongoDB client option, it is applied per operation.\n    this._maxTimeMS = mongoOptions.maxTimeMS;\n    this.canSortOnJoinTables = true;\n    delete mongoOptions.maxTimeMS;\n  }\n\n  connect() {\n    if (this.connectionPromise) {\n      return this.connectionPromise;\n    }\n\n    // parsing and re-formatting causes the auth value (if there) to get URI\n    // encoded\n    const encodedUri = formatUrl(parseUrl(this._uri));\n\n    this.connectionPromise = MongoClient.connect(encodedUri, this._mongoOptions).then(client => {\n      // Starting mongoDB 3.0, the MongoClient.connect don't return a DB anymore but a client\n      // Fortunately, we can get back the options and use them to select the proper DB.\n      // https://github.com/mongodb/node-mongodb-native/blob/2c35d76f08574225b8db02d7bef687123e6bb018/lib/mongo_client.js#L885\n      const options = client.s.options;\n      const database = client.db(options.dbName);\n      if (!database) {\n        delete this.connectionPromise;\n        return;\n      }\n      database.on('error', () => {\n        delete this.connectionPromise;\n      });\n      database.on('close', () => {\n        delete this.connectionPromise;\n      });\n      this.client = client;\n      this.database = database;\n    }).catch((err) => {\n      delete this.connectionPromise;\n      return Promise.reject(err);\n    });\n\n    return this.connectionPromise;\n  }\n\n  handleError<T>(error: ?(Error | Parse.Error)): Promise<T> {\n    if (error && error.code === 13) { // Unauthorized error\n      delete this.client;\n      delete this.database;\n      delete this.connectionPromise;\n      logger.error('Received unauthorized error', { error: error });\n    }\n    throw error;\n  }\n\n  handleShutdown() {\n    if (!this.client) {\n      return;\n    }\n    this.client.close(false);\n  }\n\n  _adaptiveCollection(name: string) {\n    return this.connect()\n      .then(() => this.database.collection(this._collectionPrefix + name))\n      .then(rawCollection => new MongoCollection(rawCollection))\n      .catch(err => this.handleError(err));\n  }\n\n  _schemaCollection(): Promise<MongoSchemaCollection> {\n    return this.connect()\n      .then(() => this._adaptiveCollection(MongoSchemaCollectionName))\n      .then(collection => new MongoSchemaCollection(collection));\n  }\n\n  classExists(name: string) {\n    return this.connect().then(() => {\n      return this.database.listCollections({ name: this._collectionPrefix + name }).toArray();\n    }).then(collections => {\n      return collections.length > 0;\n    }).catch(err => this.handleError(err));\n  }\n\n  setClassLevelPermissions(className: string, CLPs: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.updateSchema(className, {\n        $set: { '_metadata.class_permissions': CLPs }\n      })).catch(err => this.handleError(err));\n  }\n\n  setIndexesWithSchemaFormat(className: string, submittedIndexes: any, existingIndexes: any = {}, fields: any): Promise<void> {\n    if (submittedIndexes === undefined) {\n      return Promise.resolve();\n    }\n    if (Object.keys(existingIndexes).length === 0) {\n      existingIndexes = { _id_: { _id: 1} };\n    }\n    const deletePromises = [];\n    const insertedIndexes = [];\n    Object.keys(submittedIndexes).forEach(name => {\n      const field = submittedIndexes[name];\n      if (existingIndexes[name] && field.__op !== 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} exists, cannot update.`);\n      }\n      if (!existingIndexes[name] && field.__op === 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} does not exist, cannot delete.`);\n      }\n      if (field.__op === 'Delete') {\n        const promise = this.dropIndex(className, name);\n        deletePromises.push(promise);\n        delete existingIndexes[name];\n      } else {\n        Object.keys(field).forEach(key => {\n          if (!fields.hasOwnProperty(key)) {\n            throw new Parse.Error(Parse.Error.INVALID_QUERY, `Field ${key} does not exist, cannot add index.`);\n          }\n        });\n        existingIndexes[name] = field;\n        insertedIndexes.push({\n          key: field,\n          name,\n        });\n      }\n    });\n    let insertPromise = Promise.resolve();\n    if (insertedIndexes.length > 0) {\n      insertPromise = this.createIndexes(className, insertedIndexes);\n    }\n    return Promise.all(deletePromises)\n      .then(() => insertPromise)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, {\n        $set: { '_metadata.indexes':  existingIndexes }\n      }))\n      .catch(err => this.handleError(err));\n  }\n\n  setIndexesFromMongo(className: string) {\n    return this.getIndexes(className).then((indexes) => {\n      indexes = indexes.reduce((obj, index) => {\n        if (index.key._fts) {\n          delete index.key._fts;\n          delete index.key._ftsx;\n          for (const field in index.weights) {\n            index.key[field] = 'text';\n          }\n        }\n        obj[index.name] = index.key;\n        return obj;\n      }, {});\n      return this._schemaCollection()\n        .then(schemaCollection => schemaCollection.updateSchema(className, {\n          $set: { '_metadata.indexes': indexes }\n        }));\n    })\n      .catch(err => this.handleError(err))\n      .catch(() => {\n        // Ignore if collection not found\n        return Promise.resolve();\n      });\n  }\n\n  createClass(className: string, schema: SchemaType): Promise<void> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = mongoSchemaFromFieldsAndClassNameAndCLP(schema.fields, className, schema.classLevelPermissions, schema.indexes);\n    mongoObject._id = className;\n    return this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.insertSchema(mongoObject))\n      .catch(err => this.handleError(err));\n  }\n\n  addFieldIfNotExists(className: string, fieldName: string, type: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.addFieldIfNotExists(className, fieldName, type))\n      .then(() => this.createIndexesIfNeeded(className, fieldName, type))\n      .catch(err => this.handleError(err));\n  }\n\n  // Drops a collection. Resolves with true if it was a Parse Schema (eg. _User, Custom, etc.)\n  // and resolves with false if it wasn't (eg. a join table). Rejects if deletion was impossible.\n  deleteClass(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection.drop())\n      .catch(error => {\n      // 'ns not found' means collection was already gone. Ignore deletion attempt.\n        if (error.message == 'ns not found') {\n          return;\n        }\n        throw error;\n      })\n    // We've dropped the collection, now remove the _SCHEMA document\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.findAndDeleteSchema(className))\n      .catch(err => this.handleError(err));\n  }\n\n  deleteAllClasses(fast: boolean) {\n    return storageAdapterAllCollections(this)\n      .then(collections => Promise.all(collections.map(collection => fast ? collection.remove({}) : collection.drop())));\n  }\n\n  // Remove the column and all the data. For Relations, the _Join collection is handled\n  // specially, this function does not delete _Join columns. It should, however, indicate\n  // that the relation fields does not exist anymore. In mongo, this means removing it from\n  // the _SCHEMA collection.  There should be no actual data in the collection under the same name\n  // as the relation column, so it's fine to attempt to delete it. If the fields listed to be\n  // deleted do not exist, this function should return successfully anyways. Checking for\n  // attempts to delete non-existent fields is the responsibility of Parse Server.\n\n  // Pointer field names are passed for legacy reasons: the original mongo\n  // format stored pointer field names differently in the database, and therefore\n  // needed to know the type of the field before it could delete it. Future database\n  // adapters should ignore the pointerFieldNames argument. All the field names are in\n  // fieldNames, they show up additionally in the pointerFieldNames database for use\n  // by the mongo adapter, which deals with the legacy mongo format.\n\n  // This function is not obligated to delete fields atomically. It is given the field\n  // names in a list so that databases that are capable of deleting fields atomically\n  // may do so.\n\n  // Returns a Promise.\n  deleteFields(className: string, schema: SchemaType, fieldNames: string[]) {\n    const mongoFormatNames = fieldNames.map(fieldName => {\n      if (schema.fields[fieldName].type === 'Pointer') {\n        return `_p_${fieldName}`\n      } else {\n        return fieldName;\n      }\n    });\n    const collectionUpdate = { '$unset' : {} };\n    mongoFormatNames.forEach(name => {\n      collectionUpdate['$unset'][name] = null;\n    });\n\n    const schemaUpdate = { '$unset' : {} };\n    fieldNames.forEach(name => {\n      schemaUpdate['$unset'][name] = null;\n    });\n\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany({}, collectionUpdate))\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, schemaUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for all schemas known to this adapter, in Parse format. In case the\n  // schemas cannot be retrieved, returns a promise that rejects. Requirements for the\n  // rejection reason are TBD.\n  getAllClasses(): Promise<StorageClass[]> {\n    return this._schemaCollection().then(schemasCollection => schemasCollection._fetchAllSchemasFrom_SCHEMA())\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for the schema with the given name, in Parse format. If\n  // this adapter doesn't know about the schema, return a promise that rejects with\n  // undefined as the reason.\n  getClass(className: string): Promise<StorageClass> {\n    return this._schemaCollection()\n      .then(schemasCollection => schemasCollection._fetchOneSchemaFrom_SCHEMA(className))\n      .catch(err => this.handleError(err));\n  }\n\n  // TODO: As yet not particularly well specified. Creates an object. Maybe shouldn't even need the schema,\n  // and should infer from the type. Or maybe does need the schema for validations. Or maybe needs\n  // the schema only for the legacy mongo format. We'll figure that out later.\n  createObject(className: string, schema: SchemaType, object: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = parseObjectToMongoObjectForCreate(className, object, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.insertOne(mongoObject))\n      .catch(error => {\n        if (error.code === 11000) { // Duplicate value\n          const err = new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');\n          err.underlyingError = error;\n          if (error.message) {\n            const matches = error.message.match(/index:[\\sa-zA-Z0-9_\\-\\.]+\\$?([a-zA-Z_-]+)_1/);\n            if (matches && Array.isArray(matches)) {\n              err.userInfo = { duplicated_field: matches[1] };\n            }\n          }\n          throw err;\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Remove all objects that match the given Parse Query.\n  // If no objects match, reject with OBJECT_NOT_FOUND. If objects are found and deleted, resolve with undefined.\n  // If there is some other error, reject with INTERNAL_SERVER_ERROR.\n  deleteObjectsByQuery(className: string, schema: SchemaType, query: QueryType) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    return this._adaptiveCollection(className)\n      .then(collection => {\n        const mongoWhere = transformWhere(className, query, schema);\n        return collection.deleteMany(mongoWhere)\n      })\n      .catch(err => this.handleError(err))\n      .then(({ result }) => {\n        if (result.n === 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        }\n        return Promise.resolve();\n      }, () => {\n        throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Database adapter error');\n      });\n  }\n\n  // Apply the update to all objects that match the given Parse Query.\n  updateObjectsByQuery(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany(mongoWhere, mongoUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Atomically finds and updates an object based on query.\n  // Return value not currently well specified.\n  findOneAndUpdate(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.findAndModify(mongoWhere, [], mongoUpdate, { new: true }))\n      .then(result => mongoObjectToParseObject(className, result.value, schema))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Hopefully we can get rid of this. It's only used for config and hooks.\n  upsertOneObject(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.upsertOne(mongoWhere, mongoUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Executes a find. Accepts: className, query in Parse format, and { skip, limit, sort }.\n  find(className: string, schema: SchemaType, query: QueryType, { skip, limit, sort, keys, readPreference }: QueryOptions): Promise<any> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    const mongoSort = _.mapKeys(sort, (value, fieldName) => transformKey(className, fieldName, schema));\n    const mongoKeys = _.reduce(keys, (memo, key) => {\n      memo[transformKey(className, key, schema)] = 1;\n      return memo;\n    }, {});\n\n    readPreference = this._parseReadPreference(readPreference);\n    return this.createTextIndexesIfNeeded(className, query, schema)\n      .then(() => this._adaptiveCollection(className))\n      .then(collection => collection.find(mongoWhere, {\n        skip,\n        limit,\n        sort: mongoSort,\n        keys: mongoKeys,\n        maxTimeMS: this._maxTimeMS,\n        readPreference,\n      }))\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // Create a unique index. Unique indexes on nullable fields are not allowed. Since we don't\n  // currently know which fields are nullable and which aren't, we ignore that criteria.\n  // As such, we shouldn't expose this function to users of parse until we have an out-of-band\n  // Way of determining if a field is nullable. Undefined doesn't count against uniqueness,\n  // which is why we use sparse indexes.\n  ensureUniqueness(className: string, schema: SchemaType, fieldNames: string[]) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const indexCreationRequest = {};\n    const mongoFieldNames = fieldNames.map(fieldName => transformKey(className, fieldName, schema));\n    mongoFieldNames.forEach(fieldName => {\n      indexCreationRequest[fieldName] = 1;\n    });\n    return this._adaptiveCollection(className)\n      .then(collection => collection._ensureSparseUniqueIndexInBackground(indexCreationRequest))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'Tried to ensure field uniqueness for a class that already has duplicates.');\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Used in tests\n  _rawFind(className: string, query: QueryType) {\n    return this._adaptiveCollection(className).then(collection => collection.find(query, {\n      maxTimeMS: this._maxTimeMS,\n    })).catch(err => this.handleError(err));\n  }\n\n  // Executes a count.\n  count(className: string, schema: SchemaType, query: QueryType, readPreference: ?string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.count(transformWhere(className, query, schema), {\n        maxTimeMS: this._maxTimeMS,\n        readPreference,\n      }))\n      .catch(err => this.handleError(err));\n  }\n\n  distinct(className: string, schema: SchemaType, query: QueryType, fieldName: string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const isPointerField = schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';\n    if (isPointerField) {\n      fieldName = `_p_${fieldName}`\n    }\n    return this._adaptiveCollection(className)\n      .then(collection => collection.distinct(fieldName, transformWhere(className, query, schema)))\n      .then(objects => {\n        objects = objects.filter((obj) => obj != null);\n        return objects.map(object => {\n          if (isPointerField) {\n            const field = fieldName.substring(3);\n            return transformPointerString(schema, field, object);\n          }\n          return mongoObjectToParseObject(className, object, schema);\n        });\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  aggregate(className: string, schema: any, pipeline: any, readPreference: ?string) {\n    let isPointerField = false;\n    pipeline = pipeline.map((stage) => {\n      if (stage.$group) {\n        stage.$group = this._parseAggregateGroupArgs(schema, stage.$group);\n        if (stage.$group._id && (typeof stage.$group._id === 'string') && stage.$group._id.indexOf('$_p_') >= 0) {\n          isPointerField = true;\n        }\n      }\n      if (stage.$match) {\n        stage.$match = this._parseAggregateArgs(schema, stage.$match);\n      }\n      if (stage.$project) {\n        stage.$project = this._parseAggregateProjectArgs(schema, stage.$project);\n      }\n      return stage;\n    });\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.aggregate(pipeline, { readPreference, maxTimeMS: this._maxTimeMS }))\n      .catch(error => {\n        if (error.code === 16006) {\n          throw new Parse.Error(Parse.Error.INVALID_QUERY, error.message);\n        }\n        throw error;\n      })\n      .then(results => {\n        results.forEach(result => {\n          if (result.hasOwnProperty('_id')) {\n            if (isPointerField && result._id) {\n              result._id = result._id.split('$')[1];\n            }\n            if (result._id == null || _.isEmpty(result._id)) {\n              result._id = null;\n            }\n            result.objectId = result._id;\n            delete result._id;\n          }\n        });\n        return results;\n      })\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // This function will recursively traverse the pipeline and convert any Pointer or Date columns.\n  // If we detect a pointer column we will rename the column being queried for to match the column\n  // in the database. We also modify the value to what we expect the value to be in the database\n  // as well.\n  // For dates, the driver expects a Date object, but we have a string coming in. So we'll convert\n  // the string to a Date so the driver can perform the necessary comparison.\n  //\n  // The goal of this method is to look for the \"leaves\" of the pipeline and determine if it needs\n  // to be converted. The pipeline can have a few different forms. For more details, see:\n  //     https://docs.mongodb.com/manual/reference/operator/aggregation/\n  //\n  // If the pipeline is an array, it means we are probably parsing an '$and' or '$or' operator. In\n  // that case we need to loop through all of it's children to find the columns being operated on.\n  // If the pipeline is an object, then we'll loop through the keys checking to see if the key name\n  // matches one of the schema columns. If it does match a column and the column is a Pointer or\n  // a Date, then we'll convert the value as described above.\n  //\n  // As much as I hate recursion...this seemed like a good fit for it. We're essentially traversing\n  // down a tree to find a \"leaf node\" and checking to see if it needs to be converted.\n  _parseAggregateArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map((value) => this._parseAggregateArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n          if (typeof pipeline[field] === 'object') {\n            // Pass objects down to MongoDB...this is more than likely an $exists operator.\n            returnValue[`_p_${field}`] = pipeline[field];\n          } else {\n            returnValue[`_p_${field}`] = `${schema.fields[field].targetClass}$${pipeline[field]}`;\n          }\n        } else if (schema.fields[field] && schema.fields[field].type === 'Date') {\n          returnValue[field] = this._convertToDate(pipeline[field]);\n        } else {\n          returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n        }\n\n        if (field === 'objectId') {\n          returnValue['_id'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'createdAt') {\n          returnValue['_created_at'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'updatedAt') {\n          returnValue['_updated_at'] = returnValue[field];\n          delete returnValue[field];\n        }\n      }\n      return returnValue;\n    }\n    return pipeline;\n  }\n\n  // This function is slightly different than the one above. Rather than trying to combine these\n  // two functions and making the code even harder to understand, I decided to split it up. The\n  // difference with this function is we are not transforming the values, only the keys of the\n  // pipeline.\n  _parseAggregateProjectArgs(schema: any, pipeline: any): any {\n    const returnValue = {};\n    for (const field in pipeline) {\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        returnValue[`_p_${field}`] = pipeline[field];\n      } else {\n        returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n      }\n\n      if (field === 'objectId') {\n        returnValue['_id'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'createdAt') {\n        returnValue['_created_at'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'updatedAt') {\n        returnValue['_updated_at'] = returnValue[field];\n        delete returnValue[field];\n      }\n    }\n    return returnValue;\n  }\n\n  // This function is slightly different than the two above. MongoDB $group aggregate looks like:\n  //     { $group: { _id: <expression>, <field1>: { <accumulator1> : <expression1> }, ... } }\n  // The <expression> could be a column name, prefixed with the '$' character. We'll look for\n  // these <expression> and check to see if it is a 'Pointer' or if it's one of createdAt,\n  // updatedAt or objectId and change it accordingly.\n  _parseAggregateGroupArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map((value) => this._parseAggregateGroupArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        returnValue[field] = this._parseAggregateGroupArgs(schema, pipeline[field]);\n      }\n      return returnValue;\n    } else if (typeof pipeline === 'string') {\n      const field = pipeline.substring(1);\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        return `$_p_${field}`;\n      } else if (field == 'createdAt') {\n        return '$_created_at';\n      } else if (field == 'updatedAt') {\n        return '$_updated_at';\n      }\n    }\n    return pipeline;\n  }\n\n  // This function will attempt to convert the provided value to a Date object. Since this is part\n  // of an aggregation pipeline, the value can either be a string or it can be another object with\n  // an operator in it (like $gt, $lt, etc). Because of this I felt it was easier to make this a\n  // recursive method to traverse down to the \"leaf node\" which is going to be the string.\n  _convertToDate(value: any): any {\n    if (typeof value === 'string') {\n      return new Date(value);\n    }\n\n    const returnValue = {}\n    for (const field in value) {\n      returnValue[field] = this._convertToDate(value[field])\n    }\n    return returnValue;\n  }\n\n  _parseReadPreference(readPreference: ?string): ?string {\n    switch (readPreference) {\n    case 'PRIMARY':\n      readPreference = ReadPreference.PRIMARY;\n      break;\n    case 'PRIMARY_PREFERRED':\n      readPreference = ReadPreference.PRIMARY_PREFERRED;\n      break;\n    case 'SECONDARY':\n      readPreference = ReadPreference.SECONDARY;\n      break;\n    case 'SECONDARY_PREFERRED':\n      readPreference = ReadPreference.SECONDARY_PREFERRED;\n      break;\n    case 'NEAREST':\n      readPreference = ReadPreference.NEAREST;\n      break;\n    case undefined:\n      break;\n    default:\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Not supported read preference.');\n    }\n    return readPreference;\n  }\n\n  performInitialization(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  createIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndex(index))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexes(className: string, indexes: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndexes(indexes))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexesIfNeeded(className: string, fieldName: string, type: any) {\n    if (type && type.type === 'Polygon') {\n      const index = {\n        [fieldName]: '2dsphere'\n      };\n      return this.createIndex(className, index);\n    }\n    return Promise.resolve();\n  }\n\n  createTextIndexesIfNeeded(className: string, query: QueryType, schema: any): Promise<void> {\n    for(const fieldName in query) {\n      if (!query[fieldName] || !query[fieldName].$text) {\n        continue;\n      }\n      const existingIndexes = schema.indexes;\n      for (const key in existingIndexes) {\n        const index = existingIndexes[key];\n        if (index.hasOwnProperty(fieldName)) {\n          return Promise.resolve();\n        }\n      }\n      const indexName = `${fieldName}_text`;\n      const textIndex = {\n        [indexName]: { [fieldName]: 'text' }\n      };\n      return this.setIndexesWithSchemaFormat(className, textIndex, existingIndexes, schema.fields)\n        .catch((error) => {\n          if (error.code === 85) { // Index exist with different options\n            return this.setIndexesFromMongo(className);\n          }\n          throw error;\n        });\n    }\n    return Promise.resolve();\n  }\n\n  getIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.indexes())\n      .catch(err => this.handleError(err));\n  }\n\n  dropIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndex(index))\n      .catch(err => this.handleError(err));\n  }\n\n  dropAllIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndexes())\n      .catch(err => this.handleError(err));\n  }\n\n  updateSchemaWithIndexes(): Promise<any> {\n    return this.getAllClasses()\n      .then((classes) => {\n        const promises = classes.map((schema) => {\n          return this.setIndexesFromMongo(schema.className);\n        });\n        return Promise.all(promises);\n      })\n      .catch(err => this.handleError(err));\n  }\n}\n\nexport default MongoStorageAdapter;\n"]}
|
|
923
|
+
var _default = exports.default = MongoStorageAdapter;
|
|
924
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_mongodbUrl","require","_StorageAdapter","_MongoCollection","_interopRequireDefault","_MongoSchemaCollection","_MongoTransform","_node","_lodash","_defaults","_interopRequireWildcard","_logger","_Utils","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","mongodb","MongoClient","ReadPreference","MongoSchemaCollectionName","storageAdapterAllCollections","mongoAdapter","connect","then","database","collections","filter","collection","namespace","match","collectionName","indexOf","_collectionPrefix","convertParseSchemaToMongoSchema","schema","fields","_rperm","_wperm","className","_hashed_password","mongoSchemaFromFieldsAndClassNameAndCLP","classLevelPermissions","indexes","mongoObject","_id","objectId","updatedAt","createdAt","_metadata","undefined","fieldName","type","targetClass","fieldOptions","MongoSchemaCollection","parseFieldTypeToMongoFieldType","keys","length","fields_options","class_permissions","validateExplainValue","explain","explainAllowedValues","includes","Parse","Error","INVALID_QUERY","MongoStorageAdapter","constructor","uri","defaults","DefaultMongoURI","collectionPrefix","mongoOptions","_uri","_onchange","_maxTimeMS","maxTimeMS","canSortOnJoinTables","enableSchemaHooks","schemaCacheTtl","disableIndexFieldValidation","_logClientEvents","logClientEvents","_mongoOptions","key","ParseServerDatabaseOptions","watch","callback","connectionPromise","encodedUri","formatUrl","parseUrl","client","options","s","db","dbName","on","Array","isArray","forEach","eventConfig","name","event","logData","keyPath","_","logLevel","logger","logMessage","JSON","stringify","Utils","getCircularReplacer","catch","err","Promise","reject","handleError","error","code","handleShutdown","close","_adaptiveCollection","rawCollection","MongoCollection","_schemaCollection","_stream","_mongoCollection","classExists","listCollections","toArray","setClassLevelPermissions","CLPs","schemaCollection","updateSchema","$set","setIndexesWithSchemaFormat","submittedIndexes","existingIndexes","resolve","_id_","deletePromises","insertedIndexes","field","__op","promise","dropIndex","push","prototype","replace","insertPromise","createIndexes","all","setIndexesFromMongo","getIndexes","reduce","obj","index","_fts","_ftsx","weights","createClass","insertSchema","updateFieldOptions","addFieldIfNotExists","createIndexesIfNeeded","deleteClass","drop","message","findAndDeleteSchema","deleteAllClasses","fast","map","deleteMany","deleteFields","fieldNames","mongoFormatNames","collectionUpdate","$unset","collectionFilter","$or","$exists","schemaUpdate","updateMany","getAllClasses","schemasCollection","_fetchAllSchemasFrom_SCHEMA","getClass","_fetchOneSchemaFrom_SCHEMA","createObject","object","transactionalSession","parseObjectToMongoObjectForCreate","insertOne","ops","DUPLICATE_VALUE","underlyingError","matches","userInfo","duplicated_field","deleteObjectsByQuery","query","mongoWhere","transformWhere","deletedCount","OBJECT_NOT_FOUND","INTERNAL_SERVER_ERROR","updateObjectsByQuery","update","mongoUpdate","transformUpdate","findOneAndUpdate","returnDocument","session","result","mongoObjectToParseObject","upsertOneObject","upsertOne","find","skip","limit","sort","readPreference","hint","caseInsensitive","comment","mongoSort","mapKeys","value","transformKey","mongoKeys","memo","_parseReadPreference","createTextIndexesIfNeeded","objects","ensureIndex","indexName","indexCreationRequest","mongoFieldNames","indexType","defaultOptions","background","sparse","indexNameOptions","ttlOptions","ttl","expireAfterSeconds","sparseOptions","caseInsensitiveOptions","collation","caseInsensitiveCollation","indexOptions","createIndex","ensureUniqueness","_ensureSparseUniqueIndexInBackground","_rawFind","count","_estimate","distinct","isPointerField","transformField","transformPointerString","aggregate","pipeline","stage","$group","_parseAggregateGroupArgs","$match","_parseAggregateArgs","$project","_parseAggregateProjectArgs","$geoNear","results","split","isEmpty","returnValue","_convertToDate","substring","Date","isNaN","parse","toUpperCase","PRIMARY","PRIMARY_PREFERRED","SECONDARY","SECONDARY_PREFERRED","NEAREST","performInitialization","$text","textIndex","dropAllIndexes","dropIndexes","updateSchemaWithIndexes","classes","promises","createTransactionalSession","transactionalSection","startSession","startTransaction","commitTransactionalSession","commit","retries","commitTransaction","hasErrorLabel","endSession","abortTransactionalSession","abortTransaction","exports","_default"],"sources":["../../../../src/Adapters/Storage/Mongo/MongoStorageAdapter.js"],"sourcesContent":["// @flow\nimport { format as formatUrl, parse as parseUrl } from '../../../vendor/mongodbUrl';\nimport type { QueryOptions, QueryType, SchemaType, StorageClass } from '../StorageAdapter';\nimport { StorageAdapter } from '../StorageAdapter';\nimport MongoCollection from './MongoCollection';\nimport MongoSchemaCollection from './MongoSchemaCollection';\nimport {\n  mongoObjectToParseObject,\n  parseObjectToMongoObjectForCreate,\n  transformKey,\n  transformPointerString,\n  transformUpdate,\n  transformWhere,\n} from './MongoTransform';\n// @flow-disable-next\nimport Parse from 'parse/node';\n// @flow-disable-next\nimport _ from 'lodash';\nimport defaults, { ParseServerDatabaseOptions } from '../../../defaults';\nimport logger from '../../../logger';\nimport Utils from '../../../Utils';\n\n// @flow-disable-next\nconst mongodb = require('mongodb');\nconst MongoClient = mongodb.MongoClient;\nconst ReadPreference = mongodb.ReadPreference;\n\nconst MongoSchemaCollectionName = '_SCHEMA';\n\nconst storageAdapterAllCollections = mongoAdapter => {\n  return mongoAdapter\n    .connect()\n    .then(() => mongoAdapter.database.collections())\n    .then(collections => {\n      return collections.filter(collection => {\n        if (collection.namespace.match(/\\.system\\./)) {\n          return false;\n        }\n        // TODO: If you have one app with a collection prefix that happens to be a prefix of another\n        // apps prefix, this will go very very badly. We should fix that somehow.\n        return collection.collectionName.indexOf(mongoAdapter._collectionPrefix) == 0;\n      });\n    });\n};\n\nconst convertParseSchemaToMongoSchema = ({ ...schema }) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  if (schema.className === '_User') {\n    // Legacy mongo adapter knows about the difference between password and _hashed_password.\n    // Future database adapters will only know about _hashed_password.\n    // Note: Parse Server will bring back password with injectDefaultSchema, so we don't need\n    // to add _hashed_password back ever.\n    delete schema.fields._hashed_password;\n  }\n\n  return schema;\n};\n\n// Returns { code, error } if invalid, or { result }, an object\n// suitable for inserting into _SCHEMA collection, otherwise.\nconst mongoSchemaFromFieldsAndClassNameAndCLP = (\n  fields,\n  className,\n  classLevelPermissions,\n  indexes\n) => {\n  const mongoObject = {\n    _id: className,\n    objectId: 'string',\n    updatedAt: 'string',\n    createdAt: 'string',\n    _metadata: undefined,\n  };\n\n  for (const fieldName in fields) {\n    const { type, targetClass, ...fieldOptions } = fields[fieldName];\n    mongoObject[fieldName] = MongoSchemaCollection.parseFieldTypeToMongoFieldType({\n      type,\n      targetClass,\n    });\n    if (fieldOptions && Object.keys(fieldOptions).length > 0) {\n      mongoObject._metadata = mongoObject._metadata || {};\n      mongoObject._metadata.fields_options = mongoObject._metadata.fields_options || {};\n      mongoObject._metadata.fields_options[fieldName] = fieldOptions;\n    }\n  }\n\n  if (typeof classLevelPermissions !== 'undefined') {\n    mongoObject._metadata = mongoObject._metadata || {};\n    if (!classLevelPermissions) {\n      delete mongoObject._metadata.class_permissions;\n    } else {\n      mongoObject._metadata.class_permissions = classLevelPermissions;\n    }\n  }\n\n  if (indexes && typeof indexes === 'object' && Object.keys(indexes).length > 0) {\n    mongoObject._metadata = mongoObject._metadata || {};\n    mongoObject._metadata.indexes = indexes;\n  }\n\n  if (!mongoObject._metadata) {\n    // cleanup the unused _metadata\n    delete mongoObject._metadata;\n  }\n\n  return mongoObject;\n};\n\nfunction validateExplainValue(explain) {\n  if (explain) {\n    // The list of allowed explain values is from node-mongodb-native/lib/explain.js\n    const explainAllowedValues = [\n      'queryPlanner',\n      'queryPlannerExtended',\n      'executionStats',\n      'allPlansExecution',\n      false,\n      true,\n    ];\n    if (!explainAllowedValues.includes(explain)) {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Invalid value for explain');\n    }\n  }\n}\n\nexport class MongoStorageAdapter implements StorageAdapter {\n  // Private\n  _uri: string;\n  _collectionPrefix: string;\n  _mongoOptions: Object;\n  _onchange: any;\n  _stream: any;\n  _logClientEvents: ?Array<any>;\n  // Public\n  connectionPromise: ?Promise<any>;\n  database: any;\n  client: MongoClient;\n  _maxTimeMS: ?number;\n  canSortOnJoinTables: boolean;\n  enableSchemaHooks: boolean;\n  schemaCacheTtl: ?number;\n  disableIndexFieldValidation: boolean;\n\n  constructor({ uri = defaults.DefaultMongoURI, collectionPrefix = '', mongoOptions = {} }: any) {\n    this._uri = uri;\n    this._collectionPrefix = collectionPrefix;\n    this._onchange = () => {};\n\n    // MaxTimeMS is not a global MongoDB client option, it is applied per operation.\n    this._maxTimeMS = mongoOptions.maxTimeMS;\n    this.canSortOnJoinTables = true;\n    this.enableSchemaHooks = !!mongoOptions.enableSchemaHooks;\n    this.schemaCacheTtl = mongoOptions.schemaCacheTtl;\n    this.disableIndexFieldValidation = !!mongoOptions.disableIndexFieldValidation;\n    this._logClientEvents = mongoOptions.logClientEvents;\n\n    // Create a copy of mongoOptions and remove Parse Server-specific options that should not\n    // be passed to MongoDB client. Note: We only delete from this._mongoOptions, not from the\n    // original mongoOptions object, because other components (like DatabaseController) need\n    // access to these options.\n    this._mongoOptions = { ...mongoOptions };\n    for (const key of ParseServerDatabaseOptions) {\n      delete this._mongoOptions[key];\n    }\n  }\n\n  watch(callback: () => void): void {\n    this._onchange = callback;\n  }\n\n  connect() {\n    if (this.connectionPromise) {\n      return this.connectionPromise;\n    }\n\n    // parsing and re-formatting causes the auth value (if there) to get URI\n    // encoded\n    const encodedUri = formatUrl(parseUrl(this._uri));\n    this.connectionPromise = MongoClient.connect(encodedUri, this._mongoOptions)\n      .then(client => {\n        // Starting mongoDB 3.0, the MongoClient.connect don't return a DB anymore but a client\n        // Fortunately, we can get back the options and use them to select the proper DB.\n        // https://github.com/mongodb/node-mongodb-native/blob/2c35d76f08574225b8db02d7bef687123e6bb018/lib/mongo_client.js#L885\n        const options = client.s.options;\n        const database = client.db(options.dbName);\n        if (!database) {\n          delete this.connectionPromise;\n          return;\n        }\n        client.on('error', () => {\n          delete this.connectionPromise;\n        });\n        client.on('close', () => {\n          delete this.connectionPromise;\n        });\n\n        // Set up client event logging if configured\n        if (this._logClientEvents && Array.isArray(this._logClientEvents)) {\n          this._logClientEvents.forEach(eventConfig => {\n            client.on(eventConfig.name, event => {\n              let logData = {};\n              if (!eventConfig.keys || eventConfig.keys.length === 0) {\n                logData = event;\n              } else {\n                eventConfig.keys.forEach(keyPath => {\n                  logData[keyPath] = _.get(event, keyPath);\n                });\n              }\n\n              // Validate log level exists, fallback to 'info'\n              const logLevel = typeof logger[eventConfig.logLevel] === 'function' ? eventConfig.logLevel : 'info';\n\n              // Safe JSON serialization with Map/Set and circular reference support\n              const logMessage = `MongoDB client event ${eventConfig.name}: ${JSON.stringify(logData, Utils.getCircularReplacer())}`;\n\n              logger[logLevel](logMessage);\n            });\n          });\n        }\n\n        this.client = client;\n        this.database = database;\n      })\n      .catch(err => {\n        delete this.connectionPromise;\n        return Promise.reject(err);\n      });\n\n    return this.connectionPromise;\n  }\n\n  handleError<T>(error: ?(Error | Parse.Error)): Promise<T> {\n    if (error && error.code === 13) {\n      // Unauthorized error\n      delete this.client;\n      delete this.database;\n      delete this.connectionPromise;\n      logger.error('Received unauthorized error', { error: error });\n    }\n    throw error;\n  }\n\n  async handleShutdown() {\n    if (!this.client) {\n      return;\n    }\n    await this.client.close(false);\n    delete this.connectionPromise;\n  }\n\n  _adaptiveCollection(name: string) {\n    return this.connect()\n      .then(() => this.database.collection(this._collectionPrefix + name))\n      .then(rawCollection => new MongoCollection(rawCollection))\n      .catch(err => this.handleError(err));\n  }\n\n  _schemaCollection(): Promise<MongoSchemaCollection> {\n    return this.connect()\n      .then(() => this._adaptiveCollection(MongoSchemaCollectionName))\n      .then(collection => {\n        if (!this._stream && this.enableSchemaHooks) {\n          this._stream = collection._mongoCollection.watch();\n          this._stream.on('change', () => this._onchange());\n        }\n        return new MongoSchemaCollection(collection);\n      });\n  }\n\n  classExists(name: string) {\n    return this.connect()\n      .then(() => {\n        return this.database.listCollections({ name: this._collectionPrefix + name }).toArray();\n      })\n      .then(collections => {\n        return collections.length > 0;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  setClassLevelPermissions(className: string, CLPs: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection =>\n        schemaCollection.updateSchema(className, {\n          $set: { '_metadata.class_permissions': CLPs },\n        })\n      )\n      .catch(err => this.handleError(err));\n  }\n\n  setIndexesWithSchemaFormat(\n    className: string,\n    submittedIndexes: any,\n    existingIndexes: any = {},\n    fields: any\n  ): Promise<void> {\n    if (submittedIndexes === undefined) {\n      return Promise.resolve();\n    }\n    if (Object.keys(existingIndexes).length === 0) {\n      existingIndexes = { _id_: { _id: 1 } };\n    }\n    const deletePromises = [];\n    const insertedIndexes = [];\n    Object.keys(submittedIndexes).forEach(name => {\n      const field = submittedIndexes[name];\n      if (existingIndexes[name] && field.__op !== 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} exists, cannot update.`);\n      }\n      if (!existingIndexes[name] && field.__op === 'Delete') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_QUERY,\n          `Index ${name} does not exist, cannot delete.`\n        );\n      }\n      if (field.__op === 'Delete') {\n        const promise = this.dropIndex(className, name);\n        deletePromises.push(promise);\n        delete existingIndexes[name];\n      } else {\n        Object.keys(field).forEach(key => {\n          if (\n            !this.disableIndexFieldValidation &&\n            !Object.prototype.hasOwnProperty.call(\n              fields,\n              key.indexOf('_p_') === 0 ? key.replace('_p_', '') : key\n            )\n          ) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_QUERY,\n              `Field ${key} does not exist, cannot add index.`\n            );\n          }\n        });\n        existingIndexes[name] = field;\n        insertedIndexes.push({\n          key: field,\n          name,\n        });\n      }\n    });\n    let insertPromise = Promise.resolve();\n    if (insertedIndexes.length > 0) {\n      insertPromise = this.createIndexes(className, insertedIndexes);\n    }\n    return Promise.all(deletePromises)\n      .then(() => insertPromise)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection =>\n        schemaCollection.updateSchema(className, {\n          $set: { '_metadata.indexes': existingIndexes },\n        })\n      )\n      .catch(err => this.handleError(err));\n  }\n\n  setIndexesFromMongo(className: string) {\n    return this.getIndexes(className)\n      .then(indexes => {\n        indexes = indexes.reduce((obj, index) => {\n          if (index.key._fts) {\n            delete index.key._fts;\n            delete index.key._ftsx;\n            for (const field in index.weights) {\n              index.key[field] = 'text';\n            }\n          }\n          obj[index.name] = index.key;\n          return obj;\n        }, {});\n        return this._schemaCollection().then(schemaCollection =>\n          schemaCollection.updateSchema(className, {\n            $set: { '_metadata.indexes': indexes },\n          })\n        );\n      })\n      .catch(err => this.handleError(err))\n      .catch(() => {\n        // Ignore if collection not found\n        return Promise.resolve();\n      });\n  }\n\n  createClass(className: string, schema: SchemaType): Promise<void> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = mongoSchemaFromFieldsAndClassNameAndCLP(\n      schema.fields,\n      className,\n      schema.classLevelPermissions,\n      schema.indexes\n    );\n    mongoObject._id = className;\n    return this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.insertSchema(mongoObject))\n      .catch(err => this.handleError(err));\n  }\n\n  async updateFieldOptions(className: string, fieldName: string, type: any) {\n    const schemaCollection = await this._schemaCollection();\n    await schemaCollection.updateFieldOptions(className, fieldName, type);\n  }\n\n  addFieldIfNotExists(className: string, fieldName: string, type: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.addFieldIfNotExists(className, fieldName, type))\n      .then(() => this.createIndexesIfNeeded(className, fieldName, type))\n      .catch(err => this.handleError(err));\n  }\n\n  // Drops a collection. Resolves with true if it was a Parse Schema (eg. _User, Custom, etc.)\n  // and resolves with false if it wasn't (eg. a join table). Rejects if deletion was impossible.\n  deleteClass(className: string) {\n    return (\n      this._adaptiveCollection(className)\n        .then(collection => collection.drop())\n        .catch(error => {\n          // 'ns not found' means collection was already gone. Ignore deletion attempt.\n          if (error.message == 'ns not found') {\n            return;\n          }\n          throw error;\n        })\n        // We've dropped the collection, now remove the _SCHEMA document\n        .then(() => this._schemaCollection())\n        .then(schemaCollection => schemaCollection.findAndDeleteSchema(className))\n        .catch(err => this.handleError(err))\n    );\n  }\n\n  deleteAllClasses(fast: boolean) {\n    return storageAdapterAllCollections(this).then(collections =>\n      Promise.all(\n        collections.map(collection => (fast ? collection.deleteMany({}) : collection.drop()))\n      )\n    );\n  }\n\n  // Remove the column and all the data. For Relations, the _Join collection is handled\n  // specially, this function does not delete _Join columns. It should, however, indicate\n  // that the relation fields does not exist anymore. In mongo, this means removing it from\n  // the _SCHEMA collection.  There should be no actual data in the collection under the same name\n  // as the relation column, so it's fine to attempt to delete it. If the fields listed to be\n  // deleted do not exist, this function should return successfully anyways. Checking for\n  // attempts to delete non-existent fields is the responsibility of Parse Server.\n\n  // Pointer field names are passed for legacy reasons: the original mongo\n  // format stored pointer field names differently in the database, and therefore\n  // needed to know the type of the field before it could delete it. Future database\n  // adapters should ignore the pointerFieldNames argument. All the field names are in\n  // fieldNames, they show up additionally in the pointerFieldNames database for use\n  // by the mongo adapter, which deals with the legacy mongo format.\n\n  // This function is not obligated to delete fields atomically. It is given the field\n  // names in a list so that databases that are capable of deleting fields atomically\n  // may do so.\n\n  // Returns a Promise.\n  deleteFields(className: string, schema: SchemaType, fieldNames: string[]) {\n    const mongoFormatNames = fieldNames.map(fieldName => {\n      if (schema.fields[fieldName].type === 'Pointer') {\n        return `_p_${fieldName}`;\n      } else {\n        return fieldName;\n      }\n    });\n    const collectionUpdate = { $unset: {} };\n    mongoFormatNames.forEach(name => {\n      collectionUpdate['$unset'][name] = null;\n    });\n\n    const collectionFilter = { $or: [] };\n    mongoFormatNames.forEach(name => {\n      collectionFilter['$or'].push({ [name]: { $exists: true } });\n    });\n\n    const schemaUpdate = { $unset: {} };\n    fieldNames.forEach(name => {\n      schemaUpdate['$unset'][name] = null;\n      schemaUpdate['$unset'][`_metadata.fields_options.${name}`] = null;\n    });\n\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany(collectionFilter, collectionUpdate))\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, schemaUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for all schemas known to this adapter, in Parse format. In case the\n  // schemas cannot be retrieved, returns a promise that rejects. Requirements for the\n  // rejection reason are TBD.\n  getAllClasses(): Promise<StorageClass[]> {\n    return this._schemaCollection()\n      .then(schemasCollection => schemasCollection._fetchAllSchemasFrom_SCHEMA())\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for the schema with the given name, in Parse format. If\n  // this adapter doesn't know about the schema, return a promise that rejects with\n  // undefined as the reason.\n  getClass(className: string): Promise<StorageClass> {\n    return this._schemaCollection()\n      .then(schemasCollection => schemasCollection._fetchOneSchemaFrom_SCHEMA(className))\n      .catch(err => this.handleError(err));\n  }\n\n  // TODO: As yet not particularly well specified. Creates an object. Maybe shouldn't even need the schema,\n  // and should infer from the type. Or maybe does need the schema for validations. Or maybe needs\n  // the schema only for the legacy mongo format. We'll figure that out later.\n  createObject(className: string, schema: SchemaType, object: any, transactionalSession: ?any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = parseObjectToMongoObjectForCreate(className, object, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.insertOne(mongoObject, transactionalSession))\n      .then(() => ({ ops: [mongoObject] }))\n      .catch(error => {\n        if (error.code === 11000) {\n          // Duplicate value\n          const err = new Parse.Error(\n            Parse.Error.DUPLICATE_VALUE,\n            'A duplicate value for a field with unique values was provided'\n          );\n          err.underlyingError = error;\n          if (error.message) {\n            const matches = error.message.match(/index:[\\sa-zA-Z0-9_\\-\\.]+\\$?([a-zA-Z_-]+)_1/);\n            if (matches && Array.isArray(matches)) {\n              err.userInfo = { duplicated_field: matches[1] };\n            }\n          }\n          throw err;\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Remove all objects that match the given Parse Query.\n  // If no objects match, reject with OBJECT_NOT_FOUND. If objects are found and deleted, resolve with undefined.\n  // If there is some other error, reject with INTERNAL_SERVER_ERROR.\n  deleteObjectsByQuery(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    transactionalSession: ?any\n  ) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    return this._adaptiveCollection(className)\n      .then(collection => {\n        const mongoWhere = transformWhere(className, query, schema);\n        return collection.deleteMany(mongoWhere, transactionalSession);\n      })\n      .catch(err => this.handleError(err))\n      .then(\n        ({ deletedCount }) => {\n          if (deletedCount === 0) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n          return Promise.resolve();\n        },\n        () => {\n          throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Database adapter error');\n        }\n      );\n  }\n\n  // Apply the update to all objects that match the given Parse Query.\n  updateObjectsByQuery(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany(mongoWhere, mongoUpdate, transactionalSession))\n      .catch(err => this.handleError(err));\n  }\n\n  // Atomically finds and updates an object based on query.\n  // Return value not currently well specified.\n  findOneAndUpdate(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection =>\n        collection._mongoCollection.findOneAndUpdate(mongoWhere, mongoUpdate, {\n          returnDocument: 'after',\n          session: transactionalSession || undefined,\n        })\n      )\n      .then(result => mongoObjectToParseObject(className, result, schema))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(\n            Parse.Error.DUPLICATE_VALUE,\n            'A duplicate value for a field with unique values was provided'\n          );\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Hopefully we can get rid of this. It's only used for config and hooks.\n  upsertOneObject(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.upsertOne(mongoWhere, mongoUpdate, transactionalSession))\n      .catch(err => this.handleError(err));\n  }\n\n  // Executes a find. Accepts: className, query in Parse format, and { skip, limit, sort }.\n  find(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    {\n      skip,\n      limit,\n      sort,\n      keys,\n      readPreference,\n      hint,\n      caseInsensitive,\n      explain,\n      comment,\n    }: QueryOptions\n  ): Promise<any> {\n    validateExplainValue(explain);\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    const mongoSort = _.mapKeys(sort, (value, fieldName) =>\n      transformKey(className, fieldName, schema)\n    );\n    const mongoKeys = _.reduce(\n      keys,\n      (memo, key) => {\n        if (key === 'ACL') {\n          memo['_rperm'] = 1;\n          memo['_wperm'] = 1;\n        } else {\n          memo[transformKey(className, key, schema)] = 1;\n        }\n        return memo;\n      },\n      {}\n    );\n\n    // If we aren't requesting the `_id` field, we need to explicitly opt out\n    // of it. Doing so in parse-server is unusual, but it can allow us to\n    // optimize some queries with covering indexes.\n    if (keys && !mongoKeys._id) {\n      mongoKeys._id = 0;\n    }\n\n    readPreference = this._parseReadPreference(readPreference);\n    return this.createTextIndexesIfNeeded(className, query, schema)\n      .then(() => this._adaptiveCollection(className))\n      .then(collection =>\n        collection.find(mongoWhere, {\n          skip,\n          limit,\n          sort: mongoSort,\n          keys: mongoKeys,\n          maxTimeMS: this._maxTimeMS,\n          readPreference,\n          hint,\n          caseInsensitive,\n          explain,\n          comment,\n        })\n      )\n      .then(objects => {\n        if (explain) {\n          return objects;\n        }\n        return objects.map(object => mongoObjectToParseObject(className, object, schema));\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  ensureIndex(\n    className: string,\n    schema: SchemaType,\n    fieldNames: string[],\n    indexName: ?string,\n    caseInsensitive: boolean = false,\n    options?: Object = {}\n  ): Promise<any> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const indexCreationRequest = {};\n    const mongoFieldNames = fieldNames.map(fieldName => transformKey(className, fieldName, schema));\n    mongoFieldNames.forEach(fieldName => {\n      indexCreationRequest[fieldName] = options.indexType !== undefined ? options.indexType : 1;\n    });\n\n    const defaultOptions: Object = { background: true, sparse: true };\n    const indexNameOptions: Object = indexName ? { name: indexName } : {};\n    const ttlOptions: Object = options.ttl !== undefined ? { expireAfterSeconds: options.ttl } : {};\n    const sparseOptions: Object = options.sparse !== undefined ? { sparse: options.sparse } : {};\n    const caseInsensitiveOptions: Object = caseInsensitive\n      ? { collation: MongoCollection.caseInsensitiveCollation() }\n      : {};\n    const indexOptions: Object = {\n      ...defaultOptions,\n      ...caseInsensitiveOptions,\n      ...indexNameOptions,\n      ...ttlOptions,\n      ...sparseOptions,\n    };\n\n    return this._adaptiveCollection(className)\n      .then(collection =>\n        collection._mongoCollection.createIndex(indexCreationRequest, indexOptions)\n      )\n      .catch(err => this.handleError(err));\n  }\n\n  // Create a unique index. Unique indexes on nullable fields are not allowed. Since we don't\n  // currently know which fields are nullable and which aren't, we ignore that criteria.\n  // As such, we shouldn't expose this function to users of parse until we have an out-of-band\n  // Way of determining if a field is nullable. Undefined doesn't count against uniqueness,\n  // which is why we use sparse indexes.\n  ensureUniqueness(className: string, schema: SchemaType, fieldNames: string[]) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const indexCreationRequest = {};\n    const mongoFieldNames = fieldNames.map(fieldName => transformKey(className, fieldName, schema));\n    mongoFieldNames.forEach(fieldName => {\n      indexCreationRequest[fieldName] = 1;\n    });\n    return this._adaptiveCollection(className)\n      .then(collection => collection._ensureSparseUniqueIndexInBackground(indexCreationRequest))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(\n            Parse.Error.DUPLICATE_VALUE,\n            'Tried to ensure field uniqueness for a class that already has duplicates.'\n          );\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Used in tests\n  _rawFind(className: string, query: QueryType) {\n    return this._adaptiveCollection(className)\n      .then(collection =>\n        collection.find(query, {\n          maxTimeMS: this._maxTimeMS,\n        })\n      )\n      .catch(err => this.handleError(err));\n  }\n\n  // Executes a count.\n  count(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    readPreference: ?string,\n    _estimate: ?boolean,\n    hint: ?mixed,\n    comment: ?string\n  ) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection =>\n        collection.count(transformWhere(className, query, schema, true), {\n          maxTimeMS: this._maxTimeMS,\n          readPreference,\n          hint,\n          comment,\n        })\n      )\n      .catch(err => this.handleError(err));\n  }\n\n  distinct(className: string, schema: SchemaType, query: QueryType, fieldName: string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const isPointerField = schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';\n    const transformField = transformKey(className, fieldName, schema);\n\n    return this._adaptiveCollection(className)\n      .then(collection =>\n        collection.distinct(transformField, transformWhere(className, query, schema))\n      )\n      .then(objects => {\n        objects = objects.filter(obj => obj != null);\n        return objects.map(object => {\n          if (isPointerField) {\n            return transformPointerString(schema, fieldName, object);\n          }\n          return mongoObjectToParseObject(className, object, schema);\n        });\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  aggregate(\n    className: string,\n    schema: any,\n    pipeline: any,\n    readPreference: ?string,\n    hint: ?mixed,\n    explain?: boolean,\n    comment: ?string\n  ) {\n    validateExplainValue(explain);\n    let isPointerField = false;\n    pipeline = pipeline.map(stage => {\n      if (stage.$group) {\n        stage.$group = this._parseAggregateGroupArgs(schema, stage.$group);\n        if (\n          stage.$group._id &&\n          typeof stage.$group._id === 'string' &&\n          stage.$group._id.indexOf('$_p_') >= 0\n        ) {\n          isPointerField = true;\n        }\n      }\n      if (stage.$match) {\n        stage.$match = this._parseAggregateArgs(schema, stage.$match);\n      }\n      if (stage.$project) {\n        stage.$project = this._parseAggregateProjectArgs(schema, stage.$project);\n      }\n      if (stage.$geoNear && stage.$geoNear.query) {\n        stage.$geoNear.query = this._parseAggregateArgs(schema, stage.$geoNear.query);\n      }\n      return stage;\n    });\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection =>\n        collection.aggregate(pipeline, {\n          readPreference,\n          maxTimeMS: this._maxTimeMS,\n          hint,\n          explain,\n          comment,\n        })\n      )\n      .then(results => {\n        results.forEach(result => {\n          if (Object.prototype.hasOwnProperty.call(result, '_id')) {\n            if (isPointerField && result._id) {\n              result._id = result._id.split('$')[1];\n            }\n            if (\n              result._id == null ||\n              result._id == undefined ||\n              (['object', 'string'].includes(typeof result._id) && _.isEmpty(result._id))\n            ) {\n              result._id = null;\n            }\n            result.objectId = result._id;\n            delete result._id;\n          }\n        });\n        return results;\n      })\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // This function will recursively traverse the pipeline and convert any Pointer or Date columns.\n  // If we detect a pointer column we will rename the column being queried for to match the column\n  // in the database. We also modify the value to what we expect the value to be in the database\n  // as well.\n  // For dates, the driver expects a Date object, but we have a string coming in. So we'll convert\n  // the string to a Date so the driver can perform the necessary comparison.\n  //\n  // The goal of this method is to look for the \"leaves\" of the pipeline and determine if it needs\n  // to be converted. The pipeline can have a few different forms. For more details, see:\n  //     https://docs.mongodb.com/manual/reference/operator/aggregation/\n  //\n  // If the pipeline is an array, it means we are probably parsing an '$and' or '$or' operator. In\n  // that case we need to loop through all of it's children to find the columns being operated on.\n  // If the pipeline is an object, then we'll loop through the keys checking to see if the key name\n  // matches one of the schema columns. If it does match a column and the column is a Pointer or\n  // a Date, then we'll convert the value as described above.\n  //\n  // As much as I hate recursion...this seemed like a good fit for it. We're essentially traversing\n  // down a tree to find a \"leaf node\" and checking to see if it needs to be converted.\n  _parseAggregateArgs(schema: any, pipeline: any): any {\n    if (pipeline === null) {\n      return null;\n    } else if (Array.isArray(pipeline)) {\n      return pipeline.map(value => this._parseAggregateArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n          if (typeof pipeline[field] === 'object') {\n            // Pass objects down to MongoDB...this is more than likely an $exists operator.\n            returnValue[`_p_${field}`] = pipeline[field];\n          } else {\n            returnValue[`_p_${field}`] = `${schema.fields[field].targetClass}$${pipeline[field]}`;\n          }\n        } else if (schema.fields[field] && schema.fields[field].type === 'Date') {\n          returnValue[field] = this._convertToDate(pipeline[field]);\n        } else {\n          returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n        }\n\n        if (field === 'objectId') {\n          returnValue['_id'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'createdAt') {\n          returnValue['_created_at'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'updatedAt') {\n          returnValue['_updated_at'] = returnValue[field];\n          delete returnValue[field];\n        }\n      }\n      return returnValue;\n    }\n    return pipeline;\n  }\n\n  // This function is slightly different than the one above. Rather than trying to combine these\n  // two functions and making the code even harder to understand, I decided to split it up. The\n  // difference with this function is we are not transforming the values, only the keys of the\n  // pipeline.\n  _parseAggregateProjectArgs(schema: any, pipeline: any): any {\n    const returnValue = {};\n    for (const field in pipeline) {\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        returnValue[`_p_${field}`] = pipeline[field];\n      } else {\n        returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n      }\n\n      if (field === 'objectId') {\n        returnValue['_id'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'createdAt') {\n        returnValue['_created_at'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'updatedAt') {\n        returnValue['_updated_at'] = returnValue[field];\n        delete returnValue[field];\n      }\n    }\n    return returnValue;\n  }\n\n  // This function is slightly different than the two above. MongoDB $group aggregate looks like:\n  //     { $group: { _id: <expression>, <field1>: { <accumulator1> : <expression1> }, ... } }\n  // The <expression> could be a column name, prefixed with the '$' character. We'll look for\n  // these <expression> and check to see if it is a 'Pointer' or if it's one of createdAt,\n  // updatedAt or objectId and change it accordingly.\n  _parseAggregateGroupArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map(value => this._parseAggregateGroupArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        returnValue[field] = this._parseAggregateGroupArgs(schema, pipeline[field]);\n      }\n      return returnValue;\n    } else if (typeof pipeline === 'string') {\n      const field = pipeline.substring(1);\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        return `$_p_${field}`;\n      } else if (field == 'createdAt') {\n        return '$_created_at';\n      } else if (field == 'updatedAt') {\n        return '$_updated_at';\n      }\n    }\n    return pipeline;\n  }\n\n  /**\n   * Recursively converts values to Date objects. Since the passed object is part of an aggregation\n   * pipeline and can contain various logic operators (like $gt, $lt, etc), this function will\n   * traverse the object and convert any strings that can be parsed as dates into Date objects.\n   * @param {any} value The value to convert.\n   * @returns {any} The original value if not convertible to Date, or a Date object if it is.\n   */\n  _convertToDate(value: any): any {\n    if (value instanceof Date) {\n      return value;\n    }\n    if (typeof value === 'string') {\n      return isNaN(Date.parse(value)) ? value : new Date(value);\n    }\n    if (typeof value === 'object') {\n      const returnValue = {};\n      for (const field in value) {\n        returnValue[field] = this._convertToDate(value[field]);\n      }\n      return returnValue;\n    }\n    return value;\n  }\n\n  _parseReadPreference(readPreference: ?string): ?string {\n    if (readPreference) {\n      readPreference = readPreference.toUpperCase();\n    }\n    switch (readPreference) {\n      case 'PRIMARY':\n        readPreference = ReadPreference.PRIMARY;\n        break;\n      case 'PRIMARY_PREFERRED':\n        readPreference = ReadPreference.PRIMARY_PREFERRED;\n        break;\n      case 'SECONDARY':\n        readPreference = ReadPreference.SECONDARY;\n        break;\n      case 'SECONDARY_PREFERRED':\n        readPreference = ReadPreference.SECONDARY_PREFERRED;\n        break;\n      case 'NEAREST':\n        readPreference = ReadPreference.NEAREST;\n        break;\n      case undefined:\n      case null:\n      case '':\n        break;\n      default:\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Not supported read preference.');\n    }\n    return readPreference;\n  }\n\n  performInitialization(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  createIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndex(index))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexes(className: string, indexes: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndexes(indexes))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexesIfNeeded(className: string, fieldName: string, type: any) {\n    if (type && type.type === 'Polygon') {\n      const index = {\n        [fieldName]: '2dsphere',\n      };\n      return this.createIndex(className, index);\n    }\n    return Promise.resolve();\n  }\n\n  createTextIndexesIfNeeded(className: string, query: QueryType, schema: any): Promise<void> {\n    for (const fieldName in query) {\n      if (!query[fieldName] || !query[fieldName].$text) {\n        continue;\n      }\n      const existingIndexes = schema.indexes;\n      for (const key in existingIndexes) {\n        const index = existingIndexes[key];\n        if (Object.prototype.hasOwnProperty.call(index, fieldName)) {\n          return Promise.resolve();\n        }\n      }\n      const indexName = `${fieldName}_text`;\n      const textIndex = {\n        [indexName]: { [fieldName]: 'text' },\n      };\n      return this.setIndexesWithSchemaFormat(\n        className,\n        textIndex,\n        existingIndexes,\n        schema.fields\n      ).catch(error => {\n        if (error.code === 85) {\n          // Index exist with different options\n          return this.setIndexesFromMongo(className);\n        }\n        throw error;\n      });\n    }\n    return Promise.resolve();\n  }\n\n  getIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.indexes())\n      .catch(err => this.handleError(err));\n  }\n\n  dropIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndex(index))\n      .catch(err => this.handleError(err));\n  }\n\n  dropAllIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndexes())\n      .catch(err => this.handleError(err));\n  }\n\n  updateSchemaWithIndexes(): Promise<any> {\n    return this.getAllClasses()\n      .then(classes => {\n        const promises = classes.map(schema => {\n          return this.setIndexesFromMongo(schema.className);\n        });\n        return Promise.all(promises);\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  createTransactionalSession(): Promise<any> {\n    const transactionalSection = this.client.startSession();\n    transactionalSection.startTransaction();\n    return Promise.resolve(transactionalSection);\n  }\n\n  commitTransactionalSession(transactionalSection: any): Promise<void> {\n    const commit = retries => {\n      return transactionalSection\n        .commitTransaction()\n        .catch(error => {\n          if (error && error.hasErrorLabel('TransientTransactionError') && retries > 0) {\n            return commit(retries - 1);\n          }\n          throw error;\n        })\n        .then(() => {\n          transactionalSection.endSession();\n        });\n    };\n    return commit(5);\n  }\n\n  abortTransactionalSession(transactionalSection: any): Promise<void> {\n    return transactionalSection.abortTransaction().then(() => {\n      transactionalSection.endSession();\n    });\n  }\n}\n\nexport default MongoStorageAdapter;\n"],"mappings":"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","ignoreList":[]}
|