parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21
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/README.md +45 -17
- package/lib/AccountLockout.js +11 -26
- package/lib/Adapters/AdapterLoader.js +8 -14
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
- package/lib/Adapters/Auth/AuthAdapter.js +7 -16
- package/lib/Adapters/Auth/OAuth1Client.js +32 -57
- package/lib/Adapters/Auth/apple.js +6 -22
- package/lib/Adapters/Auth/facebook.js +7 -37
- package/lib/Adapters/Auth/gcenter.js +8 -37
- package/lib/Adapters/Auth/github.js +7 -10
- package/lib/Adapters/Auth/google.js +11 -34
- package/lib/Adapters/Auth/gpgames.js +5 -8
- package/lib/Adapters/Auth/httpsRequest.js +1 -7
- package/lib/Adapters/Auth/index.js +20 -65
- package/lib/Adapters/Auth/instagram.js +5 -9
- package/lib/Adapters/Auth/janraincapture.js +8 -12
- package/lib/Adapters/Auth/janrainengage.js +7 -11
- package/lib/Adapters/Auth/keycloak.js +5 -19
- package/lib/Adapters/Auth/ldap.js +1 -15
- package/lib/Adapters/Auth/line.js +7 -10
- package/lib/Adapters/Auth/linkedin.js +7 -12
- package/lib/Adapters/Auth/meetup.js +7 -10
- package/lib/Adapters/Auth/microsoft.js +7 -10
- package/lib/Adapters/Auth/oauth2.js +6 -18
- package/lib/Adapters/Auth/phantauth.js +8 -10
- package/lib/Adapters/Auth/qq.js +7 -13
- package/lib/Adapters/Auth/spotify.js +7 -14
- package/lib/Adapters/Auth/twitter.js +5 -15
- package/lib/Adapters/Auth/vkontakte.js +9 -15
- package/lib/Adapters/Auth/wechat.js +7 -10
- package/lib/Adapters/Auth/weibo.js +7 -11
- package/lib/Adapters/Cache/CacheAdapter.js +4 -12
- package/lib/Adapters/Cache/InMemoryCache.js +5 -19
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
- package/lib/Adapters/Cache/LRUCache.js +1 -11
- package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
- package/lib/Adapters/Cache/SchemaCache.js +1 -6
- package/lib/Adapters/Email/MailAdapter.js +2 -7
- package/lib/Adapters/Files/FilesAdapter.js +7 -21
- package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
- package/lib/Adapters/Logger/WinstonLogger.js +3 -30
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
- package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
- package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
- package/lib/Adapters/Push/PushAdapter.js +2 -8
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
- package/lib/Auth.js +54 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +69 -113
- package/lib/Controllers/AdaptableController.js +6 -18
- package/lib/Controllers/AnalyticsController.js +1 -9
- package/lib/Controllers/CacheController.js +3 -23
- package/lib/Controllers/DatabaseController.js +147 -345
- package/lib/Controllers/FilesController.js +5 -34
- package/lib/Controllers/HooksController.js +1 -51
- package/lib/Controllers/LiveQueryController.js +4 -23
- package/lib/Controllers/LoggerController.js +15 -54
- package/lib/Controllers/ParseGraphQLController.js +49 -104
- package/lib/Controllers/PushController.js +20 -59
- package/lib/Controllers/SchemaController.js +154 -344
- package/lib/Controllers/UserController.js +11 -72
- package/lib/Controllers/index.js +19 -68
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +1 -8
- package/lib/Deprecator/Deprecator.js +9 -18
- package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
- package/lib/GraphQL/ParseGraphQLServer.js +2 -29
- package/lib/GraphQL/helpers/objectsMutations.js +2 -12
- package/lib/GraphQL/helpers/objectsQueries.js +18 -76
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
- package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
- package/lib/GraphQL/loaders/filesMutations.js +2 -19
- package/lib/GraphQL/loaders/functionsMutations.js +6 -17
- package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
- package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
- package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
- package/lib/GraphQL/loaders/schemaMutations.js +1 -20
- package/lib/GraphQL/loaders/schemaQueries.js +1 -14
- package/lib/GraphQL/loaders/schemaTypes.js +2 -6
- package/lib/GraphQL/loaders/usersMutations.js +6 -28
- package/lib/GraphQL/loaders/usersQueries.js +4 -26
- package/lib/GraphQL/parseGraphQLUtils.js +6 -19
- package/lib/GraphQL/transformers/className.js +1 -4
- package/lib/GraphQL/transformers/constraintType.js +1 -20
- package/lib/GraphQL/transformers/inputType.js +1 -20
- package/lib/GraphQL/transformers/mutation.js +6 -51
- package/lib/GraphQL/transformers/outputType.js +1 -20
- package/lib/GraphQL/transformers/query.js +6 -42
- package/lib/GraphQL/transformers/schemaFields.js +7 -34
- package/lib/KeyPromiseQueue.js +1 -12
- package/lib/LiveQuery/Client.js +1 -25
- package/lib/LiveQuery/Id.js +1 -7
- package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
- package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
- package/lib/LiveQuery/ParsePubSub.js +1 -12
- package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
- package/lib/LiveQuery/QueryTools.js +14 -116
- package/lib/LiveQuery/RequestSchema.js +1 -1
- package/lib/LiveQuery/SessionTokenCache.js +1 -17
- package/lib/LiveQuery/Subscription.js +4 -18
- package/lib/LiveQuery/equalObjects.js +2 -14
- package/lib/Options/Definitions.js +79 -10
- package/lib/Options/docs.js +23 -3
- package/lib/Options/index.js +4 -12
- package/lib/Options/parsers.js +1 -18
- package/lib/Page.js +1 -9
- package/lib/ParseMessageQueue.js +1 -10
- package/lib/ParseServer.js +144 -182
- package/lib/ParseServerRESTController.js +6 -33
- package/lib/PromiseRouter.js +16 -50
- package/lib/Push/PushQueue.js +3 -15
- package/lib/Push/PushWorker.js +7 -32
- package/lib/Push/utils.js +9 -38
- package/lib/RestQuery.js +105 -242
- package/lib/RestWrite.js +212 -377
- package/lib/Routers/AggregateRouter.js +14 -51
- package/lib/Routers/AnalyticsRouter.js +2 -8
- package/lib/Routers/AudiencesRouter.js +1 -15
- package/lib/Routers/ClassesRouter.js +3 -53
- package/lib/Routers/CloudCodeRouter.js +1 -19
- package/lib/Routers/FeaturesRouter.js +1 -10
- package/lib/Routers/FilesRouter.js +29 -76
- package/lib/Routers/FunctionsRouter.js +5 -28
- package/lib/Routers/GlobalConfigRouter.js +4 -18
- package/lib/Routers/GraphQLRouter.js +1 -14
- package/lib/Routers/HooksRouter.js +1 -29
- package/lib/Routers/IAPValidationRouter.js +6 -29
- package/lib/Routers/InstallationsRouter.js +2 -12
- package/lib/Routers/LogsRouter.js +4 -16
- package/lib/Routers/PagesRouter.js +69 -129
- package/lib/Routers/PublicAPIRouter.js +3 -62
- package/lib/Routers/PurgeRouter.js +1 -15
- package/lib/Routers/PushRouter.js +2 -18
- package/lib/Routers/RolesRouter.js +1 -7
- package/lib/Routers/SchemasRouter.js +4 -34
- package/lib/Routers/SecurityRouter.js +1 -12
- package/lib/Routers/SessionsRouter.js +3 -19
- package/lib/Routers/UsersRouter.js +58 -155
- package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
- package/lib/SchemaMigrations/Migrations.js +2 -8
- package/lib/Security/Check.js +8 -16
- package/lib/Security/CheckGroup.js +4 -11
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
- package/lib/Security/CheckGroups/CheckGroups.js +1 -4
- package/lib/Security/CheckRunner.js +22 -41
- package/lib/StatusHandler.js +12 -69
- package/lib/TestUtils.js +1 -6
- package/lib/Utils.js +27 -66
- package/lib/batch.js +17 -28
- package/lib/cache.js +1 -3
- package/lib/cli/definitions/parse-live-query-server.js +1 -3
- package/lib/cli/definitions/parse-server.js +1 -3
- package/lib/cli/parse-live-query-server.js +1 -6
- package/lib/cli/parse-server.js +11 -21
- package/lib/cli/utils/commander.js +13 -51
- package/lib/cli/utils/runner.js +1 -14
- package/lib/cloud-code/Parse.Cloud.js +71 -92
- package/lib/cryptoUtils.js +11 -19
- package/lib/defaults.js +2 -14
- package/lib/deprecated.js +1 -2
- package/lib/index.js +16 -34
- package/lib/logger.js +6 -13
- package/lib/middlewares.js +147 -151
- package/lib/password.js +6 -10
- package/lib/request.js +173 -2
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +19 -41
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +125 -305
- package/package.json +22 -19
- package/lib/cloud-code/HTTPResponse.js +0 -73
- package/lib/cloud-code/httpRequest.js +0 -192
|
@@ -4,87 +4,62 @@ Object.defineProperty(exports, "__esModule", {
|
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
6
|
exports.default = exports.PostgresStorageAdapter = void 0;
|
|
7
|
-
|
|
8
7
|
var _PostgresClient = require("./PostgresClient");
|
|
9
|
-
|
|
10
8
|
var _node = _interopRequireDefault(require("parse/node"));
|
|
11
|
-
|
|
12
9
|
var _lodash = _interopRequireDefault(require("lodash"));
|
|
13
|
-
|
|
14
10
|
var _uuid = require("uuid");
|
|
15
|
-
|
|
16
11
|
var _sql = _interopRequireDefault(require("./sql"));
|
|
17
|
-
|
|
18
12
|
var _StorageAdapter = require("../StorageAdapter");
|
|
19
|
-
|
|
20
13
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
21
|
-
|
|
22
|
-
function
|
|
23
|
-
|
|
24
|
-
function
|
|
25
|
-
|
|
26
|
-
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
27
|
-
|
|
14
|
+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
|
|
15
|
+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
|
|
16
|
+
function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
17
|
+
function _toPropertyKey(arg) { var key = _toPrimitive(arg, "string"); return typeof key === "symbol" ? key : String(key); }
|
|
18
|
+
function _toPrimitive(input, hint) { if (typeof input !== "object" || input === null) return input; var prim = input[Symbol.toPrimitive]; if (prim !== undefined) { var res = prim.call(input, hint || "default"); if (typeof res !== "object") return res; throw new TypeError("@@toPrimitive must return a primitive value."); } return (hint === "string" ? String : Number)(input); }
|
|
28
19
|
const Utils = require('../../../Utils');
|
|
29
|
-
|
|
30
20
|
const PostgresRelationDoesNotExistError = '42P01';
|
|
31
21
|
const PostgresDuplicateRelationError = '42P07';
|
|
32
22
|
const PostgresDuplicateColumnError = '42701';
|
|
33
23
|
const PostgresMissingColumnError = '42703';
|
|
34
24
|
const PostgresUniqueIndexViolationError = '23505';
|
|
35
|
-
|
|
36
25
|
const logger = require('../../../logger');
|
|
37
|
-
|
|
38
26
|
const debug = function (...args) {
|
|
39
27
|
args = ['PG: ' + arguments[0]].concat(args.slice(1, args.length));
|
|
40
28
|
const log = logger.getLogger();
|
|
41
29
|
log.debug.apply(log, args);
|
|
42
30
|
};
|
|
43
|
-
|
|
44
31
|
const parseTypeToPostgresType = type => {
|
|
45
32
|
switch (type.type) {
|
|
46
33
|
case 'String':
|
|
47
34
|
return 'text';
|
|
48
|
-
|
|
49
35
|
case 'Date':
|
|
50
36
|
return 'timestamp with time zone';
|
|
51
|
-
|
|
52
37
|
case 'Object':
|
|
53
38
|
return 'jsonb';
|
|
54
|
-
|
|
55
39
|
case 'File':
|
|
56
40
|
return 'text';
|
|
57
|
-
|
|
58
41
|
case 'Boolean':
|
|
59
42
|
return 'boolean';
|
|
60
|
-
|
|
61
43
|
case 'Pointer':
|
|
62
44
|
return 'text';
|
|
63
|
-
|
|
64
45
|
case 'Number':
|
|
65
46
|
return 'double precision';
|
|
66
|
-
|
|
67
47
|
case 'GeoPoint':
|
|
68
48
|
return 'point';
|
|
69
|
-
|
|
70
49
|
case 'Bytes':
|
|
71
50
|
return 'jsonb';
|
|
72
|
-
|
|
73
51
|
case 'Polygon':
|
|
74
52
|
return 'polygon';
|
|
75
|
-
|
|
76
53
|
case 'Array':
|
|
77
54
|
if (type.contents && type.contents.type === 'String') {
|
|
78
55
|
return 'text[]';
|
|
79
56
|
} else {
|
|
80
57
|
return 'jsonb';
|
|
81
58
|
}
|
|
82
|
-
|
|
83
59
|
default:
|
|
84
60
|
throw `no type for ${JSON.stringify(type)} yet`;
|
|
85
61
|
}
|
|
86
62
|
};
|
|
87
|
-
|
|
88
63
|
const ParseToPosgresComparator = {
|
|
89
64
|
$gt: '>',
|
|
90
65
|
$lt: '<',
|
|
@@ -105,50 +80,40 @@ const mongoAggregateToPostgres = {
|
|
|
105
80
|
$week: 'WEEK',
|
|
106
81
|
$year: 'YEAR'
|
|
107
82
|
};
|
|
108
|
-
|
|
109
83
|
const toPostgresValue = value => {
|
|
110
84
|
if (typeof value === 'object') {
|
|
111
85
|
if (value.__type === 'Date') {
|
|
112
86
|
return value.iso;
|
|
113
87
|
}
|
|
114
|
-
|
|
115
88
|
if (value.__type === 'File') {
|
|
116
89
|
return value.name;
|
|
117
90
|
}
|
|
118
91
|
}
|
|
119
|
-
|
|
120
92
|
return value;
|
|
121
93
|
};
|
|
122
|
-
|
|
123
94
|
const toPostgresValueCastType = value => {
|
|
124
95
|
const postgresValue = toPostgresValue(value);
|
|
125
96
|
let castType;
|
|
126
|
-
|
|
127
97
|
switch (typeof postgresValue) {
|
|
128
98
|
case 'number':
|
|
129
99
|
castType = 'double precision';
|
|
130
100
|
break;
|
|
131
|
-
|
|
132
101
|
case 'boolean':
|
|
133
102
|
castType = 'boolean';
|
|
134
103
|
break;
|
|
135
|
-
|
|
136
104
|
default:
|
|
137
105
|
castType = undefined;
|
|
138
106
|
}
|
|
139
|
-
|
|
140
107
|
return castType;
|
|
141
108
|
};
|
|
142
|
-
|
|
143
109
|
const transformValue = value => {
|
|
144
110
|
if (typeof value === 'object' && value.__type === 'Pointer') {
|
|
145
111
|
return value.objectId;
|
|
146
112
|
}
|
|
147
|
-
|
|
148
113
|
return value;
|
|
149
|
-
};
|
|
150
|
-
|
|
114
|
+
};
|
|
151
115
|
|
|
116
|
+
// Duplicate from then mongo adapter...
|
|
152
117
|
const emptyCLPS = Object.freeze({
|
|
153
118
|
find: {},
|
|
154
119
|
get: {},
|
|
@@ -185,29 +150,22 @@ const defaultCLPS = Object.freeze({
|
|
|
185
150
|
'*': []
|
|
186
151
|
}
|
|
187
152
|
});
|
|
188
|
-
|
|
189
153
|
const toParseSchema = schema => {
|
|
190
154
|
if (schema.className === '_User') {
|
|
191
155
|
delete schema.fields._hashed_password;
|
|
192
156
|
}
|
|
193
|
-
|
|
194
157
|
if (schema.fields) {
|
|
195
158
|
delete schema.fields._wperm;
|
|
196
159
|
delete schema.fields._rperm;
|
|
197
160
|
}
|
|
198
|
-
|
|
199
161
|
let clps = defaultCLPS;
|
|
200
|
-
|
|
201
162
|
if (schema.classLevelPermissions) {
|
|
202
163
|
clps = _objectSpread(_objectSpread({}, emptyCLPS), schema.classLevelPermissions);
|
|
203
164
|
}
|
|
204
|
-
|
|
205
165
|
let indexes = {};
|
|
206
|
-
|
|
207
166
|
if (schema.indexes) {
|
|
208
167
|
indexes = _objectSpread({}, schema.indexes);
|
|
209
168
|
}
|
|
210
|
-
|
|
211
169
|
return {
|
|
212
170
|
className: schema.className,
|
|
213
171
|
fields: schema.fields,
|
|
@@ -215,12 +173,10 @@ const toParseSchema = schema => {
|
|
|
215
173
|
indexes
|
|
216
174
|
};
|
|
217
175
|
};
|
|
218
|
-
|
|
219
176
|
const toPostgresSchema = schema => {
|
|
220
177
|
if (!schema) {
|
|
221
178
|
return schema;
|
|
222
179
|
}
|
|
223
|
-
|
|
224
180
|
schema.fields = schema.fields || {};
|
|
225
181
|
schema.fields._wperm = {
|
|
226
182
|
type: 'Array',
|
|
@@ -234,7 +190,6 @@ const toPostgresSchema = schema => {
|
|
|
234
190
|
type: 'String'
|
|
235
191
|
}
|
|
236
192
|
};
|
|
237
|
-
|
|
238
193
|
if (schema.className === '_User') {
|
|
239
194
|
schema.fields._hashed_password = {
|
|
240
195
|
type: 'String'
|
|
@@ -243,10 +198,8 @@ const toPostgresSchema = schema => {
|
|
|
243
198
|
type: 'Array'
|
|
244
199
|
};
|
|
245
200
|
}
|
|
246
|
-
|
|
247
201
|
return schema;
|
|
248
202
|
};
|
|
249
|
-
|
|
250
203
|
const handleDotFields = object => {
|
|
251
204
|
Object.keys(object).forEach(fieldName => {
|
|
252
205
|
if (fieldName.indexOf('.') > -1) {
|
|
@@ -256,85 +209,68 @@ const handleDotFields = object => {
|
|
|
256
209
|
let currentObj = object[first];
|
|
257
210
|
let next;
|
|
258
211
|
let value = object[fieldName];
|
|
259
|
-
|
|
260
212
|
if (value && value.__op === 'Delete') {
|
|
261
213
|
value = undefined;
|
|
262
214
|
}
|
|
263
215
|
/* eslint-disable no-cond-assign */
|
|
264
|
-
|
|
265
|
-
|
|
266
216
|
while (next = components.shift()) {
|
|
267
217
|
/* eslint-enable no-cond-assign */
|
|
268
218
|
currentObj[next] = currentObj[next] || {};
|
|
269
|
-
|
|
270
219
|
if (components.length === 0) {
|
|
271
220
|
currentObj[next] = value;
|
|
272
221
|
}
|
|
273
|
-
|
|
274
222
|
currentObj = currentObj[next];
|
|
275
223
|
}
|
|
276
|
-
|
|
277
224
|
delete object[fieldName];
|
|
278
225
|
}
|
|
279
226
|
});
|
|
280
227
|
return object;
|
|
281
228
|
};
|
|
282
|
-
|
|
283
229
|
const transformDotFieldToComponents = fieldName => {
|
|
284
230
|
return fieldName.split('.').map((cmpt, index) => {
|
|
285
231
|
if (index === 0) {
|
|
286
232
|
return `"${cmpt}"`;
|
|
287
233
|
}
|
|
288
|
-
|
|
289
234
|
return `'${cmpt}'`;
|
|
290
235
|
});
|
|
291
236
|
};
|
|
292
|
-
|
|
293
237
|
const transformDotField = fieldName => {
|
|
294
238
|
if (fieldName.indexOf('.') === -1) {
|
|
295
239
|
return `"${fieldName}"`;
|
|
296
240
|
}
|
|
297
|
-
|
|
298
241
|
const components = transformDotFieldToComponents(fieldName);
|
|
299
242
|
let name = components.slice(0, components.length - 1).join('->');
|
|
300
243
|
name += '->>' + components[components.length - 1];
|
|
301
244
|
return name;
|
|
302
245
|
};
|
|
303
|
-
|
|
304
246
|
const transformAggregateField = fieldName => {
|
|
305
247
|
if (typeof fieldName !== 'string') {
|
|
306
248
|
return fieldName;
|
|
307
249
|
}
|
|
308
|
-
|
|
309
250
|
if (fieldName === '$_created_at') {
|
|
310
251
|
return 'createdAt';
|
|
311
252
|
}
|
|
312
|
-
|
|
313
253
|
if (fieldName === '$_updated_at') {
|
|
314
254
|
return 'updatedAt';
|
|
315
255
|
}
|
|
316
|
-
|
|
317
256
|
return fieldName.substr(1);
|
|
318
257
|
};
|
|
319
|
-
|
|
320
258
|
const validateKeys = object => {
|
|
321
259
|
if (typeof object == 'object') {
|
|
322
260
|
for (const key in object) {
|
|
323
261
|
if (typeof object[key] == 'object') {
|
|
324
262
|
validateKeys(object[key]);
|
|
325
263
|
}
|
|
326
|
-
|
|
327
264
|
if (key.includes('$') || key.includes('.')) {
|
|
328
265
|
throw new _node.default.Error(_node.default.Error.INVALID_NESTED_KEY, "Nested keys should not contain the '$' or '.' characters");
|
|
329
266
|
}
|
|
330
267
|
}
|
|
331
268
|
}
|
|
332
|
-
};
|
|
333
|
-
|
|
269
|
+
};
|
|
334
270
|
|
|
271
|
+
// Returns the list of join tables on a schema
|
|
335
272
|
const joinTablesForSchema = schema => {
|
|
336
273
|
const list = [];
|
|
337
|
-
|
|
338
274
|
if (schema) {
|
|
339
275
|
Object.keys(schema.fields).forEach(field => {
|
|
340
276
|
if (schema.fields[field].type === 'Relation') {
|
|
@@ -342,10 +278,8 @@ const joinTablesForSchema = schema => {
|
|
|
342
278
|
}
|
|
343
279
|
});
|
|
344
280
|
}
|
|
345
|
-
|
|
346
281
|
return list;
|
|
347
282
|
};
|
|
348
|
-
|
|
349
283
|
const buildWhereClause = ({
|
|
350
284
|
schema,
|
|
351
285
|
query,
|
|
@@ -356,21 +290,19 @@ const buildWhereClause = ({
|
|
|
356
290
|
let values = [];
|
|
357
291
|
const sorts = [];
|
|
358
292
|
schema = toPostgresSchema(schema);
|
|
359
|
-
|
|
360
293
|
for (const fieldName in query) {
|
|
361
294
|
const isArrayField = schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array';
|
|
362
295
|
const initialPatternsLength = patterns.length;
|
|
363
|
-
const fieldValue = query[fieldName];
|
|
296
|
+
const fieldValue = query[fieldName];
|
|
364
297
|
|
|
298
|
+
// nothing in the schema, it's gonna blow up
|
|
365
299
|
if (!schema.fields[fieldName]) {
|
|
366
300
|
// as it won't exist
|
|
367
301
|
if (fieldValue && fieldValue.$exists === false) {
|
|
368
302
|
continue;
|
|
369
303
|
}
|
|
370
304
|
}
|
|
371
|
-
|
|
372
305
|
const authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);
|
|
373
|
-
|
|
374
306
|
if (authDataMatch) {
|
|
375
307
|
// TODO: Handle querying by _auth_data_provider, authData is stored in authData field
|
|
376
308
|
continue;
|
|
@@ -380,7 +312,6 @@ const buildWhereClause = ({
|
|
|
380
312
|
index += 2;
|
|
381
313
|
} else if (fieldName.indexOf('.') >= 0) {
|
|
382
314
|
let name = transformDotField(fieldName);
|
|
383
|
-
|
|
384
315
|
if (fieldValue === null) {
|
|
385
316
|
patterns.push(`$${index}:raw IS NULL`);
|
|
386
317
|
values.push(name);
|
|
@@ -392,7 +323,8 @@ const buildWhereClause = ({
|
|
|
392
323
|
patterns.push(`($${index}:raw)::jsonb @> $${index + 1}::jsonb`);
|
|
393
324
|
values.push(name, JSON.stringify(fieldValue.$in));
|
|
394
325
|
index += 2;
|
|
395
|
-
} else if (fieldValue.$regex) {
|
|
326
|
+
} else if (fieldValue.$regex) {
|
|
327
|
+
// Handle later
|
|
396
328
|
} else if (typeof fieldValue !== 'object') {
|
|
397
329
|
patterns.push(`$${index}:raw = $${index + 1}::text`);
|
|
398
330
|
values.push(name, fieldValue);
|
|
@@ -409,8 +341,8 @@ const buildWhereClause = ({
|
|
|
409
341
|
values.push(fieldName, fieldValue);
|
|
410
342
|
index += 2;
|
|
411
343
|
} else if (typeof fieldValue === 'boolean') {
|
|
412
|
-
patterns.push(`$${index}:name = $${index + 1}`);
|
|
413
|
-
|
|
344
|
+
patterns.push(`$${index}:name = $${index + 1}`);
|
|
345
|
+
// Can't cast boolean to double precision
|
|
414
346
|
if (schema.fields[fieldName] && schema.fields[fieldName].type === 'Number') {
|
|
415
347
|
// Should always return zero results
|
|
416
348
|
const MAX_INT_PLUS_ONE = 9223372036854775808;
|
|
@@ -418,7 +350,6 @@ const buildWhereClause = ({
|
|
|
418
350
|
} else {
|
|
419
351
|
values.push(fieldName, fieldValue);
|
|
420
352
|
}
|
|
421
|
-
|
|
422
353
|
index += 2;
|
|
423
354
|
} else if (typeof fieldValue === 'number') {
|
|
424
355
|
patterns.push(`$${index}:name = $${index + 1}`);
|
|
@@ -434,7 +365,6 @@ const buildWhereClause = ({
|
|
|
434
365
|
index,
|
|
435
366
|
caseInsensitive
|
|
436
367
|
});
|
|
437
|
-
|
|
438
368
|
if (clause.pattern.length > 0) {
|
|
439
369
|
clauses.push(clause.pattern);
|
|
440
370
|
clauseValues.push(...clause.values);
|
|
@@ -446,7 +376,6 @@ const buildWhereClause = ({
|
|
|
446
376
|
patterns.push(`${not}(${clauses.join(orOrAnd)})`);
|
|
447
377
|
values.push(...clauseValues);
|
|
448
378
|
}
|
|
449
|
-
|
|
450
379
|
if (fieldValue.$ne !== undefined) {
|
|
451
380
|
if (isArrayField) {
|
|
452
381
|
fieldValue.$ne = JSON.stringify([fieldValue.$ne]);
|
|
@@ -474,7 +403,6 @@ const buildWhereClause = ({
|
|
|
474
403
|
}
|
|
475
404
|
}
|
|
476
405
|
}
|
|
477
|
-
|
|
478
406
|
if (fieldValue.$ne.__type === 'GeoPoint') {
|
|
479
407
|
const point = fieldValue.$ne;
|
|
480
408
|
values.push(fieldName, point.longitude, point.latitude);
|
|
@@ -485,7 +413,6 @@ const buildWhereClause = ({
|
|
|
485
413
|
index += 2;
|
|
486
414
|
}
|
|
487
415
|
}
|
|
488
|
-
|
|
489
416
|
if (fieldValue.$eq !== undefined) {
|
|
490
417
|
if (fieldValue.$eq === null) {
|
|
491
418
|
patterns.push(`$${index}:name IS NULL`);
|
|
@@ -506,9 +433,7 @@ const buildWhereClause = ({
|
|
|
506
433
|
}
|
|
507
434
|
}
|
|
508
435
|
}
|
|
509
|
-
|
|
510
436
|
const isInOrNin = Array.isArray(fieldValue.$in) || Array.isArray(fieldValue.$nin);
|
|
511
|
-
|
|
512
437
|
if (Array.isArray(fieldValue.$in) && isArrayField && schema.fields[fieldName].contents && schema.fields[fieldName].contents.type === 'String') {
|
|
513
438
|
const inPatterns = [];
|
|
514
439
|
let allowNull = false;
|
|
@@ -521,18 +446,15 @@ const buildWhereClause = ({
|
|
|
521
446
|
inPatterns.push(`$${index + 1 + listIndex - (allowNull ? 1 : 0)}`);
|
|
522
447
|
}
|
|
523
448
|
});
|
|
524
|
-
|
|
525
449
|
if (allowNull) {
|
|
526
450
|
patterns.push(`($${index}:name IS NULL OR $${index}:name && ARRAY[${inPatterns.join()}])`);
|
|
527
451
|
} else {
|
|
528
452
|
patterns.push(`$${index}:name && ARRAY[${inPatterns.join()}]`);
|
|
529
453
|
}
|
|
530
|
-
|
|
531
454
|
index = index + 1 + inPatterns.length;
|
|
532
455
|
} else if (isInOrNin) {
|
|
533
456
|
var createConstraint = (baseArray, notIn) => {
|
|
534
457
|
const not = notIn ? ' NOT ' : '';
|
|
535
|
-
|
|
536
458
|
if (baseArray.length > 0) {
|
|
537
459
|
if (isArrayField) {
|
|
538
460
|
patterns.push(`${not} array_contains($${index}:name, $${index + 1})`);
|
|
@@ -543,7 +465,6 @@ const buildWhereClause = ({
|
|
|
543
465
|
if (fieldName.indexOf('.') >= 0) {
|
|
544
466
|
return;
|
|
545
467
|
}
|
|
546
|
-
|
|
547
468
|
const inPatterns = [];
|
|
548
469
|
values.push(fieldName);
|
|
549
470
|
baseArray.forEach((listElem, listIndex) => {
|
|
@@ -572,7 +493,6 @@ const buildWhereClause = ({
|
|
|
572
493
|
if (fieldValue.$in) {
|
|
573
494
|
createConstraint(_lodash.default.flatMap(fieldValue.$in, elt => elt), false);
|
|
574
495
|
}
|
|
575
|
-
|
|
576
496
|
if (fieldValue.$nin) {
|
|
577
497
|
createConstraint(_lodash.default.flatMap(fieldValue.$nin, elt => elt), true);
|
|
578
498
|
}
|
|
@@ -581,23 +501,19 @@ const buildWhereClause = ({
|
|
|
581
501
|
} else if (typeof fieldValue.$nin !== 'undefined') {
|
|
582
502
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $nin value');
|
|
583
503
|
}
|
|
584
|
-
|
|
585
504
|
if (Array.isArray(fieldValue.$all) && isArrayField) {
|
|
586
505
|
if (isAnyValueRegexStartsWith(fieldValue.$all)) {
|
|
587
506
|
if (!isAllValuesRegexOrNone(fieldValue.$all)) {
|
|
588
507
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'All $all values must be of regex type or none: ' + fieldValue.$all);
|
|
589
508
|
}
|
|
590
|
-
|
|
591
509
|
for (let i = 0; i < fieldValue.$all.length; i += 1) {
|
|
592
510
|
const value = processRegexPattern(fieldValue.$all[i].$regex);
|
|
593
511
|
fieldValue.$all[i] = value.substring(1) + '%';
|
|
594
512
|
}
|
|
595
|
-
|
|
596
513
|
patterns.push(`array_contains_all_regex($${index}:name, $${index + 1}::jsonb)`);
|
|
597
514
|
} else {
|
|
598
515
|
patterns.push(`array_contains_all($${index}:name, $${index + 1}::jsonb)`);
|
|
599
516
|
}
|
|
600
|
-
|
|
601
517
|
values.push(fieldName, JSON.stringify(fieldValue.$all));
|
|
602
518
|
index += 2;
|
|
603
519
|
} else if (Array.isArray(fieldValue.$all)) {
|
|
@@ -607,7 +523,6 @@ const buildWhereClause = ({
|
|
|
607
523
|
index += 2;
|
|
608
524
|
}
|
|
609
525
|
}
|
|
610
|
-
|
|
611
526
|
if (typeof fieldValue.$exists !== 'undefined') {
|
|
612
527
|
if (typeof fieldValue.$exists === 'object' && fieldValue.$exists.$relativeTime) {
|
|
613
528
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators');
|
|
@@ -616,58 +531,46 @@ const buildWhereClause = ({
|
|
|
616
531
|
} else {
|
|
617
532
|
patterns.push(`$${index}:name IS NULL`);
|
|
618
533
|
}
|
|
619
|
-
|
|
620
534
|
values.push(fieldName);
|
|
621
535
|
index += 1;
|
|
622
536
|
}
|
|
623
|
-
|
|
624
537
|
if (fieldValue.$containedBy) {
|
|
625
538
|
const arr = fieldValue.$containedBy;
|
|
626
|
-
|
|
627
539
|
if (!(arr instanceof Array)) {
|
|
628
540
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $containedBy: should be an array`);
|
|
629
541
|
}
|
|
630
|
-
|
|
631
542
|
patterns.push(`$${index}:name <@ $${index + 1}::jsonb`);
|
|
632
543
|
values.push(fieldName, JSON.stringify(arr));
|
|
633
544
|
index += 2;
|
|
634
545
|
}
|
|
635
|
-
|
|
636
546
|
if (fieldValue.$text) {
|
|
637
547
|
const search = fieldValue.$text.$search;
|
|
638
548
|
let language = 'english';
|
|
639
|
-
|
|
640
549
|
if (typeof search !== 'object') {
|
|
641
550
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $search, should be object`);
|
|
642
551
|
}
|
|
643
|
-
|
|
644
552
|
if (!search.$term || typeof search.$term !== 'string') {
|
|
645
553
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $term, should be string`);
|
|
646
554
|
}
|
|
647
|
-
|
|
648
555
|
if (search.$language && typeof search.$language !== 'string') {
|
|
649
556
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $language, should be string`);
|
|
650
557
|
} else if (search.$language) {
|
|
651
558
|
language = search.$language;
|
|
652
559
|
}
|
|
653
|
-
|
|
654
560
|
if (search.$caseSensitive && typeof search.$caseSensitive !== 'boolean') {
|
|
655
561
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $caseSensitive, should be boolean`);
|
|
656
562
|
} else if (search.$caseSensitive) {
|
|
657
563
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $caseSensitive not supported, please use $regex or create a separate lower case column.`);
|
|
658
564
|
}
|
|
659
|
-
|
|
660
565
|
if (search.$diacriticSensitive && typeof search.$diacriticSensitive !== 'boolean') {
|
|
661
566
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $diacriticSensitive, should be boolean`);
|
|
662
567
|
} else if (search.$diacriticSensitive === false) {
|
|
663
568
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $text: $diacriticSensitive - false not supported, install Postgres Unaccent Extension`);
|
|
664
569
|
}
|
|
665
|
-
|
|
666
570
|
patterns.push(`to_tsvector($${index}, $${index + 1}:name) @@ to_tsquery($${index + 2}, $${index + 3})`);
|
|
667
571
|
values.push(language, fieldName, language, search.$term);
|
|
668
572
|
index += 4;
|
|
669
573
|
}
|
|
670
|
-
|
|
671
574
|
if (fieldValue.$nearSphere) {
|
|
672
575
|
const point = fieldValue.$nearSphere;
|
|
673
576
|
const distance = fieldValue.$maxDistance;
|
|
@@ -677,7 +580,6 @@ const buildWhereClause = ({
|
|
|
677
580
|
values.push(fieldName, point.longitude, point.latitude, distanceInKM);
|
|
678
581
|
index += 4;
|
|
679
582
|
}
|
|
680
|
-
|
|
681
583
|
if (fieldValue.$within && fieldValue.$within.$box) {
|
|
682
584
|
const box = fieldValue.$within.$box;
|
|
683
585
|
const left = box[0].longitude;
|
|
@@ -688,114 +590,90 @@ const buildWhereClause = ({
|
|
|
688
590
|
values.push(fieldName, `((${left}, ${bottom}), (${right}, ${top}))`);
|
|
689
591
|
index += 2;
|
|
690
592
|
}
|
|
691
|
-
|
|
692
593
|
if (fieldValue.$geoWithin && fieldValue.$geoWithin.$centerSphere) {
|
|
693
594
|
const centerSphere = fieldValue.$geoWithin.$centerSphere;
|
|
694
|
-
|
|
695
595
|
if (!(centerSphere instanceof Array) || centerSphere.length < 2) {
|
|
696
596
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoWithin value; $centerSphere should be an array of Parse.GeoPoint and distance');
|
|
697
|
-
}
|
|
698
|
-
|
|
699
|
-
|
|
597
|
+
}
|
|
598
|
+
// Get point, convert to geo point if necessary and validate
|
|
700
599
|
let point = centerSphere[0];
|
|
701
|
-
|
|
702
600
|
if (point instanceof Array && point.length === 2) {
|
|
703
601
|
point = new _node.default.GeoPoint(point[1], point[0]);
|
|
704
602
|
} else if (!GeoPointCoder.isValidJSON(point)) {
|
|
705
603
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoWithin value; $centerSphere geo point invalid');
|
|
706
604
|
}
|
|
707
|
-
|
|
708
|
-
|
|
709
|
-
|
|
710
|
-
|
|
605
|
+
_node.default.GeoPoint._validate(point.latitude, point.longitude);
|
|
606
|
+
// Get distance and validate
|
|
711
607
|
const distance = centerSphere[1];
|
|
712
|
-
|
|
713
608
|
if (isNaN(distance) || distance < 0) {
|
|
714
609
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoWithin value; $centerSphere distance invalid');
|
|
715
610
|
}
|
|
716
|
-
|
|
717
611
|
const distanceInKM = distance * 6371 * 1000;
|
|
718
612
|
patterns.push(`ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${index + 2})::geometry) <= $${index + 3}`);
|
|
719
613
|
values.push(fieldName, point.longitude, point.latitude, distanceInKM);
|
|
720
614
|
index += 4;
|
|
721
615
|
}
|
|
722
|
-
|
|
723
616
|
if (fieldValue.$geoWithin && fieldValue.$geoWithin.$polygon) {
|
|
724
617
|
const polygon = fieldValue.$geoWithin.$polygon;
|
|
725
618
|
let points;
|
|
726
|
-
|
|
727
619
|
if (typeof polygon === 'object' && polygon.__type === 'Polygon') {
|
|
728
620
|
if (!polygon.coordinates || polygon.coordinates.length < 3) {
|
|
729
621
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoWithin value; Polygon.coordinates should contain at least 3 lon/lat pairs');
|
|
730
622
|
}
|
|
731
|
-
|
|
732
623
|
points = polygon.coordinates;
|
|
733
624
|
} else if (polygon instanceof Array) {
|
|
734
625
|
if (polygon.length < 3) {
|
|
735
626
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoWithin value; $polygon should contain at least 3 GeoPoints');
|
|
736
627
|
}
|
|
737
|
-
|
|
738
628
|
points = polygon;
|
|
739
629
|
} else {
|
|
740
630
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, "bad $geoWithin value; $polygon should be Polygon object or Array of Parse.GeoPoint's");
|
|
741
631
|
}
|
|
742
|
-
|
|
743
632
|
points = points.map(point => {
|
|
744
633
|
if (point instanceof Array && point.length === 2) {
|
|
745
634
|
_node.default.GeoPoint._validate(point[1], point[0]);
|
|
746
|
-
|
|
747
635
|
return `(${point[0]}, ${point[1]})`;
|
|
748
636
|
}
|
|
749
|
-
|
|
750
637
|
if (typeof point !== 'object' || point.__type !== 'GeoPoint') {
|
|
751
638
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoWithin value');
|
|
752
639
|
} else {
|
|
753
640
|
_node.default.GeoPoint._validate(point.latitude, point.longitude);
|
|
754
641
|
}
|
|
755
|
-
|
|
756
642
|
return `(${point.longitude}, ${point.latitude})`;
|
|
757
643
|
}).join(', ');
|
|
758
644
|
patterns.push(`$${index}:name::point <@ $${index + 1}::polygon`);
|
|
759
645
|
values.push(fieldName, `(${points})`);
|
|
760
646
|
index += 2;
|
|
761
647
|
}
|
|
762
|
-
|
|
763
648
|
if (fieldValue.$geoIntersects && fieldValue.$geoIntersects.$point) {
|
|
764
649
|
const point = fieldValue.$geoIntersects.$point;
|
|
765
|
-
|
|
766
650
|
if (typeof point !== 'object' || point.__type !== 'GeoPoint') {
|
|
767
651
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, 'bad $geoIntersect value; $point should be GeoPoint');
|
|
768
652
|
} else {
|
|
769
653
|
_node.default.GeoPoint._validate(point.latitude, point.longitude);
|
|
770
654
|
}
|
|
771
|
-
|
|
772
655
|
patterns.push(`$${index}:name::polygon @> $${index + 1}::point`);
|
|
773
656
|
values.push(fieldName, `(${point.longitude}, ${point.latitude})`);
|
|
774
657
|
index += 2;
|
|
775
658
|
}
|
|
776
|
-
|
|
777
659
|
if (fieldValue.$regex) {
|
|
778
660
|
let regex = fieldValue.$regex;
|
|
779
661
|
let operator = '~';
|
|
780
662
|
const opts = fieldValue.$options;
|
|
781
|
-
|
|
782
663
|
if (opts) {
|
|
783
664
|
if (opts.indexOf('i') >= 0) {
|
|
784
665
|
operator = '~*';
|
|
785
666
|
}
|
|
786
|
-
|
|
787
667
|
if (opts.indexOf('x') >= 0) {
|
|
788
668
|
regex = removeWhiteSpace(regex);
|
|
789
669
|
}
|
|
790
670
|
}
|
|
791
|
-
|
|
792
671
|
const name = transformDotField(fieldName);
|
|
793
672
|
regex = processRegexPattern(regex);
|
|
794
673
|
patterns.push(`$${index}:raw ${operator} '$${index + 1}:raw'`);
|
|
795
674
|
values.push(name, regex);
|
|
796
675
|
index += 2;
|
|
797
676
|
}
|
|
798
|
-
|
|
799
677
|
if (fieldValue.__type === 'Pointer') {
|
|
800
678
|
if (isArrayField) {
|
|
801
679
|
patterns.push(`array_contains($${index}:name, $${index + 1})`);
|
|
@@ -807,32 +685,27 @@ const buildWhereClause = ({
|
|
|
807
685
|
index += 2;
|
|
808
686
|
}
|
|
809
687
|
}
|
|
810
|
-
|
|
811
688
|
if (fieldValue.__type === 'Date') {
|
|
812
689
|
patterns.push(`$${index}:name = $${index + 1}`);
|
|
813
690
|
values.push(fieldName, fieldValue.iso);
|
|
814
691
|
index += 2;
|
|
815
692
|
}
|
|
816
|
-
|
|
817
693
|
if (fieldValue.__type === 'GeoPoint') {
|
|
818
694
|
patterns.push(`$${index}:name ~= POINT($${index + 1}, $${index + 2})`);
|
|
819
695
|
values.push(fieldName, fieldValue.longitude, fieldValue.latitude);
|
|
820
696
|
index += 3;
|
|
821
697
|
}
|
|
822
|
-
|
|
823
698
|
if (fieldValue.__type === 'Polygon') {
|
|
824
699
|
const value = convertPolygonToSQL(fieldValue.coordinates);
|
|
825
700
|
patterns.push(`$${index}:name ~= $${index + 1}::polygon`);
|
|
826
701
|
values.push(fieldName, value);
|
|
827
702
|
index += 2;
|
|
828
703
|
}
|
|
829
|
-
|
|
830
704
|
Object.keys(ParseToPosgresComparator).forEach(cmp => {
|
|
831
705
|
if (fieldValue[cmp] || fieldValue[cmp] === 0) {
|
|
832
706
|
const pgComparator = ParseToPosgresComparator[cmp];
|
|
833
707
|
let constraintFieldName;
|
|
834
708
|
let postgresValue = toPostgresValue(fieldValue[cmp]);
|
|
835
|
-
|
|
836
709
|
if (fieldName.indexOf('.') >= 0) {
|
|
837
710
|
const castType = toPostgresValueCastType(fieldValue[cmp]);
|
|
838
711
|
constraintFieldName = castType ? `CAST ((${transformDotField(fieldName)}) AS ${castType})` : transformDotField(fieldName);
|
|
@@ -841,9 +714,7 @@ const buildWhereClause = ({
|
|
|
841
714
|
if (schema.fields[fieldName].type !== 'Date') {
|
|
842
715
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, '$relativeTime can only be used with Date field');
|
|
843
716
|
}
|
|
844
|
-
|
|
845
717
|
const parserResult = Utils.relativeTimeToDate(postgresValue.$relativeTime);
|
|
846
|
-
|
|
847
718
|
if (parserResult.status === 'success') {
|
|
848
719
|
postgresValue = toPostgresValue(parserResult.result);
|
|
849
720
|
} else {
|
|
@@ -851,21 +722,17 @@ const buildWhereClause = ({
|
|
|
851
722
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `bad $relativeTime (${postgresValue.$relativeTime}) value. ${parserResult.info}`);
|
|
852
723
|
}
|
|
853
724
|
}
|
|
854
|
-
|
|
855
725
|
constraintFieldName = `$${index++}:name`;
|
|
856
726
|
values.push(fieldName);
|
|
857
727
|
}
|
|
858
|
-
|
|
859
728
|
values.push(postgresValue);
|
|
860
729
|
patterns.push(`${constraintFieldName} ${pgComparator} $${index++}`);
|
|
861
730
|
}
|
|
862
731
|
});
|
|
863
|
-
|
|
864
732
|
if (initialPatternsLength === patterns.length) {
|
|
865
733
|
throw new _node.default.Error(_node.default.Error.OPERATION_FORBIDDEN, `Postgres doesn't support this query type yet ${JSON.stringify(fieldValue)}`);
|
|
866
734
|
}
|
|
867
735
|
}
|
|
868
|
-
|
|
869
736
|
values = values.map(transformValue);
|
|
870
737
|
return {
|
|
871
738
|
pattern: patterns.join(' AND '),
|
|
@@ -873,9 +740,9 @@ const buildWhereClause = ({
|
|
|
873
740
|
sorts
|
|
874
741
|
};
|
|
875
742
|
};
|
|
876
|
-
|
|
877
743
|
class PostgresStorageAdapter {
|
|
878
744
|
// Private
|
|
745
|
+
|
|
879
746
|
constructor({
|
|
880
747
|
uri,
|
|
881
748
|
collectionPrefix = '',
|
|
@@ -889,19 +756,16 @@ class PostgresStorageAdapter {
|
|
|
889
756
|
pgp
|
|
890
757
|
} = (0, _PostgresClient.createClient)(uri, databaseOptions);
|
|
891
758
|
this._client = client;
|
|
892
|
-
|
|
893
759
|
this._onchange = () => {};
|
|
894
|
-
|
|
895
760
|
this._pgp = pgp;
|
|
896
761
|
this._uuid = (0, _uuid.v4)();
|
|
897
762
|
this.canSortOnJoinTables = false;
|
|
898
763
|
}
|
|
899
|
-
|
|
900
764
|
watch(callback) {
|
|
901
765
|
this._onchange = callback;
|
|
902
|
-
}
|
|
903
|
-
|
|
766
|
+
}
|
|
904
767
|
|
|
768
|
+
//Note that analyze=true will run the query, executing INSERTS, DELETES, etc.
|
|
905
769
|
createExplainableQuery(query, analyze = false) {
|
|
906
770
|
if (analyze) {
|
|
907
771
|
return 'EXPLAIN (ANALYZE, FORMAT JSON) ' + query;
|
|
@@ -909,39 +773,30 @@ class PostgresStorageAdapter {
|
|
|
909
773
|
return 'EXPLAIN (FORMAT JSON) ' + query;
|
|
910
774
|
}
|
|
911
775
|
}
|
|
912
|
-
|
|
913
776
|
handleShutdown() {
|
|
914
777
|
if (this._stream) {
|
|
915
778
|
this._stream.done();
|
|
916
|
-
|
|
917
779
|
delete this._stream;
|
|
918
780
|
}
|
|
919
|
-
|
|
920
781
|
if (!this._client) {
|
|
921
782
|
return;
|
|
922
783
|
}
|
|
923
|
-
|
|
924
784
|
this._client.$pool.end();
|
|
925
785
|
}
|
|
926
|
-
|
|
927
786
|
async _listenToSchema() {
|
|
928
787
|
if (!this._stream && this.enableSchemaHooks) {
|
|
929
788
|
this._stream = await this._client.connect({
|
|
930
789
|
direct: true
|
|
931
790
|
});
|
|
932
|
-
|
|
933
791
|
this._stream.client.on('notification', data => {
|
|
934
792
|
const payload = JSON.parse(data.payload);
|
|
935
|
-
|
|
936
793
|
if (payload.senderId !== this._uuid) {
|
|
937
794
|
this._onchange();
|
|
938
795
|
}
|
|
939
796
|
});
|
|
940
|
-
|
|
941
797
|
await this._stream.none('LISTEN $1~', 'schema.change');
|
|
942
798
|
}
|
|
943
799
|
}
|
|
944
|
-
|
|
945
800
|
_notifySchemaChange() {
|
|
946
801
|
if (this._stream) {
|
|
947
802
|
this._stream.none('NOTIFY $1~, $2', ['schema.change', {
|
|
@@ -958,28 +813,22 @@ class PostgresStorageAdapter {
|
|
|
958
813
|
throw error;
|
|
959
814
|
});
|
|
960
815
|
}
|
|
961
|
-
|
|
962
816
|
async classExists(name) {
|
|
963
817
|
return this._client.one('SELECT EXISTS (SELECT 1 FROM information_schema.tables WHERE table_name = $1)', [name], a => a.exists);
|
|
964
818
|
}
|
|
965
|
-
|
|
966
819
|
async setClassLevelPermissions(className, CLPs) {
|
|
967
820
|
await this._client.task('set-class-level-permissions', async t => {
|
|
968
821
|
const values = [className, 'schema', 'classLevelPermissions', JSON.stringify(CLPs)];
|
|
969
822
|
await t.none(`UPDATE "_SCHEMA" SET $2:name = json_object_set_key($2:name, $3::text, $4::jsonb) WHERE "className" = $1`, values);
|
|
970
823
|
});
|
|
971
|
-
|
|
972
824
|
this._notifySchemaChange();
|
|
973
825
|
}
|
|
974
|
-
|
|
975
826
|
async setIndexesWithSchemaFormat(className, submittedIndexes, existingIndexes = {}, fields, conn) {
|
|
976
827
|
conn = conn || this._client;
|
|
977
828
|
const self = this;
|
|
978
|
-
|
|
979
829
|
if (submittedIndexes === undefined) {
|
|
980
830
|
return Promise.resolve();
|
|
981
831
|
}
|
|
982
|
-
|
|
983
832
|
if (Object.keys(existingIndexes).length === 0) {
|
|
984
833
|
existingIndexes = {
|
|
985
834
|
_id_: {
|
|
@@ -987,20 +836,16 @@ class PostgresStorageAdapter {
|
|
|
987
836
|
}
|
|
988
837
|
};
|
|
989
838
|
}
|
|
990
|
-
|
|
991
839
|
const deletedIndexes = [];
|
|
992
840
|
const insertedIndexes = [];
|
|
993
841
|
Object.keys(submittedIndexes).forEach(name => {
|
|
994
842
|
const field = submittedIndexes[name];
|
|
995
|
-
|
|
996
843
|
if (existingIndexes[name] && field.__op !== 'Delete') {
|
|
997
844
|
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, `Index ${name} exists, cannot update.`);
|
|
998
845
|
}
|
|
999
|
-
|
|
1000
846
|
if (!existingIndexes[name] && field.__op === 'Delete') {
|
|
1001
847
|
throw new _node.default.Error(_node.default.Error.INVALID_QUERY, `Index ${name} does not exist, cannot delete.`);
|
|
1002
848
|
}
|
|
1003
|
-
|
|
1004
849
|
if (field.__op === 'Delete') {
|
|
1005
850
|
deletedIndexes.push(name);
|
|
1006
851
|
delete existingIndexes[name];
|
|
@@ -1021,17 +866,13 @@ class PostgresStorageAdapter {
|
|
|
1021
866
|
if (insertedIndexes.length > 0) {
|
|
1022
867
|
await self.createIndexes(className, insertedIndexes, t);
|
|
1023
868
|
}
|
|
1024
|
-
|
|
1025
869
|
if (deletedIndexes.length > 0) {
|
|
1026
870
|
await self.dropIndexes(className, deletedIndexes, t);
|
|
1027
871
|
}
|
|
1028
|
-
|
|
1029
872
|
await t.none('UPDATE "_SCHEMA" SET $2:name = json_object_set_key($2:name, $3::text, $4::jsonb) WHERE "className" = $1', [className, 'schema', 'indexes', JSON.stringify(existingIndexes)]);
|
|
1030
873
|
});
|
|
1031
|
-
|
|
1032
874
|
this._notifySchemaChange();
|
|
1033
875
|
}
|
|
1034
|
-
|
|
1035
876
|
async createClass(className, schema, conn) {
|
|
1036
877
|
conn = conn || this._client;
|
|
1037
878
|
const parseSchema = await conn.tx('create-class', async t => {
|
|
@@ -1046,23 +887,19 @@ class PostgresStorageAdapter {
|
|
|
1046
887
|
if (err.code === PostgresUniqueIndexViolationError && err.detail.includes(className)) {
|
|
1047
888
|
throw new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, `Class ${className} already exists.`);
|
|
1048
889
|
}
|
|
1049
|
-
|
|
1050
890
|
throw err;
|
|
1051
891
|
});
|
|
1052
|
-
|
|
1053
892
|
this._notifySchemaChange();
|
|
1054
|
-
|
|
1055
893
|
return parseSchema;
|
|
1056
|
-
}
|
|
1057
|
-
|
|
894
|
+
}
|
|
1058
895
|
|
|
896
|
+
// Just create a table, do not insert in schema
|
|
1059
897
|
async createTable(className, schema, conn) {
|
|
1060
898
|
conn = conn || this._client;
|
|
1061
899
|
debug('createTable');
|
|
1062
900
|
const valuesArray = [];
|
|
1063
901
|
const patternsArray = [];
|
|
1064
902
|
const fields = Object.assign({}, schema.fields);
|
|
1065
|
-
|
|
1066
903
|
if (className === '_User') {
|
|
1067
904
|
fields._email_verify_token_expires_at = {
|
|
1068
905
|
type: 'Date'
|
|
@@ -1089,32 +926,27 @@ class PostgresStorageAdapter {
|
|
|
1089
926
|
type: 'Array'
|
|
1090
927
|
};
|
|
1091
928
|
}
|
|
1092
|
-
|
|
1093
929
|
let index = 2;
|
|
1094
930
|
const relations = [];
|
|
1095
931
|
Object.keys(fields).forEach(fieldName => {
|
|
1096
|
-
const parseType = fields[fieldName];
|
|
932
|
+
const parseType = fields[fieldName];
|
|
933
|
+
// Skip when it's a relation
|
|
1097
934
|
// We'll create the tables later
|
|
1098
|
-
|
|
1099
935
|
if (parseType.type === 'Relation') {
|
|
1100
936
|
relations.push(fieldName);
|
|
1101
937
|
return;
|
|
1102
938
|
}
|
|
1103
|
-
|
|
1104
939
|
if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {
|
|
1105
940
|
parseType.contents = {
|
|
1106
941
|
type: 'String'
|
|
1107
942
|
};
|
|
1108
943
|
}
|
|
1109
|
-
|
|
1110
944
|
valuesArray.push(fieldName);
|
|
1111
945
|
valuesArray.push(parseTypeToPostgresType(parseType));
|
|
1112
946
|
patternsArray.push(`$${index}:name $${index + 1}:raw`);
|
|
1113
|
-
|
|
1114
947
|
if (fieldName === 'objectId') {
|
|
1115
948
|
patternsArray.push(`PRIMARY KEY ($${index}:name)`);
|
|
1116
949
|
}
|
|
1117
|
-
|
|
1118
950
|
index = index + 2;
|
|
1119
951
|
});
|
|
1120
952
|
const qs = `CREATE TABLE IF NOT EXISTS $1:name (${patternsArray.join()})`;
|
|
@@ -1125,8 +957,8 @@ class PostgresStorageAdapter {
|
|
|
1125
957
|
} catch (error) {
|
|
1126
958
|
if (error.code !== PostgresDuplicateRelationError) {
|
|
1127
959
|
throw error;
|
|
1128
|
-
}
|
|
1129
|
-
|
|
960
|
+
}
|
|
961
|
+
// ELSE: Table already exists, must have been created by a different request. Ignore the error.
|
|
1130
962
|
}
|
|
1131
963
|
|
|
1132
964
|
await t.tx('create-table-tx', tx => {
|
|
@@ -1138,7 +970,6 @@ class PostgresStorageAdapter {
|
|
|
1138
970
|
});
|
|
1139
971
|
});
|
|
1140
972
|
}
|
|
1141
|
-
|
|
1142
973
|
async schemaUpgrade(className, schema, conn) {
|
|
1143
974
|
debug('schemaUpgrade');
|
|
1144
975
|
conn = conn || this._client;
|
|
@@ -1151,7 +982,6 @@ class PostgresStorageAdapter {
|
|
|
1151
982
|
await t.batch(newColumns);
|
|
1152
983
|
});
|
|
1153
984
|
}
|
|
1154
|
-
|
|
1155
985
|
async addFieldIfNotExists(className, fieldName, type) {
|
|
1156
986
|
// TODO: Must be revised for invalid logic...
|
|
1157
987
|
debug('addFieldIfNotExists');
|
|
@@ -1172,23 +1002,20 @@ class PostgresStorageAdapter {
|
|
|
1172
1002
|
}
|
|
1173
1003
|
}, t);
|
|
1174
1004
|
}
|
|
1175
|
-
|
|
1176
1005
|
if (error.code !== PostgresDuplicateColumnError) {
|
|
1177
1006
|
throw error;
|
|
1178
|
-
}
|
|
1179
|
-
|
|
1007
|
+
}
|
|
1008
|
+
// Column already exists, created by other request. Carry on to see if it's the right type.
|
|
1180
1009
|
}
|
|
1181
1010
|
} else {
|
|
1182
1011
|
await t.none('CREATE TABLE IF NOT EXISTS $<joinTable:name> ("relatedId" varChar(120), "owningId" varChar(120), PRIMARY KEY("relatedId", "owningId") )', {
|
|
1183
1012
|
joinTable: `_Join:${fieldName}:${className}`
|
|
1184
1013
|
});
|
|
1185
1014
|
}
|
|
1186
|
-
|
|
1187
1015
|
const result = await t.any('SELECT "schema" FROM "_SCHEMA" WHERE "className" = $<className> and ("schema"::json->\'fields\'->$<fieldName>) is not null', {
|
|
1188
1016
|
className,
|
|
1189
1017
|
fieldName
|
|
1190
1018
|
});
|
|
1191
|
-
|
|
1192
1019
|
if (result[0]) {
|
|
1193
1020
|
throw 'Attempted to add a field that already exists';
|
|
1194
1021
|
} else {
|
|
@@ -1200,10 +1027,8 @@ class PostgresStorageAdapter {
|
|
|
1200
1027
|
});
|
|
1201
1028
|
}
|
|
1202
1029
|
});
|
|
1203
|
-
|
|
1204
1030
|
this._notifySchemaChange();
|
|
1205
1031
|
}
|
|
1206
|
-
|
|
1207
1032
|
async updateFieldOptions(className, fieldName, type) {
|
|
1208
1033
|
await this._client.tx('update-schema-field-options', async t => {
|
|
1209
1034
|
const path = `{fields,${fieldName}}`;
|
|
@@ -1213,10 +1038,10 @@ class PostgresStorageAdapter {
|
|
|
1213
1038
|
className
|
|
1214
1039
|
});
|
|
1215
1040
|
});
|
|
1216
|
-
}
|
|
1217
|
-
// and resolves with false if it wasn't (eg. a join table). Rejects if deletion was impossible.
|
|
1218
|
-
|
|
1041
|
+
}
|
|
1219
1042
|
|
|
1043
|
+
// Drops a collection. Resolves with true if it was a Parse Schema (eg. _User, Custom, etc.)
|
|
1044
|
+
// and resolves with false if it wasn't (eg. a join table). Rejects if deletion was impossible.
|
|
1220
1045
|
async deleteClass(className) {
|
|
1221
1046
|
const operations = [{
|
|
1222
1047
|
query: `DROP TABLE IF EXISTS $1:name`,
|
|
@@ -1228,11 +1053,10 @@ class PostgresStorageAdapter {
|
|
|
1228
1053
|
const response = await this._client.tx(t => t.none(this._pgp.helpers.concat(operations))).then(() => className.indexOf('_Join:') != 0); // resolves with false when _Join table
|
|
1229
1054
|
|
|
1230
1055
|
this._notifySchemaChange();
|
|
1231
|
-
|
|
1232
1056
|
return response;
|
|
1233
|
-
}
|
|
1234
|
-
|
|
1057
|
+
}
|
|
1235
1058
|
|
|
1059
|
+
// Delete all data known to this adapter. Used for testing.
|
|
1236
1060
|
async deleteAllClasses() {
|
|
1237
1061
|
const now = new Date().getTime();
|
|
1238
1062
|
const helpers = this._pgp.helpers;
|
|
@@ -1254,34 +1078,34 @@ class PostgresStorageAdapter {
|
|
|
1254
1078
|
} catch (error) {
|
|
1255
1079
|
if (error.code !== PostgresRelationDoesNotExistError) {
|
|
1256
1080
|
throw error;
|
|
1257
|
-
}
|
|
1258
|
-
|
|
1081
|
+
}
|
|
1082
|
+
// No _SCHEMA collection. Don't delete anything.
|
|
1259
1083
|
}
|
|
1260
1084
|
}).then(() => {
|
|
1261
1085
|
debug(`deleteAllClasses done in ${new Date().getTime() - now}`);
|
|
1262
1086
|
});
|
|
1263
|
-
}
|
|
1087
|
+
}
|
|
1088
|
+
|
|
1089
|
+
// Remove the column and all the data. For Relations, the _Join collection is handled
|
|
1264
1090
|
// specially, this function does not delete _Join columns. It should, however, indicate
|
|
1265
1091
|
// that the relation fields does not exist anymore. In mongo, this means removing it from
|
|
1266
1092
|
// the _SCHEMA collection. There should be no actual data in the collection under the same name
|
|
1267
1093
|
// as the relation column, so it's fine to attempt to delete it. If the fields listed to be
|
|
1268
1094
|
// deleted do not exist, this function should return successfully anyways. Checking for
|
|
1269
1095
|
// attempts to delete non-existent fields is the responsibility of Parse Server.
|
|
1096
|
+
|
|
1270
1097
|
// This function is not obligated to delete fields atomically. It is given the field
|
|
1271
1098
|
// names in a list so that databases that are capable of deleting fields atomically
|
|
1272
1099
|
// may do so.
|
|
1273
|
-
// Returns a Promise.
|
|
1274
|
-
|
|
1275
1100
|
|
|
1101
|
+
// Returns a Promise.
|
|
1276
1102
|
async deleteFields(className, schema, fieldNames) {
|
|
1277
1103
|
debug('deleteFields');
|
|
1278
1104
|
fieldNames = fieldNames.reduce((list, fieldName) => {
|
|
1279
1105
|
const field = schema.fields[fieldName];
|
|
1280
|
-
|
|
1281
1106
|
if (field.type !== 'Relation') {
|
|
1282
1107
|
list.push(fieldName);
|
|
1283
1108
|
}
|
|
1284
|
-
|
|
1285
1109
|
delete schema.fields[fieldName];
|
|
1286
1110
|
return list;
|
|
1287
1111
|
}, []);
|
|
@@ -1294,29 +1118,27 @@ class PostgresStorageAdapter {
|
|
|
1294
1118
|
schema,
|
|
1295
1119
|
className
|
|
1296
1120
|
});
|
|
1297
|
-
|
|
1298
1121
|
if (values.length > 1) {
|
|
1299
1122
|
await t.none(`ALTER TABLE $1:name DROP COLUMN IF EXISTS ${columns}`, values);
|
|
1300
1123
|
}
|
|
1301
1124
|
});
|
|
1302
|
-
|
|
1303
1125
|
this._notifySchemaChange();
|
|
1304
|
-
}
|
|
1126
|
+
}
|
|
1127
|
+
|
|
1128
|
+
// Return a promise for all schemas known to this adapter, in Parse format. In case the
|
|
1305
1129
|
// schemas cannot be retrieved, returns a promise that rejects. Requirements for the
|
|
1306
1130
|
// rejection reason are TBD.
|
|
1307
|
-
|
|
1308
|
-
|
|
1309
1131
|
async getAllClasses() {
|
|
1310
1132
|
return this._client.task('get-all-classes', async t => {
|
|
1311
1133
|
return await t.map('SELECT * FROM "_SCHEMA"', null, row => toParseSchema(_objectSpread({
|
|
1312
1134
|
className: row.className
|
|
1313
1135
|
}, row.schema)));
|
|
1314
1136
|
});
|
|
1315
|
-
}
|
|
1137
|
+
}
|
|
1138
|
+
|
|
1139
|
+
// Return a promise for the schema with the given name, in Parse format. If
|
|
1316
1140
|
// this adapter doesn't know about the schema, return a promise that rejects with
|
|
1317
1141
|
// undefined as the reason.
|
|
1318
|
-
|
|
1319
|
-
|
|
1320
1142
|
async getClass(className) {
|
|
1321
1143
|
debug('getClass');
|
|
1322
1144
|
return this._client.any('SELECT * FROM "_SCHEMA" WHERE "className" = $<className>', {
|
|
@@ -1325,12 +1147,11 @@ class PostgresStorageAdapter {
|
|
|
1325
1147
|
if (result.length !== 1) {
|
|
1326
1148
|
throw undefined;
|
|
1327
1149
|
}
|
|
1328
|
-
|
|
1329
1150
|
return result[0].schema;
|
|
1330
1151
|
}).then(toParseSchema);
|
|
1331
|
-
}
|
|
1332
|
-
|
|
1152
|
+
}
|
|
1333
1153
|
|
|
1154
|
+
// TODO: remove the mongo format dependency in the return value
|
|
1334
1155
|
async createObject(className, schema, object, transactionalSession) {
|
|
1335
1156
|
debug('createObject');
|
|
1336
1157
|
let columnsArray = [];
|
|
@@ -1343,29 +1164,24 @@ class PostgresStorageAdapter {
|
|
|
1343
1164
|
if (object[fieldName] === null) {
|
|
1344
1165
|
return;
|
|
1345
1166
|
}
|
|
1346
|
-
|
|
1347
1167
|
var authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);
|
|
1348
1168
|
const authDataAlreadyExists = !!object.authData;
|
|
1349
|
-
|
|
1350
1169
|
if (authDataMatch) {
|
|
1351
1170
|
var provider = authDataMatch[1];
|
|
1352
1171
|
object['authData'] = object['authData'] || {};
|
|
1353
1172
|
object['authData'][provider] = object[fieldName];
|
|
1354
1173
|
delete object[fieldName];
|
|
1355
|
-
fieldName = 'authData';
|
|
1356
|
-
|
|
1174
|
+
fieldName = 'authData';
|
|
1175
|
+
// Avoid adding authData multiple times to the query
|
|
1357
1176
|
if (authDataAlreadyExists) {
|
|
1358
1177
|
return;
|
|
1359
1178
|
}
|
|
1360
1179
|
}
|
|
1361
|
-
|
|
1362
1180
|
columnsArray.push(fieldName);
|
|
1363
|
-
|
|
1364
1181
|
if (!schema.fields[fieldName] && className === '_User') {
|
|
1365
1182
|
if (fieldName === '_email_verify_token' || fieldName === '_failed_login_count' || fieldName === '_perishable_token' || fieldName === '_password_history') {
|
|
1366
1183
|
valuesArray.push(object[fieldName]);
|
|
1367
1184
|
}
|
|
1368
|
-
|
|
1369
1185
|
if (fieldName === '_email_verify_token_expires_at') {
|
|
1370
1186
|
if (object[fieldName]) {
|
|
1371
1187
|
valuesArray.push(object[fieldName].iso);
|
|
@@ -1373,7 +1189,6 @@ class PostgresStorageAdapter {
|
|
|
1373
1189
|
valuesArray.push(null);
|
|
1374
1190
|
}
|
|
1375
1191
|
}
|
|
1376
|
-
|
|
1377
1192
|
if (fieldName === '_account_lockout_expires_at' || fieldName === '_perishable_token_expires_at' || fieldName === '_password_changed_at') {
|
|
1378
1193
|
if (object[fieldName]) {
|
|
1379
1194
|
valuesArray.push(object[fieldName].iso);
|
|
@@ -1381,10 +1196,8 @@ class PostgresStorageAdapter {
|
|
|
1381
1196
|
valuesArray.push(null);
|
|
1382
1197
|
}
|
|
1383
1198
|
}
|
|
1384
|
-
|
|
1385
1199
|
return;
|
|
1386
1200
|
}
|
|
1387
|
-
|
|
1388
1201
|
switch (schema.fields[fieldName].type) {
|
|
1389
1202
|
case 'Date':
|
|
1390
1203
|
if (object[fieldName]) {
|
|
@@ -1392,22 +1205,17 @@ class PostgresStorageAdapter {
|
|
|
1392
1205
|
} else {
|
|
1393
1206
|
valuesArray.push(null);
|
|
1394
1207
|
}
|
|
1395
|
-
|
|
1396
1208
|
break;
|
|
1397
|
-
|
|
1398
1209
|
case 'Pointer':
|
|
1399
1210
|
valuesArray.push(object[fieldName].objectId);
|
|
1400
1211
|
break;
|
|
1401
|
-
|
|
1402
1212
|
case 'Array':
|
|
1403
1213
|
if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {
|
|
1404
1214
|
valuesArray.push(object[fieldName]);
|
|
1405
1215
|
} else {
|
|
1406
1216
|
valuesArray.push(JSON.stringify(object[fieldName]));
|
|
1407
1217
|
}
|
|
1408
|
-
|
|
1409
1218
|
break;
|
|
1410
|
-
|
|
1411
1219
|
case 'Object':
|
|
1412
1220
|
case 'Bytes':
|
|
1413
1221
|
case 'String':
|
|
@@ -1415,24 +1223,20 @@ class PostgresStorageAdapter {
|
|
|
1415
1223
|
case 'Boolean':
|
|
1416
1224
|
valuesArray.push(object[fieldName]);
|
|
1417
1225
|
break;
|
|
1418
|
-
|
|
1419
1226
|
case 'File':
|
|
1420
1227
|
valuesArray.push(object[fieldName].name);
|
|
1421
1228
|
break;
|
|
1422
|
-
|
|
1423
1229
|
case 'Polygon':
|
|
1424
1230
|
{
|
|
1425
1231
|
const value = convertPolygonToSQL(object[fieldName].coordinates);
|
|
1426
1232
|
valuesArray.push(value);
|
|
1427
1233
|
break;
|
|
1428
1234
|
}
|
|
1429
|
-
|
|
1430
1235
|
case 'GeoPoint':
|
|
1431
1236
|
// pop the point and process later
|
|
1432
1237
|
geoPoints[fieldName] = object[fieldName];
|
|
1433
1238
|
columnsArray.pop();
|
|
1434
1239
|
break;
|
|
1435
|
-
|
|
1436
1240
|
default:
|
|
1437
1241
|
throw `Type ${schema.fields[fieldName].type} not supported yet`;
|
|
1438
1242
|
}
|
|
@@ -1441,13 +1245,11 @@ class PostgresStorageAdapter {
|
|
|
1441
1245
|
const initialValues = valuesArray.map((val, index) => {
|
|
1442
1246
|
let termination = '';
|
|
1443
1247
|
const fieldName = columnsArray[index];
|
|
1444
|
-
|
|
1445
1248
|
if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {
|
|
1446
1249
|
termination = '::text[]';
|
|
1447
1250
|
} else if (schema.fields[fieldName] && schema.fields[fieldName].type === 'Array') {
|
|
1448
1251
|
termination = '::jsonb';
|
|
1449
1252
|
}
|
|
1450
|
-
|
|
1451
1253
|
return `$${index + 2 + columnsArray.length}${termination}`;
|
|
1452
1254
|
});
|
|
1453
1255
|
const geoPointsInjects = Object.keys(geoPoints).map(key => {
|
|
@@ -1466,33 +1268,27 @@ class PostgresStorageAdapter {
|
|
|
1466
1268
|
if (error.code === PostgresUniqueIndexViolationError) {
|
|
1467
1269
|
const err = new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');
|
|
1468
1270
|
err.underlyingError = error;
|
|
1469
|
-
|
|
1470
1271
|
if (error.constraint) {
|
|
1471
1272
|
const matches = error.constraint.match(/unique_([a-zA-Z]+)/);
|
|
1472
|
-
|
|
1473
1273
|
if (matches && Array.isArray(matches)) {
|
|
1474
1274
|
err.userInfo = {
|
|
1475
1275
|
duplicated_field: matches[1]
|
|
1476
1276
|
};
|
|
1477
1277
|
}
|
|
1478
1278
|
}
|
|
1479
|
-
|
|
1480
1279
|
error = err;
|
|
1481
1280
|
}
|
|
1482
|
-
|
|
1483
1281
|
throw error;
|
|
1484
1282
|
});
|
|
1485
|
-
|
|
1486
1283
|
if (transactionalSession) {
|
|
1487
1284
|
transactionalSession.batch.push(promise);
|
|
1488
1285
|
}
|
|
1489
|
-
|
|
1490
1286
|
return promise;
|
|
1491
|
-
}
|
|
1287
|
+
}
|
|
1288
|
+
|
|
1289
|
+
// Remove all objects that match the given Parse Query.
|
|
1492
1290
|
// If no objects match, reject with OBJECT_NOT_FOUND. If objects are found and deleted, resolve with undefined.
|
|
1493
1291
|
// If there is some other error, reject with INTERNAL_SERVER_ERROR.
|
|
1494
|
-
|
|
1495
|
-
|
|
1496
1292
|
async deleteObjectsByQuery(className, schema, query, transactionalSession) {
|
|
1497
1293
|
debug('deleteObjectsByQuery');
|
|
1498
1294
|
const values = [className];
|
|
@@ -1504,11 +1300,9 @@ class PostgresStorageAdapter {
|
|
|
1504
1300
|
caseInsensitive: false
|
|
1505
1301
|
});
|
|
1506
1302
|
values.push(...where.values);
|
|
1507
|
-
|
|
1508
1303
|
if (Object.keys(query).length === 0) {
|
|
1509
1304
|
where.pattern = 'TRUE';
|
|
1510
1305
|
}
|
|
1511
|
-
|
|
1512
1306
|
const qs = `WITH deleted AS (DELETE FROM $1:name WHERE ${where.pattern} RETURNING *) SELECT count(*) FROM deleted`;
|
|
1513
1307
|
const promise = (transactionalSession ? transactionalSession.t : this._client).one(qs, values, a => +a.count).then(count => {
|
|
1514
1308
|
if (count === 0) {
|
|
@@ -1519,34 +1313,31 @@ class PostgresStorageAdapter {
|
|
|
1519
1313
|
}).catch(error => {
|
|
1520
1314
|
if (error.code !== PostgresRelationDoesNotExistError) {
|
|
1521
1315
|
throw error;
|
|
1522
|
-
}
|
|
1523
|
-
|
|
1316
|
+
}
|
|
1317
|
+
// ELSE: Don't delete anything if doesn't exist
|
|
1524
1318
|
});
|
|
1525
1319
|
|
|
1526
1320
|
if (transactionalSession) {
|
|
1527
1321
|
transactionalSession.batch.push(promise);
|
|
1528
1322
|
}
|
|
1529
|
-
|
|
1530
1323
|
return promise;
|
|
1531
|
-
}
|
|
1532
|
-
|
|
1533
|
-
|
|
1324
|
+
}
|
|
1325
|
+
// Return value not currently well specified.
|
|
1534
1326
|
async findOneAndUpdate(className, schema, query, update, transactionalSession) {
|
|
1535
1327
|
debug('findOneAndUpdate');
|
|
1536
1328
|
return this.updateObjectsByQuery(className, schema, query, update, transactionalSession).then(val => val[0]);
|
|
1537
|
-
}
|
|
1538
|
-
|
|
1329
|
+
}
|
|
1539
1330
|
|
|
1331
|
+
// Apply the update to all objects that match the given Parse Query.
|
|
1540
1332
|
async updateObjectsByQuery(className, schema, query, update, transactionalSession) {
|
|
1541
1333
|
debug('updateObjectsByQuery');
|
|
1542
1334
|
const updatePatterns = [];
|
|
1543
1335
|
const values = [className];
|
|
1544
1336
|
let index = 2;
|
|
1545
1337
|
schema = toPostgresSchema(schema);
|
|
1338
|
+
const originalUpdate = _objectSpread({}, update);
|
|
1546
1339
|
|
|
1547
|
-
|
|
1548
|
-
|
|
1549
|
-
|
|
1340
|
+
// Set flag for dot notation fields
|
|
1550
1341
|
const dotNotationOptions = {};
|
|
1551
1342
|
Object.keys(update).forEach(fieldName => {
|
|
1552
1343
|
if (fieldName.indexOf('.') > -1) {
|
|
@@ -1557,12 +1348,11 @@ class PostgresStorageAdapter {
|
|
|
1557
1348
|
dotNotationOptions[fieldName] = false;
|
|
1558
1349
|
}
|
|
1559
1350
|
});
|
|
1560
|
-
update = handleDotFields(update);
|
|
1351
|
+
update = handleDotFields(update);
|
|
1352
|
+
// Resolve authData first,
|
|
1561
1353
|
// So we don't end up with multiple key updates
|
|
1562
|
-
|
|
1563
1354
|
for (const fieldName in update) {
|
|
1564
1355
|
const authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);
|
|
1565
|
-
|
|
1566
1356
|
if (authDataMatch) {
|
|
1567
1357
|
var provider = authDataMatch[1];
|
|
1568
1358
|
const value = update[fieldName];
|
|
@@ -1571,10 +1361,9 @@ class PostgresStorageAdapter {
|
|
|
1571
1361
|
update['authData'][provider] = value;
|
|
1572
1362
|
}
|
|
1573
1363
|
}
|
|
1574
|
-
|
|
1575
1364
|
for (const fieldName in update) {
|
|
1576
|
-
const fieldValue = update[fieldName];
|
|
1577
|
-
|
|
1365
|
+
const fieldValue = update[fieldName];
|
|
1366
|
+
// Drop any undefined values.
|
|
1578
1367
|
if (typeof fieldValue === 'undefined') {
|
|
1579
1368
|
delete update[fieldName];
|
|
1580
1369
|
} else if (fieldValue === null) {
|
|
@@ -1587,7 +1376,6 @@ class PostgresStorageAdapter {
|
|
|
1587
1376
|
const generate = (jsonb, key, value) => {
|
|
1588
1377
|
return `json_object_set_key(COALESCE(${jsonb}, '{}'::jsonb), ${key}, ${value})::jsonb`;
|
|
1589
1378
|
};
|
|
1590
|
-
|
|
1591
1379
|
const lastKey = `$${index}:name`;
|
|
1592
1380
|
const fieldNameIndex = index;
|
|
1593
1381
|
index += 1;
|
|
@@ -1596,7 +1384,6 @@ class PostgresStorageAdapter {
|
|
|
1596
1384
|
const str = generate(lastKey, `$${index}::text`, `$${index + 1}::jsonb`);
|
|
1597
1385
|
index += 2;
|
|
1598
1386
|
let value = fieldValue[key];
|
|
1599
|
-
|
|
1600
1387
|
if (value) {
|
|
1601
1388
|
if (value.__op === 'Delete') {
|
|
1602
1389
|
value = null;
|
|
@@ -1604,7 +1391,6 @@ class PostgresStorageAdapter {
|
|
|
1604
1391
|
value = JSON.stringify(value);
|
|
1605
1392
|
}
|
|
1606
1393
|
}
|
|
1607
|
-
|
|
1608
1394
|
values.push(key, value);
|
|
1609
1395
|
return str;
|
|
1610
1396
|
}, lastKey);
|
|
@@ -1667,7 +1453,8 @@ class PostgresStorageAdapter {
|
|
|
1667
1453
|
updatePatterns.push(`$${index}:name = $${index + 1}::polygon`);
|
|
1668
1454
|
values.push(fieldName, value);
|
|
1669
1455
|
index += 2;
|
|
1670
|
-
} else if (fieldValue.__type === 'Relation') {
|
|
1456
|
+
} else if (fieldValue.__type === 'Relation') {
|
|
1457
|
+
// noop
|
|
1671
1458
|
} else if (typeof fieldValue === 'number') {
|
|
1672
1459
|
updatePatterns.push(`$${index}:name = $${index + 1}`);
|
|
1673
1460
|
values.push(fieldName, fieldValue);
|
|
@@ -1683,18 +1470,16 @@ class PostgresStorageAdapter {
|
|
|
1683
1470
|
return value && value.__op === 'Increment' && k.split('.').length === 2 && k.split('.')[0] === fieldName;
|
|
1684
1471
|
}).map(k => k.split('.')[1]);
|
|
1685
1472
|
let incrementPatterns = '';
|
|
1686
|
-
|
|
1687
1473
|
if (keysToIncrement.length > 0) {
|
|
1688
1474
|
incrementPatterns = ' || ' + keysToIncrement.map(c => {
|
|
1689
1475
|
const amount = fieldValue[c].amount;
|
|
1690
1476
|
return `CONCAT('{"${c}":', COALESCE($${index}:name->>'${c}','0')::int + ${amount}, '}')::jsonb`;
|
|
1691
|
-
}).join(' || ');
|
|
1692
|
-
|
|
1477
|
+
}).join(' || ');
|
|
1478
|
+
// Strip the keys
|
|
1693
1479
|
keysToIncrement.forEach(key => {
|
|
1694
1480
|
delete fieldValue[key];
|
|
1695
1481
|
});
|
|
1696
1482
|
}
|
|
1697
|
-
|
|
1698
1483
|
const keysToDelete = Object.keys(originalUpdate).filter(k => {
|
|
1699
1484
|
// choose top level fields that have a delete operation set.
|
|
1700
1485
|
const value = originalUpdate[k];
|
|
@@ -1702,21 +1487,18 @@ class PostgresStorageAdapter {
|
|
|
1702
1487
|
}).map(k => k.split('.')[1]);
|
|
1703
1488
|
const deletePatterns = keysToDelete.reduce((p, c, i) => {
|
|
1704
1489
|
return p + ` - '$${index + 1 + i}:value'`;
|
|
1705
|
-
}, '');
|
|
1706
|
-
|
|
1490
|
+
}, '');
|
|
1491
|
+
// Override Object
|
|
1707
1492
|
let updateObject = "'{}'::jsonb";
|
|
1708
|
-
|
|
1709
1493
|
if (dotNotationOptions[fieldName]) {
|
|
1710
1494
|
// Merge Object
|
|
1711
1495
|
updateObject = `COALESCE($${index}:name, '{}'::jsonb)`;
|
|
1712
1496
|
}
|
|
1713
|
-
|
|
1714
1497
|
updatePatterns.push(`$${index}:name = (${updateObject} ${deletePatterns} ${incrementPatterns} || $${index + 1 + keysToDelete.length}::jsonb )`);
|
|
1715
1498
|
values.push(fieldName, ...keysToDelete, JSON.stringify(fieldValue));
|
|
1716
1499
|
index += 2 + keysToDelete.length;
|
|
1717
1500
|
} else if (Array.isArray(fieldValue) && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array') {
|
|
1718
1501
|
const expectedType = parseTypeToPostgresType(schema.fields[fieldName]);
|
|
1719
|
-
|
|
1720
1502
|
if (expectedType === 'text[]') {
|
|
1721
1503
|
updatePatterns.push(`$${index}:name = $${index + 1}::text[]`);
|
|
1722
1504
|
values.push(fieldName, fieldValue);
|
|
@@ -1734,7 +1516,6 @@ class PostgresStorageAdapter {
|
|
|
1734
1516
|
return Promise.reject(new _node.default.Error(_node.default.Error.OPERATION_FORBIDDEN, `Postgres doesn't support update ${JSON.stringify(fieldValue)} yet`));
|
|
1735
1517
|
}
|
|
1736
1518
|
}
|
|
1737
|
-
|
|
1738
1519
|
const where = buildWhereClause({
|
|
1739
1520
|
schema,
|
|
1740
1521
|
index,
|
|
@@ -1745,15 +1526,13 @@ class PostgresStorageAdapter {
|
|
|
1745
1526
|
const whereClause = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';
|
|
1746
1527
|
const qs = `UPDATE $1:name SET ${updatePatterns.join()} ${whereClause} RETURNING *`;
|
|
1747
1528
|
const promise = (transactionalSession ? transactionalSession.t : this._client).any(qs, values);
|
|
1748
|
-
|
|
1749
1529
|
if (transactionalSession) {
|
|
1750
1530
|
transactionalSession.batch.push(promise);
|
|
1751
1531
|
}
|
|
1752
|
-
|
|
1753
1532
|
return promise;
|
|
1754
|
-
}
|
|
1755
|
-
|
|
1533
|
+
}
|
|
1756
1534
|
|
|
1535
|
+
// Hopefully, we can get rid of this. It's only used for config and hooks.
|
|
1757
1536
|
upsertOneObject(className, schema, query, update, transactionalSession) {
|
|
1758
1537
|
debug('upsertOneObject');
|
|
1759
1538
|
const createValue = Object.assign({}, query, update);
|
|
@@ -1762,11 +1541,9 @@ class PostgresStorageAdapter {
|
|
|
1762
1541
|
if (error.code !== _node.default.Error.DUPLICATE_VALUE) {
|
|
1763
1542
|
throw error;
|
|
1764
1543
|
}
|
|
1765
|
-
|
|
1766
1544
|
return this.findOneAndUpdate(className, schema, query, update, transactionalSession);
|
|
1767
1545
|
});
|
|
1768
1546
|
}
|
|
1769
|
-
|
|
1770
1547
|
find(className, schema, query, {
|
|
1771
1548
|
skip,
|
|
1772
1549
|
limit,
|
|
@@ -1788,39 +1565,30 @@ class PostgresStorageAdapter {
|
|
|
1788
1565
|
values.push(...where.values);
|
|
1789
1566
|
const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';
|
|
1790
1567
|
const limitPattern = hasLimit ? `LIMIT $${values.length + 1}` : '';
|
|
1791
|
-
|
|
1792
1568
|
if (hasLimit) {
|
|
1793
1569
|
values.push(limit);
|
|
1794
1570
|
}
|
|
1795
|
-
|
|
1796
1571
|
const skipPattern = hasSkip ? `OFFSET $${values.length + 1}` : '';
|
|
1797
|
-
|
|
1798
1572
|
if (hasSkip) {
|
|
1799
1573
|
values.push(skip);
|
|
1800
1574
|
}
|
|
1801
|
-
|
|
1802
1575
|
let sortPattern = '';
|
|
1803
|
-
|
|
1804
1576
|
if (sort) {
|
|
1805
1577
|
const sortCopy = sort;
|
|
1806
1578
|
const sorting = Object.keys(sort).map(key => {
|
|
1807
|
-
const transformKey = transformDotFieldToComponents(key).join('->');
|
|
1808
|
-
|
|
1579
|
+
const transformKey = transformDotFieldToComponents(key).join('->');
|
|
1580
|
+
// Using $idx pattern gives: non-integer constant in ORDER BY
|
|
1809
1581
|
if (sortCopy[key] === 1) {
|
|
1810
1582
|
return `${transformKey} ASC`;
|
|
1811
1583
|
}
|
|
1812
|
-
|
|
1813
1584
|
return `${transformKey} DESC`;
|
|
1814
1585
|
}).join();
|
|
1815
1586
|
sortPattern = sort !== undefined && Object.keys(sort).length > 0 ? `ORDER BY ${sorting}` : '';
|
|
1816
1587
|
}
|
|
1817
|
-
|
|
1818
1588
|
if (where.sorts && Object.keys(where.sorts).length > 0) {
|
|
1819
1589
|
sortPattern = `ORDER BY ${where.sorts.join()}`;
|
|
1820
1590
|
}
|
|
1821
|
-
|
|
1822
1591
|
let columns = '*';
|
|
1823
|
-
|
|
1824
1592
|
if (keys) {
|
|
1825
1593
|
// Exclude empty keys
|
|
1826
1594
|
// Replace ACL by it's keys
|
|
@@ -1828,22 +1596,23 @@ class PostgresStorageAdapter {
|
|
|
1828
1596
|
if (key === 'ACL') {
|
|
1829
1597
|
memo.push('_rperm');
|
|
1830
1598
|
memo.push('_wperm');
|
|
1831
|
-
} else if (key.length > 0 && (
|
|
1599
|
+
} else if (key.length > 0 && (
|
|
1600
|
+
// Remove selected field not referenced in the schema
|
|
1601
|
+
// Relation is not a column in postgres
|
|
1602
|
+
// $score is a Parse special field and is also not a column
|
|
1603
|
+
schema.fields[key] && schema.fields[key].type !== 'Relation' || key === '$score')) {
|
|
1832
1604
|
memo.push(key);
|
|
1833
1605
|
}
|
|
1834
|
-
|
|
1835
1606
|
return memo;
|
|
1836
1607
|
}, []);
|
|
1837
1608
|
columns = keys.map((key, index) => {
|
|
1838
1609
|
if (key === '$score') {
|
|
1839
1610
|
return `ts_rank_cd(to_tsvector($${2}, $${3}:name), to_tsquery($${4}, $${5}), 32) as score`;
|
|
1840
1611
|
}
|
|
1841
|
-
|
|
1842
1612
|
return `$${index + values.length + 1}:name`;
|
|
1843
1613
|
}).join();
|
|
1844
1614
|
values = values.concat(keys);
|
|
1845
1615
|
}
|
|
1846
|
-
|
|
1847
1616
|
const originalQuery = `SELECT ${columns} FROM $1:name ${wherePattern} ${sortPattern} ${limitPattern} ${skipPattern}`;
|
|
1848
1617
|
const qs = explain ? this.createExplainableQuery(originalQuery) : originalQuery;
|
|
1849
1618
|
return this._client.any(qs, values).catch(error => {
|
|
@@ -1851,19 +1620,17 @@ class PostgresStorageAdapter {
|
|
|
1851
1620
|
if (error.code !== PostgresRelationDoesNotExistError) {
|
|
1852
1621
|
throw error;
|
|
1853
1622
|
}
|
|
1854
|
-
|
|
1855
1623
|
return [];
|
|
1856
1624
|
}).then(results => {
|
|
1857
1625
|
if (explain) {
|
|
1858
1626
|
return results;
|
|
1859
1627
|
}
|
|
1860
|
-
|
|
1861
1628
|
return results.map(object => this.postgresObjectToParseObject(className, object, schema));
|
|
1862
1629
|
});
|
|
1863
|
-
}
|
|
1864
|
-
// Does not strip out anything based on a lack of authentication.
|
|
1865
|
-
|
|
1630
|
+
}
|
|
1866
1631
|
|
|
1632
|
+
// Converts from a postgres-format object to a REST-format object.
|
|
1633
|
+
// Does not strip out anything based on a lack of authentication.
|
|
1867
1634
|
postgresObjectToParseObject(className, object, schema) {
|
|
1868
1635
|
Object.keys(schema.fields).forEach(fieldName => {
|
|
1869
1636
|
if (schema.fields[fieldName].type === 'Pointer' && object[fieldName]) {
|
|
@@ -1873,14 +1640,12 @@ class PostgresStorageAdapter {
|
|
|
1873
1640
|
className: schema.fields[fieldName].targetClass
|
|
1874
1641
|
};
|
|
1875
1642
|
}
|
|
1876
|
-
|
|
1877
1643
|
if (schema.fields[fieldName].type === 'Relation') {
|
|
1878
1644
|
object[fieldName] = {
|
|
1879
1645
|
__type: 'Relation',
|
|
1880
1646
|
className: schema.fields[fieldName].targetClass
|
|
1881
1647
|
};
|
|
1882
1648
|
}
|
|
1883
|
-
|
|
1884
1649
|
if (object[fieldName] && schema.fields[fieldName].type === 'GeoPoint') {
|
|
1885
1650
|
object[fieldName] = {
|
|
1886
1651
|
__type: 'GeoPoint',
|
|
@@ -1888,7 +1653,6 @@ class PostgresStorageAdapter {
|
|
|
1888
1653
|
longitude: object[fieldName].x
|
|
1889
1654
|
};
|
|
1890
1655
|
}
|
|
1891
|
-
|
|
1892
1656
|
if (object[fieldName] && schema.fields[fieldName].type === 'Polygon') {
|
|
1893
1657
|
let coords = object[fieldName];
|
|
1894
1658
|
coords = coords.substr(2, coords.length - 4).split('),(');
|
|
@@ -1900,63 +1664,54 @@ class PostgresStorageAdapter {
|
|
|
1900
1664
|
coordinates: coords
|
|
1901
1665
|
};
|
|
1902
1666
|
}
|
|
1903
|
-
|
|
1904
1667
|
if (object[fieldName] && schema.fields[fieldName].type === 'File') {
|
|
1905
1668
|
object[fieldName] = {
|
|
1906
1669
|
__type: 'File',
|
|
1907
1670
|
name: object[fieldName]
|
|
1908
1671
|
};
|
|
1909
1672
|
}
|
|
1910
|
-
});
|
|
1911
|
-
|
|
1673
|
+
});
|
|
1674
|
+
//TODO: remove this reliance on the mongo format. DB adapter shouldn't know there is a difference between created at and any other date field.
|
|
1912
1675
|
if (object.createdAt) {
|
|
1913
1676
|
object.createdAt = object.createdAt.toISOString();
|
|
1914
1677
|
}
|
|
1915
|
-
|
|
1916
1678
|
if (object.updatedAt) {
|
|
1917
1679
|
object.updatedAt = object.updatedAt.toISOString();
|
|
1918
1680
|
}
|
|
1919
|
-
|
|
1920
1681
|
if (object.expiresAt) {
|
|
1921
1682
|
object.expiresAt = {
|
|
1922
1683
|
__type: 'Date',
|
|
1923
1684
|
iso: object.expiresAt.toISOString()
|
|
1924
1685
|
};
|
|
1925
1686
|
}
|
|
1926
|
-
|
|
1927
1687
|
if (object._email_verify_token_expires_at) {
|
|
1928
1688
|
object._email_verify_token_expires_at = {
|
|
1929
1689
|
__type: 'Date',
|
|
1930
1690
|
iso: object._email_verify_token_expires_at.toISOString()
|
|
1931
1691
|
};
|
|
1932
1692
|
}
|
|
1933
|
-
|
|
1934
1693
|
if (object._account_lockout_expires_at) {
|
|
1935
1694
|
object._account_lockout_expires_at = {
|
|
1936
1695
|
__type: 'Date',
|
|
1937
1696
|
iso: object._account_lockout_expires_at.toISOString()
|
|
1938
1697
|
};
|
|
1939
1698
|
}
|
|
1940
|
-
|
|
1941
1699
|
if (object._perishable_token_expires_at) {
|
|
1942
1700
|
object._perishable_token_expires_at = {
|
|
1943
1701
|
__type: 'Date',
|
|
1944
1702
|
iso: object._perishable_token_expires_at.toISOString()
|
|
1945
1703
|
};
|
|
1946
1704
|
}
|
|
1947
|
-
|
|
1948
1705
|
if (object._password_changed_at) {
|
|
1949
1706
|
object._password_changed_at = {
|
|
1950
1707
|
__type: 'Date',
|
|
1951
1708
|
iso: object._password_changed_at.toISOString()
|
|
1952
1709
|
};
|
|
1953
1710
|
}
|
|
1954
|
-
|
|
1955
1711
|
for (const fieldName in object) {
|
|
1956
1712
|
if (object[fieldName] === null) {
|
|
1957
1713
|
delete object[fieldName];
|
|
1958
1714
|
}
|
|
1959
|
-
|
|
1960
1715
|
if (object[fieldName] instanceof Date) {
|
|
1961
1716
|
object[fieldName] = {
|
|
1962
1717
|
__type: 'Date',
|
|
@@ -1964,21 +1719,21 @@ class PostgresStorageAdapter {
|
|
|
1964
1719
|
};
|
|
1965
1720
|
}
|
|
1966
1721
|
}
|
|
1967
|
-
|
|
1968
1722
|
return object;
|
|
1969
|
-
}
|
|
1723
|
+
}
|
|
1724
|
+
|
|
1725
|
+
// Create a unique index. Unique indexes on nullable fields are not allowed. Since we don't
|
|
1970
1726
|
// currently know which fields are nullable and which aren't, we ignore that criteria.
|
|
1971
1727
|
// As such, we shouldn't expose this function to users of parse until we have an out-of-band
|
|
1972
1728
|
// Way of determining if a field is nullable. Undefined doesn't count against uniqueness,
|
|
1973
1729
|
// which is why we use sparse indexes.
|
|
1974
|
-
|
|
1975
|
-
|
|
1976
1730
|
async ensureUniqueness(className, schema, fieldNames) {
|
|
1977
1731
|
const constraintName = `${className}_unique_${fieldNames.sort().join('_')}`;
|
|
1978
1732
|
const constraintPatterns = fieldNames.map((fieldName, index) => `$${index + 3}:name`);
|
|
1979
1733
|
const qs = `CREATE UNIQUE INDEX IF NOT EXISTS $2:name ON $1:name(${constraintPatterns.join()})`;
|
|
1980
1734
|
return this._client.none(qs, [className, constraintName, ...fieldNames]).catch(error => {
|
|
1981
|
-
if (error.code === PostgresDuplicateRelationError && error.message.includes(constraintName)) {
|
|
1735
|
+
if (error.code === PostgresDuplicateRelationError && error.message.includes(constraintName)) {
|
|
1736
|
+
// Index already exists. Ignore error.
|
|
1982
1737
|
} else if (error.code === PostgresUniqueIndexViolationError && error.message.includes(constraintName)) {
|
|
1983
1738
|
// Cast the error into the proper parse error
|
|
1984
1739
|
throw new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');
|
|
@@ -1986,9 +1741,9 @@ class PostgresStorageAdapter {
|
|
|
1986
1741
|
throw error;
|
|
1987
1742
|
}
|
|
1988
1743
|
});
|
|
1989
|
-
}
|
|
1990
|
-
|
|
1744
|
+
}
|
|
1991
1745
|
|
|
1746
|
+
// Executes a count.
|
|
1992
1747
|
async count(className, schema, query, readPreference, estimate = true) {
|
|
1993
1748
|
debug('count');
|
|
1994
1749
|
const values = [className];
|
|
@@ -2001,13 +1756,11 @@ class PostgresStorageAdapter {
|
|
|
2001
1756
|
values.push(...where.values);
|
|
2002
1757
|
const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';
|
|
2003
1758
|
let qs = '';
|
|
2004
|
-
|
|
2005
1759
|
if (where.pattern.length > 0 || !estimate) {
|
|
2006
1760
|
qs = `SELECT count(*) FROM $1:name ${wherePattern}`;
|
|
2007
1761
|
} else {
|
|
2008
1762
|
qs = 'SELECT reltuples AS approximate_row_count FROM pg_class WHERE relname = $1';
|
|
2009
1763
|
}
|
|
2010
|
-
|
|
2011
1764
|
return this._client.one(qs, values, a => {
|
|
2012
1765
|
if (a.approximate_row_count == null || a.approximate_row_count == -1) {
|
|
2013
1766
|
return !isNaN(+a.count) ? +a.count : 0;
|
|
@@ -2018,22 +1771,18 @@ class PostgresStorageAdapter {
|
|
|
2018
1771
|
if (error.code !== PostgresRelationDoesNotExistError) {
|
|
2019
1772
|
throw error;
|
|
2020
1773
|
}
|
|
2021
|
-
|
|
2022
1774
|
return 0;
|
|
2023
1775
|
});
|
|
2024
1776
|
}
|
|
2025
|
-
|
|
2026
1777
|
async distinct(className, schema, query, fieldName) {
|
|
2027
1778
|
debug('distinct');
|
|
2028
1779
|
let field = fieldName;
|
|
2029
1780
|
let column = fieldName;
|
|
2030
1781
|
const isNested = fieldName.indexOf('.') >= 0;
|
|
2031
|
-
|
|
2032
1782
|
if (isNested) {
|
|
2033
1783
|
field = transformDotFieldToComponents(fieldName).join('->');
|
|
2034
1784
|
column = fieldName.split('.')[0];
|
|
2035
1785
|
}
|
|
2036
|
-
|
|
2037
1786
|
const isArrayField = schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array';
|
|
2038
1787
|
const isPointerField = schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';
|
|
2039
1788
|
const values = [field, column, className];
|
|
@@ -2047,16 +1796,13 @@ class PostgresStorageAdapter {
|
|
|
2047
1796
|
const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';
|
|
2048
1797
|
const transformer = isArrayField ? 'jsonb_array_elements' : 'ON';
|
|
2049
1798
|
let qs = `SELECT DISTINCT ${transformer}($1:name) $2:name FROM $3:name ${wherePattern}`;
|
|
2050
|
-
|
|
2051
1799
|
if (isNested) {
|
|
2052
1800
|
qs = `SELECT DISTINCT ${transformer}($1:raw) $2:raw FROM $3:name ${wherePattern}`;
|
|
2053
1801
|
}
|
|
2054
|
-
|
|
2055
1802
|
return this._client.any(qs, values).catch(error => {
|
|
2056
1803
|
if (error.code === PostgresMissingColumnError) {
|
|
2057
1804
|
return [];
|
|
2058
1805
|
}
|
|
2059
|
-
|
|
2060
1806
|
throw error;
|
|
2061
1807
|
}).then(results => {
|
|
2062
1808
|
if (!isNested) {
|
|
@@ -2065,7 +1811,6 @@ class PostgresStorageAdapter {
|
|
|
2065
1811
|
if (!isPointerField) {
|
|
2066
1812
|
return object[field];
|
|
2067
1813
|
}
|
|
2068
|
-
|
|
2069
1814
|
return {
|
|
2070
1815
|
__type: 'Pointer',
|
|
2071
1816
|
className: schema.fields[fieldName].targetClass,
|
|
@@ -2073,12 +1818,10 @@ class PostgresStorageAdapter {
|
|
|
2073
1818
|
};
|
|
2074
1819
|
});
|
|
2075
1820
|
}
|
|
2076
|
-
|
|
2077
1821
|
const child = fieldName.split('.')[1];
|
|
2078
1822
|
return results.map(object => object[column][child]);
|
|
2079
1823
|
}).then(results => results.map(object => this.postgresObjectToParseObject(className, object, schema)));
|
|
2080
1824
|
}
|
|
2081
|
-
|
|
2082
1825
|
async aggregate(className, schema, pipeline, readPreference, hint, explain) {
|
|
2083
1826
|
debug('aggregate');
|
|
2084
1827
|
const values = [className];
|
|
@@ -2091,18 +1834,14 @@ class PostgresStorageAdapter {
|
|
|
2091
1834
|
let skipPattern = '';
|
|
2092
1835
|
let sortPattern = '';
|
|
2093
1836
|
let groupPattern = '';
|
|
2094
|
-
|
|
2095
1837
|
for (let i = 0; i < pipeline.length; i += 1) {
|
|
2096
1838
|
const stage = pipeline[i];
|
|
2097
|
-
|
|
2098
1839
|
if (stage.$group) {
|
|
2099
1840
|
for (const field in stage.$group) {
|
|
2100
1841
|
const value = stage.$group[field];
|
|
2101
|
-
|
|
2102
1842
|
if (value === null || value === undefined) {
|
|
2103
1843
|
continue;
|
|
2104
1844
|
}
|
|
2105
|
-
|
|
2106
1845
|
if (field === '_id' && typeof value === 'string' && value !== '') {
|
|
2107
1846
|
columns.push(`$${index}:name AS "objectId"`);
|
|
2108
1847
|
groupPattern = `GROUP BY $${index}:name`;
|
|
@@ -2110,44 +1849,36 @@ class PostgresStorageAdapter {
|
|
|
2110
1849
|
index += 1;
|
|
2111
1850
|
continue;
|
|
2112
1851
|
}
|
|
2113
|
-
|
|
2114
1852
|
if (field === '_id' && typeof value === 'object' && Object.keys(value).length !== 0) {
|
|
2115
1853
|
groupValues = value;
|
|
2116
1854
|
const groupByFields = [];
|
|
2117
|
-
|
|
2118
1855
|
for (const alias in value) {
|
|
2119
1856
|
if (typeof value[alias] === 'string' && value[alias]) {
|
|
2120
1857
|
const source = transformAggregateField(value[alias]);
|
|
2121
|
-
|
|
2122
1858
|
if (!groupByFields.includes(`"${source}"`)) {
|
|
2123
1859
|
groupByFields.push(`"${source}"`);
|
|
2124
1860
|
}
|
|
2125
|
-
|
|
2126
1861
|
values.push(source, alias);
|
|
2127
1862
|
columns.push(`$${index}:name AS $${index + 1}:name`);
|
|
2128
1863
|
index += 2;
|
|
2129
1864
|
} else {
|
|
2130
1865
|
const operation = Object.keys(value[alias])[0];
|
|
2131
1866
|
const source = transformAggregateField(value[alias][operation]);
|
|
2132
|
-
|
|
2133
1867
|
if (mongoAggregateToPostgres[operation]) {
|
|
2134
1868
|
if (!groupByFields.includes(`"${source}"`)) {
|
|
2135
1869
|
groupByFields.push(`"${source}"`);
|
|
2136
1870
|
}
|
|
2137
|
-
|
|
2138
1871
|
columns.push(`EXTRACT(${mongoAggregateToPostgres[operation]} FROM $${index}:name AT TIME ZONE 'UTC')::integer AS $${index + 1}:name`);
|
|
2139
1872
|
values.push(source, alias);
|
|
2140
1873
|
index += 2;
|
|
2141
1874
|
}
|
|
2142
1875
|
}
|
|
2143
1876
|
}
|
|
2144
|
-
|
|
2145
1877
|
groupPattern = `GROUP BY $${index}:raw`;
|
|
2146
1878
|
values.push(groupByFields.join());
|
|
2147
1879
|
index += 1;
|
|
2148
1880
|
continue;
|
|
2149
1881
|
}
|
|
2150
|
-
|
|
2151
1882
|
if (typeof value === 'object') {
|
|
2152
1883
|
if (value.$sum) {
|
|
2153
1884
|
if (typeof value.$sum === 'string') {
|
|
@@ -2161,19 +1892,16 @@ class PostgresStorageAdapter {
|
|
|
2161
1892
|
index += 1;
|
|
2162
1893
|
}
|
|
2163
1894
|
}
|
|
2164
|
-
|
|
2165
1895
|
if (value.$max) {
|
|
2166
1896
|
columns.push(`MAX($${index}:name) AS $${index + 1}:name`);
|
|
2167
1897
|
values.push(transformAggregateField(value.$max), field);
|
|
2168
1898
|
index += 2;
|
|
2169
1899
|
}
|
|
2170
|
-
|
|
2171
1900
|
if (value.$min) {
|
|
2172
1901
|
columns.push(`MIN($${index}:name) AS $${index + 1}:name`);
|
|
2173
1902
|
values.push(transformAggregateField(value.$min), field);
|
|
2174
1903
|
index += 2;
|
|
2175
1904
|
}
|
|
2176
|
-
|
|
2177
1905
|
if (value.$avg) {
|
|
2178
1906
|
columns.push(`AVG($${index}:name) AS $${index + 1}:name`);
|
|
2179
1907
|
values.push(transformAggregateField(value.$avg), field);
|
|
@@ -2184,15 +1912,12 @@ class PostgresStorageAdapter {
|
|
|
2184
1912
|
} else {
|
|
2185
1913
|
columns.push('*');
|
|
2186
1914
|
}
|
|
2187
|
-
|
|
2188
1915
|
if (stage.$project) {
|
|
2189
1916
|
if (columns.includes('*')) {
|
|
2190
1917
|
columns = [];
|
|
2191
1918
|
}
|
|
2192
|
-
|
|
2193
1919
|
for (const field in stage.$project) {
|
|
2194
1920
|
const value = stage.$project[field];
|
|
2195
|
-
|
|
2196
1921
|
if (value === 1 || value === true) {
|
|
2197
1922
|
columns.push(`$${index}:name`);
|
|
2198
1923
|
values.push(field);
|
|
@@ -2200,11 +1925,9 @@ class PostgresStorageAdapter {
|
|
|
2200
1925
|
}
|
|
2201
1926
|
}
|
|
2202
1927
|
}
|
|
2203
|
-
|
|
2204
1928
|
if (stage.$match) {
|
|
2205
1929
|
const patterns = [];
|
|
2206
1930
|
const orOrAnd = Object.prototype.hasOwnProperty.call(stage.$match, '$or') ? ' OR ' : ' AND ';
|
|
2207
|
-
|
|
2208
1931
|
if (stage.$match.$or) {
|
|
2209
1932
|
const collapse = {};
|
|
2210
1933
|
stage.$match.$or.forEach(element => {
|
|
@@ -2214,9 +1937,11 @@ class PostgresStorageAdapter {
|
|
|
2214
1937
|
});
|
|
2215
1938
|
stage.$match = collapse;
|
|
2216
1939
|
}
|
|
2217
|
-
|
|
2218
|
-
for (const field in stage.$match) {
|
|
1940
|
+
for (let field in stage.$match) {
|
|
2219
1941
|
const value = stage.$match[field];
|
|
1942
|
+
if (field === '_id') {
|
|
1943
|
+
field = 'objectId';
|
|
1944
|
+
}
|
|
2220
1945
|
const matchPatterns = [];
|
|
2221
1946
|
Object.keys(ParseToPosgresComparator).forEach(cmp => {
|
|
2222
1947
|
if (value[cmp]) {
|
|
@@ -2226,33 +1951,27 @@ class PostgresStorageAdapter {
|
|
|
2226
1951
|
index += 2;
|
|
2227
1952
|
}
|
|
2228
1953
|
});
|
|
2229
|
-
|
|
2230
1954
|
if (matchPatterns.length > 0) {
|
|
2231
1955
|
patterns.push(`(${matchPatterns.join(' AND ')})`);
|
|
2232
1956
|
}
|
|
2233
|
-
|
|
2234
1957
|
if (schema.fields[field] && schema.fields[field].type && matchPatterns.length === 0) {
|
|
2235
1958
|
patterns.push(`$${index}:name = $${index + 1}`);
|
|
2236
1959
|
values.push(field, value);
|
|
2237
1960
|
index += 2;
|
|
2238
1961
|
}
|
|
2239
1962
|
}
|
|
2240
|
-
|
|
2241
1963
|
wherePattern = patterns.length > 0 ? `WHERE ${patterns.join(` ${orOrAnd} `)}` : '';
|
|
2242
1964
|
}
|
|
2243
|
-
|
|
2244
1965
|
if (stage.$limit) {
|
|
2245
1966
|
limitPattern = `LIMIT $${index}`;
|
|
2246
1967
|
values.push(stage.$limit);
|
|
2247
1968
|
index += 1;
|
|
2248
1969
|
}
|
|
2249
|
-
|
|
2250
1970
|
if (stage.$skip) {
|
|
2251
1971
|
skipPattern = `OFFSET $${index}`;
|
|
2252
1972
|
values.push(stage.$skip);
|
|
2253
1973
|
index += 1;
|
|
2254
1974
|
}
|
|
2255
|
-
|
|
2256
1975
|
if (stage.$sort) {
|
|
2257
1976
|
const sort = stage.$sort;
|
|
2258
1977
|
const keys = Object.keys(sort);
|
|
@@ -2266,7 +1985,6 @@ class PostgresStorageAdapter {
|
|
|
2266
1985
|
sortPattern = sort !== undefined && sorting.length > 0 ? `ORDER BY ${sorting}` : '';
|
|
2267
1986
|
}
|
|
2268
1987
|
}
|
|
2269
|
-
|
|
2270
1988
|
if (groupPattern) {
|
|
2271
1989
|
columns.forEach((e, i, a) => {
|
|
2272
1990
|
if (e && e.trim() === '*') {
|
|
@@ -2274,29 +1992,24 @@ class PostgresStorageAdapter {
|
|
|
2274
1992
|
}
|
|
2275
1993
|
});
|
|
2276
1994
|
}
|
|
2277
|
-
|
|
2278
1995
|
const originalQuery = `SELECT ${columns.filter(Boolean).join()} FROM $1:name ${wherePattern} ${skipPattern} ${groupPattern} ${sortPattern} ${limitPattern}`;
|
|
2279
1996
|
const qs = explain ? this.createExplainableQuery(originalQuery) : originalQuery;
|
|
2280
1997
|
return this._client.any(qs, values).then(a => {
|
|
2281
1998
|
if (explain) {
|
|
2282
1999
|
return a;
|
|
2283
2000
|
}
|
|
2284
|
-
|
|
2285
2001
|
const results = a.map(object => this.postgresObjectToParseObject(className, object, schema));
|
|
2286
2002
|
results.forEach(result => {
|
|
2287
2003
|
if (!Object.prototype.hasOwnProperty.call(result, 'objectId')) {
|
|
2288
2004
|
result.objectId = null;
|
|
2289
2005
|
}
|
|
2290
|
-
|
|
2291
2006
|
if (groupValues) {
|
|
2292
2007
|
result.objectId = {};
|
|
2293
|
-
|
|
2294
2008
|
for (const key in groupValues) {
|
|
2295
2009
|
result.objectId[key] = result[key];
|
|
2296
2010
|
delete result[key];
|
|
2297
2011
|
}
|
|
2298
2012
|
}
|
|
2299
|
-
|
|
2300
2013
|
if (countField) {
|
|
2301
2014
|
result[countField] = parseInt(result[countField], 10);
|
|
2302
2015
|
}
|
|
@@ -2304,7 +2017,6 @@ class PostgresStorageAdapter {
|
|
|
2304
2017
|
return results;
|
|
2305
2018
|
});
|
|
2306
2019
|
}
|
|
2307
|
-
|
|
2308
2020
|
async performInitialization({
|
|
2309
2021
|
VolatileClassesSchemas
|
|
2310
2022
|
}) {
|
|
@@ -2316,7 +2028,6 @@ class PostgresStorageAdapter {
|
|
|
2316
2028
|
if (err.code === PostgresDuplicateRelationError || err.code === _node.default.Error.INVALID_CLASS_NAME) {
|
|
2317
2029
|
return Promise.resolve();
|
|
2318
2030
|
}
|
|
2319
|
-
|
|
2320
2031
|
throw err;
|
|
2321
2032
|
}).then(() => this.schemaUpgrade(schema.className, schema));
|
|
2322
2033
|
});
|
|
@@ -2339,17 +2050,14 @@ class PostgresStorageAdapter {
|
|
|
2339
2050
|
console.error(error);
|
|
2340
2051
|
});
|
|
2341
2052
|
}
|
|
2342
|
-
|
|
2343
2053
|
async createIndexes(className, indexes, conn) {
|
|
2344
2054
|
return (conn || this._client).tx(t => t.batch(indexes.map(i => {
|
|
2345
2055
|
return t.none('CREATE INDEX IF NOT EXISTS $1:name ON $2:name ($3:name)', [i.name, className, i.key]);
|
|
2346
2056
|
})));
|
|
2347
2057
|
}
|
|
2348
|
-
|
|
2349
2058
|
async createIndexesIfNeeded(className, fieldName, type, conn) {
|
|
2350
2059
|
await (conn || this._client).none('CREATE INDEX IF NOT EXISTS $1:name ON $2:name ($3:name)', [fieldName, className, type]);
|
|
2351
2060
|
}
|
|
2352
|
-
|
|
2353
2061
|
async dropIndexes(className, indexes, conn) {
|
|
2354
2062
|
const queries = indexes.map(i => ({
|
|
2355
2063
|
query: 'DROP INDEX $1:name',
|
|
@@ -2357,23 +2065,20 @@ class PostgresStorageAdapter {
|
|
|
2357
2065
|
}));
|
|
2358
2066
|
await (conn || this._client).tx(t => t.none(this._pgp.helpers.concat(queries)));
|
|
2359
2067
|
}
|
|
2360
|
-
|
|
2361
2068
|
async getIndexes(className) {
|
|
2362
2069
|
const qs = 'SELECT * FROM pg_indexes WHERE tablename = ${className}';
|
|
2363
2070
|
return this._client.any(qs, {
|
|
2364
2071
|
className
|
|
2365
2072
|
});
|
|
2366
2073
|
}
|
|
2367
|
-
|
|
2368
2074
|
async updateSchemaWithIndexes() {
|
|
2369
2075
|
return Promise.resolve();
|
|
2370
|
-
}
|
|
2371
|
-
|
|
2076
|
+
}
|
|
2372
2077
|
|
|
2078
|
+
// Used for testing purposes
|
|
2373
2079
|
async updateEstimatedCount(className) {
|
|
2374
2080
|
return this._client.none('ANALYZE $1:name', [className]);
|
|
2375
2081
|
}
|
|
2376
|
-
|
|
2377
2082
|
async createTransactionalSession() {
|
|
2378
2083
|
return new Promise(resolve => {
|
|
2379
2084
|
const transactionalSession = {};
|
|
@@ -2388,19 +2093,16 @@ class PostgresStorageAdapter {
|
|
|
2388
2093
|
});
|
|
2389
2094
|
});
|
|
2390
2095
|
}
|
|
2391
|
-
|
|
2392
2096
|
commitTransactionalSession(transactionalSession) {
|
|
2393
2097
|
transactionalSession.resolve(transactionalSession.t.batch(transactionalSession.batch));
|
|
2394
2098
|
return transactionalSession.result;
|
|
2395
2099
|
}
|
|
2396
|
-
|
|
2397
2100
|
abortTransactionalSession(transactionalSession) {
|
|
2398
2101
|
const result = transactionalSession.result.catch();
|
|
2399
2102
|
transactionalSession.batch.push(Promise.reject());
|
|
2400
2103
|
transactionalSession.resolve(transactionalSession.t.batch(transactionalSession.batch));
|
|
2401
2104
|
return result;
|
|
2402
2105
|
}
|
|
2403
|
-
|
|
2404
2106
|
async ensureIndex(className, schema, fieldNames, indexName, caseInsensitive = false, options = {}) {
|
|
2405
2107
|
const conn = options.conn !== undefined ? options.conn : this._client;
|
|
2406
2108
|
const defaultIndexName = `parse_default_${fieldNames.sort().join('_')}`;
|
|
@@ -2412,13 +2114,12 @@ class PostgresStorageAdapter {
|
|
|
2412
2114
|
const constraintPatterns = caseInsensitive ? fieldNames.map((fieldName, index) => `lower($${index + 3}:name) varchar_pattern_ops`) : fieldNames.map((fieldName, index) => `$${index + 3}:name`);
|
|
2413
2115
|
const qs = `CREATE INDEX IF NOT EXISTS $1:name ON $2:name (${constraintPatterns.join()})`;
|
|
2414
2116
|
const setIdempotencyFunction = options.setIdempotencyFunction !== undefined ? options.setIdempotencyFunction : false;
|
|
2415
|
-
|
|
2416
2117
|
if (setIdempotencyFunction) {
|
|
2417
2118
|
await this.ensureIdempotencyFunctionExists(options);
|
|
2418
2119
|
}
|
|
2419
|
-
|
|
2420
2120
|
await conn.none(qs, [indexNameOptions.name, className, ...fieldNames]).catch(error => {
|
|
2421
|
-
if (error.code === PostgresDuplicateRelationError && error.message.includes(indexNameOptions.name)) {
|
|
2121
|
+
if (error.code === PostgresDuplicateRelationError && error.message.includes(indexNameOptions.name)) {
|
|
2122
|
+
// Index already exists. Ignore error.
|
|
2422
2123
|
} else if (error.code === PostgresUniqueIndexViolationError && error.message.includes(indexNameOptions.name)) {
|
|
2423
2124
|
// Cast the error into the proper parse error
|
|
2424
2125
|
throw new _node.default.Error(_node.default.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');
|
|
@@ -2427,7 +2128,6 @@ class PostgresStorageAdapter {
|
|
|
2427
2128
|
}
|
|
2428
2129
|
});
|
|
2429
2130
|
}
|
|
2430
|
-
|
|
2431
2131
|
async deleteIdempotencyFunction(options = {}) {
|
|
2432
2132
|
const conn = options.conn !== undefined ? options.conn : this._client;
|
|
2433
2133
|
const qs = 'DROP FUNCTION IF EXISTS idempotency_delete_expired_records()';
|
|
@@ -2435,7 +2135,6 @@ class PostgresStorageAdapter {
|
|
|
2435
2135
|
throw error;
|
|
2436
2136
|
});
|
|
2437
2137
|
}
|
|
2438
|
-
|
|
2439
2138
|
async ensureIdempotencyFunctionExists(options = {}) {
|
|
2440
2139
|
const conn = options.conn !== undefined ? options.conn : this._client;
|
|
2441
2140
|
const ttlOptions = options.ttl !== undefined ? `${options.ttl} seconds` : '60 seconds';
|
|
@@ -2444,59 +2143,49 @@ class PostgresStorageAdapter {
|
|
|
2444
2143
|
throw error;
|
|
2445
2144
|
});
|
|
2446
2145
|
}
|
|
2447
|
-
|
|
2448
2146
|
}
|
|
2449
|
-
|
|
2450
2147
|
exports.PostgresStorageAdapter = PostgresStorageAdapter;
|
|
2451
|
-
|
|
2452
2148
|
function convertPolygonToSQL(polygon) {
|
|
2453
2149
|
if (polygon.length < 3) {
|
|
2454
2150
|
throw new _node.default.Error(_node.default.Error.INVALID_JSON, `Polygon must have at least 3 values`);
|
|
2455
2151
|
}
|
|
2456
|
-
|
|
2457
2152
|
if (polygon[0][0] !== polygon[polygon.length - 1][0] || polygon[0][1] !== polygon[polygon.length - 1][1]) {
|
|
2458
2153
|
polygon.push(polygon[0]);
|
|
2459
2154
|
}
|
|
2460
|
-
|
|
2461
2155
|
const unique = polygon.filter((item, index, ar) => {
|
|
2462
2156
|
let foundIndex = -1;
|
|
2463
|
-
|
|
2464
2157
|
for (let i = 0; i < ar.length; i += 1) {
|
|
2465
2158
|
const pt = ar[i];
|
|
2466
|
-
|
|
2467
2159
|
if (pt[0] === item[0] && pt[1] === item[1]) {
|
|
2468
2160
|
foundIndex = i;
|
|
2469
2161
|
break;
|
|
2470
2162
|
}
|
|
2471
2163
|
}
|
|
2472
|
-
|
|
2473
2164
|
return foundIndex === index;
|
|
2474
2165
|
});
|
|
2475
|
-
|
|
2476
2166
|
if (unique.length < 3) {
|
|
2477
2167
|
throw new _node.default.Error(_node.default.Error.INTERNAL_SERVER_ERROR, 'GeoJSON: Loop must have at least 3 different vertices');
|
|
2478
2168
|
}
|
|
2479
|
-
|
|
2480
2169
|
const points = polygon.map(point => {
|
|
2481
2170
|
_node.default.GeoPoint._validate(parseFloat(point[1]), parseFloat(point[0]));
|
|
2482
|
-
|
|
2483
2171
|
return `(${point[1]}, ${point[0]})`;
|
|
2484
2172
|
}).join(', ');
|
|
2485
2173
|
return `(${points})`;
|
|
2486
2174
|
}
|
|
2487
|
-
|
|
2488
2175
|
function removeWhiteSpace(regex) {
|
|
2489
2176
|
if (!regex.endsWith('\n')) {
|
|
2490
2177
|
regex += '\n';
|
|
2491
|
-
}
|
|
2492
|
-
|
|
2178
|
+
}
|
|
2493
2179
|
|
|
2494
|
-
|
|
2495
|
-
.replace(
|
|
2496
|
-
|
|
2180
|
+
// remove non escaped comments
|
|
2181
|
+
return regex.replace(/([^\\])#.*\n/gim, '$1')
|
|
2182
|
+
// remove lines starting with a comment
|
|
2183
|
+
.replace(/^#.*\n/gim, '')
|
|
2184
|
+
// remove non escaped whitespace
|
|
2185
|
+
.replace(/([^\\])\s+/gim, '$1')
|
|
2186
|
+
// remove whitespace at the beginning of a line
|
|
2497
2187
|
.replace(/^\s+/, '').trim();
|
|
2498
2188
|
}
|
|
2499
|
-
|
|
2500
2189
|
function processRegexPattern(s) {
|
|
2501
2190
|
if (s && s.startsWith('^')) {
|
|
2502
2191
|
// regex for startsWith
|
|
@@ -2504,92 +2193,76 @@ function processRegexPattern(s) {
|
|
|
2504
2193
|
} else if (s && s.endsWith('$')) {
|
|
2505
2194
|
// regex for endsWith
|
|
2506
2195
|
return literalizeRegexPart(s.slice(0, s.length - 1)) + '$';
|
|
2507
|
-
}
|
|
2508
|
-
|
|
2196
|
+
}
|
|
2509
2197
|
|
|
2198
|
+
// regex for contains
|
|
2510
2199
|
return literalizeRegexPart(s);
|
|
2511
2200
|
}
|
|
2512
|
-
|
|
2513
2201
|
function isStartsWithRegex(value) {
|
|
2514
2202
|
if (!value || typeof value !== 'string' || !value.startsWith('^')) {
|
|
2515
2203
|
return false;
|
|
2516
2204
|
}
|
|
2517
|
-
|
|
2518
2205
|
const matches = value.match(/\^\\Q.*\\E/);
|
|
2519
2206
|
return !!matches;
|
|
2520
2207
|
}
|
|
2521
|
-
|
|
2522
2208
|
function isAllValuesRegexOrNone(values) {
|
|
2523
2209
|
if (!values || !Array.isArray(values) || values.length === 0) {
|
|
2524
2210
|
return true;
|
|
2525
2211
|
}
|
|
2526
|
-
|
|
2527
2212
|
const firstValuesIsRegex = isStartsWithRegex(values[0].$regex);
|
|
2528
|
-
|
|
2529
2213
|
if (values.length === 1) {
|
|
2530
2214
|
return firstValuesIsRegex;
|
|
2531
2215
|
}
|
|
2532
|
-
|
|
2533
2216
|
for (let i = 1, length = values.length; i < length; ++i) {
|
|
2534
2217
|
if (firstValuesIsRegex !== isStartsWithRegex(values[i].$regex)) {
|
|
2535
2218
|
return false;
|
|
2536
2219
|
}
|
|
2537
2220
|
}
|
|
2538
|
-
|
|
2539
2221
|
return true;
|
|
2540
2222
|
}
|
|
2541
|
-
|
|
2542
2223
|
function isAnyValueRegexStartsWith(values) {
|
|
2543
2224
|
return values.some(function (value) {
|
|
2544
2225
|
return isStartsWithRegex(value.$regex);
|
|
2545
2226
|
});
|
|
2546
2227
|
}
|
|
2547
|
-
|
|
2548
2228
|
function createLiteralRegex(remaining) {
|
|
2549
2229
|
return remaining.split('').map(c => {
|
|
2550
2230
|
const regex = RegExp('[0-9 ]|\\p{L}', 'u'); // Support all unicode letter chars
|
|
2551
|
-
|
|
2552
2231
|
if (c.match(regex) !== null) {
|
|
2553
2232
|
// don't escape alphanumeric characters
|
|
2554
2233
|
return c;
|
|
2555
|
-
}
|
|
2556
|
-
|
|
2557
|
-
|
|
2234
|
+
}
|
|
2235
|
+
// escape everything else (single quotes with single quotes, everything else with a backslash)
|
|
2558
2236
|
return c === `'` ? `''` : `\\${c}`;
|
|
2559
2237
|
}).join('');
|
|
2560
2238
|
}
|
|
2561
|
-
|
|
2562
2239
|
function literalizeRegexPart(s) {
|
|
2563
2240
|
const matcher1 = /\\Q((?!\\E).*)\\E$/;
|
|
2564
2241
|
const result1 = s.match(matcher1);
|
|
2565
|
-
|
|
2566
2242
|
if (result1 && result1.length > 1 && result1.index > -1) {
|
|
2567
2243
|
// process regex that has a beginning and an end specified for the literal text
|
|
2568
2244
|
const prefix = s.substr(0, result1.index);
|
|
2569
2245
|
const remaining = result1[1];
|
|
2570
2246
|
return literalizeRegexPart(prefix) + createLiteralRegex(remaining);
|
|
2571
|
-
}
|
|
2572
|
-
|
|
2247
|
+
}
|
|
2573
2248
|
|
|
2249
|
+
// process regex that has a beginning specified for the literal text
|
|
2574
2250
|
const matcher2 = /\\Q((?!\\E).*)$/;
|
|
2575
2251
|
const result2 = s.match(matcher2);
|
|
2576
|
-
|
|
2577
2252
|
if (result2 && result2.length > 1 && result2.index > -1) {
|
|
2578
2253
|
const prefix = s.substr(0, result2.index);
|
|
2579
2254
|
const remaining = result2[1];
|
|
2580
2255
|
return literalizeRegexPart(prefix) + createLiteralRegex(remaining);
|
|
2581
|
-
}
|
|
2582
|
-
|
|
2256
|
+
}
|
|
2583
2257
|
|
|
2258
|
+
// remove all instances of \Q and \E from the remaining text & escape single quotes
|
|
2584
2259
|
return s.replace(/([^\\])(\\E)/, '$1').replace(/([^\\])(\\Q)/, '$1').replace(/^\\E/, '').replace(/^\\Q/, '').replace(/([^'])'/, `$1''`).replace(/^'([^'])/, `''$1`);
|
|
2585
2260
|
}
|
|
2586
|
-
|
|
2587
2261
|
var GeoPointCoder = {
|
|
2588
2262
|
isValidJSON(value) {
|
|
2589
2263
|
return typeof value === 'object' && value !== null && value.__type === 'GeoPoint';
|
|
2590
2264
|
}
|
|
2591
|
-
|
|
2592
2265
|
};
|
|
2593
2266
|
var _default = PostgresStorageAdapter;
|
|
2594
2267
|
exports.default = _default;
|
|
2595
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Adapters/Storage/Postgres/PostgresStorageAdapter.js"],"names":["Utils","require","PostgresRelationDoesNotExistError","PostgresDuplicateRelationError","PostgresDuplicateColumnError","PostgresMissingColumnError","PostgresUniqueIndexViolationError","logger","debug","args","arguments","concat","slice","length","log","getLogger","apply","parseTypeToPostgresType","type","contents","JSON","stringify","ParseToPosgresComparator","$gt","$lt","$gte","$lte","mongoAggregateToPostgres","$dayOfMonth","$dayOfWeek","$dayOfYear","$isoDayOfWeek","$isoWeekYear","$hour","$minute","$second","$millisecond","$month","$week","$year","toPostgresValue","value","__type","iso","name","toPostgresValueCastType","postgresValue","castType","undefined","transformValue","objectId","emptyCLPS","Object","freeze","find","get","count","create","update","delete","addField","protectedFields","defaultCLPS","toParseSchema","schema","className","fields","_hashed_password","_wperm","_rperm","clps","classLevelPermissions","indexes","toPostgresSchema","_password_history","handleDotFields","object","keys","forEach","fieldName","indexOf","components","split","first","shift","currentObj","next","__op","transformDotFieldToComponents","map","cmpt","index","transformDotField","join","transformAggregateField","substr","validateKeys","key","includes","Parse","Error","INVALID_NESTED_KEY","joinTablesForSchema","list","field","push","buildWhereClause","query","caseInsensitive","patterns","values","sorts","isArrayField","initialPatternsLength","fieldValue","$exists","authDataMatch","match","$in","$regex","MAX_INT_PLUS_ONE","clauses","clauseValues","subQuery","clause","pattern","orOrAnd","not","$ne","constraintFieldName","$relativeTime","INVALID_JSON","point","longitude","latitude","$eq","isInOrNin","Array","isArray","$nin","inPatterns","allowNull","listElem","listIndex","createConstraint","baseArray","notIn","_","flatMap","elt","$all","isAnyValueRegexStartsWith","isAllValuesRegexOrNone","i","processRegexPattern","substring","$containedBy","arr","$text","search","$search","language","$term","$language","$caseSensitive","$diacriticSensitive","$nearSphere","distance","$maxDistance","distanceInKM","$within","$box","box","left","bottom","right","top","$geoWithin","$centerSphere","centerSphere","GeoPoint","GeoPointCoder","isValidJSON","_validate","isNaN","$polygon","polygon","points","coordinates","$geoIntersects","$point","regex","operator","opts","$options","removeWhiteSpace","convertPolygonToSQL","cmp","pgComparator","parserResult","relativeTimeToDate","status","result","console","error","info","OPERATION_FORBIDDEN","PostgresStorageAdapter","constructor","uri","collectionPrefix","databaseOptions","_collectionPrefix","enableSchemaHooks","client","pgp","_client","_onchange","_pgp","_uuid","canSortOnJoinTables","watch","callback","createExplainableQuery","analyze","handleShutdown","_stream","done","$pool","end","_listenToSchema","connect","direct","on","data","payload","parse","senderId","none","_notifySchemaChange","catch","_ensureSchemaCollectionExists","conn","classExists","one","a","exists","setClassLevelPermissions","CLPs","task","t","setIndexesWithSchemaFormat","submittedIndexes","existingIndexes","self","Promise","resolve","_id_","_id","deletedIndexes","insertedIndexes","INVALID_QUERY","prototype","hasOwnProperty","call","tx","createIndexes","dropIndexes","createClass","parseSchema","createTable","err","code","detail","DUPLICATE_VALUE","valuesArray","patternsArray","assign","_email_verify_token_expires_at","_email_verify_token","_account_lockout_expires_at","_failed_login_count","_perishable_token","_perishable_token_expires_at","_password_changed_at","relations","parseType","qs","batch","joinTable","schemaUpgrade","columns","column_name","newColumns","filter","item","addFieldIfNotExists","postgresType","any","path","updateFieldOptions","deleteClass","operations","response","helpers","then","deleteAllClasses","now","Date","getTime","results","joins","reduce","classes","queries","deleteFields","fieldNames","idx","getAllClasses","row","getClass","createObject","transactionalSession","columnsArray","geoPoints","authDataAlreadyExists","authData","provider","pop","initialValues","val","termination","geoPointsInjects","l","columnsPattern","col","valuesPattern","promise","ops","underlyingError","constraint","matches","userInfo","duplicated_field","deleteObjectsByQuery","where","OBJECT_NOT_FOUND","findOneAndUpdate","updateObjectsByQuery","updatePatterns","originalUpdate","dotNotationOptions","generate","jsonb","lastKey","fieldNameIndex","str","amount","objects","keysToIncrement","k","incrementPatterns","c","keysToDelete","deletePatterns","p","updateObject","expectedType","reject","whereClause","upsertOneObject","createValue","skip","limit","sort","explain","hasLimit","hasSkip","wherePattern","limitPattern","skipPattern","sortPattern","sortCopy","sorting","transformKey","memo","originalQuery","postgresObjectToParseObject","targetClass","y","x","coords","parseFloat","createdAt","toISOString","updatedAt","expiresAt","ensureUniqueness","constraintName","constraintPatterns","message","readPreference","estimate","approximate_row_count","distinct","column","isNested","isPointerField","transformer","child","aggregate","pipeline","hint","countField","groupValues","groupPattern","stage","$group","groupByFields","alias","source","operation","$sum","$max","$min","$avg","$project","$match","$or","collapse","element","matchPatterns","$limit","$skip","$sort","order","e","trim","Boolean","parseInt","performInitialization","VolatileClassesSchemas","promises","INVALID_CLASS_NAME","all","sql","misc","jsonObjectSetKeys","array","add","addUnique","remove","containsAll","containsAllRegex","contains","ctx","duration","createIndexesIfNeeded","getIndexes","updateSchemaWithIndexes","updateEstimatedCount","createTransactionalSession","commitTransactionalSession","abortTransactionalSession","ensureIndex","indexName","options","defaultIndexName","indexNameOptions","setIdempotencyFunction","ensureIdempotencyFunctionExists","deleteIdempotencyFunction","ttlOptions","ttl","unique","ar","foundIndex","pt","INTERNAL_SERVER_ERROR","endsWith","replace","s","startsWith","literalizeRegexPart","isStartsWithRegex","firstValuesIsRegex","some","createLiteralRegex","remaining","RegExp","matcher1","result1","prefix","matcher2","result2"],"mappings":"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","sourcesContent":["// @flow\nimport { createClient } from './PostgresClient';\n// @flow-disable-next\nimport Parse from 'parse/node';\n// @flow-disable-next\nimport _ from 'lodash';\n// @flow-disable-next\nimport { v4 as uuidv4 } from 'uuid';\nimport sql from './sql';\nimport { StorageAdapter } from '../StorageAdapter';\nimport type { SchemaType, QueryType, QueryOptions } from '../StorageAdapter';\nconst Utils = require('../../../Utils');\n\nconst PostgresRelationDoesNotExistError = '42P01';\nconst PostgresDuplicateRelationError = '42P07';\nconst PostgresDuplicateColumnError = '42701';\nconst PostgresMissingColumnError = '42703';\nconst PostgresUniqueIndexViolationError = '23505';\nconst logger = require('../../../logger');\n\nconst debug = function (...args: any) {\n  args = ['PG: ' + arguments[0]].concat(args.slice(1, args.length));\n  const log = logger.getLogger();\n  log.debug.apply(log, args);\n};\n\nconst parseTypeToPostgresType = type => {\n  switch (type.type) {\n    case 'String':\n      return 'text';\n    case 'Date':\n      return 'timestamp with time zone';\n    case 'Object':\n      return 'jsonb';\n    case 'File':\n      return 'text';\n    case 'Boolean':\n      return 'boolean';\n    case 'Pointer':\n      return 'text';\n    case 'Number':\n      return 'double precision';\n    case 'GeoPoint':\n      return 'point';\n    case 'Bytes':\n      return 'jsonb';\n    case 'Polygon':\n      return 'polygon';\n    case 'Array':\n      if (type.contents && type.contents.type === 'String') {\n        return 'text[]';\n      } else {\n        return 'jsonb';\n      }\n    default:\n      throw `no type for ${JSON.stringify(type)} yet`;\n  }\n};\n\nconst ParseToPosgresComparator = {\n  $gt: '>',\n  $lt: '<',\n  $gte: '>=',\n  $lte: '<=',\n};\n\nconst mongoAggregateToPostgres = {\n  $dayOfMonth: 'DAY',\n  $dayOfWeek: 'DOW',\n  $dayOfYear: 'DOY',\n  $isoDayOfWeek: 'ISODOW',\n  $isoWeekYear: 'ISOYEAR',\n  $hour: 'HOUR',\n  $minute: 'MINUTE',\n  $second: 'SECOND',\n  $millisecond: 'MILLISECONDS',\n  $month: 'MONTH',\n  $week: 'WEEK',\n  $year: 'YEAR',\n};\n\nconst toPostgresValue = value => {\n  if (typeof value === 'object') {\n    if (value.__type === 'Date') {\n      return value.iso;\n    }\n    if (value.__type === 'File') {\n      return value.name;\n    }\n  }\n  return value;\n};\n\nconst toPostgresValueCastType = value => {\n  const postgresValue = toPostgresValue(value);\n  let castType;\n  switch (typeof postgresValue) {\n    case 'number':\n      castType = 'double precision';\n      break;\n    case 'boolean':\n      castType = 'boolean';\n      break;\n    default:\n      castType = undefined;\n  }\n  return castType;\n};\n\nconst transformValue = value => {\n  if (typeof value === 'object' && value.__type === 'Pointer') {\n    return value.objectId;\n  }\n  return value;\n};\n\n// Duplicate from then mongo adapter...\nconst emptyCLPS = Object.freeze({\n  find: {},\n  get: {},\n  count: {},\n  create: {},\n  update: {},\n  delete: {},\n  addField: {},\n  protectedFields: {},\n});\n\nconst defaultCLPS = Object.freeze({\n  find: { '*': true },\n  get: { '*': true },\n  count: { '*': true },\n  create: { '*': true },\n  update: { '*': true },\n  delete: { '*': true },\n  addField: { '*': true },\n  protectedFields: { '*': [] },\n});\n\nconst toParseSchema = schema => {\n  if (schema.className === '_User') {\n    delete schema.fields._hashed_password;\n  }\n  if (schema.fields) {\n    delete schema.fields._wperm;\n    delete schema.fields._rperm;\n  }\n  let clps = defaultCLPS;\n  if (schema.classLevelPermissions) {\n    clps = { ...emptyCLPS, ...schema.classLevelPermissions };\n  }\n  let indexes = {};\n  if (schema.indexes) {\n    indexes = { ...schema.indexes };\n  }\n  return {\n    className: schema.className,\n    fields: schema.fields,\n    classLevelPermissions: clps,\n    indexes,\n  };\n};\n\nconst toPostgresSchema = schema => {\n  if (!schema) {\n    return schema;\n  }\n  schema.fields = schema.fields || {};\n  schema.fields._wperm = { type: 'Array', contents: { type: 'String' } };\n  schema.fields._rperm = { type: 'Array', contents: { type: 'String' } };\n  if (schema.className === '_User') {\n    schema.fields._hashed_password = { type: 'String' };\n    schema.fields._password_history = { type: 'Array' };\n  }\n  return schema;\n};\n\nconst handleDotFields = object => {\n  Object.keys(object).forEach(fieldName => {\n    if (fieldName.indexOf('.') > -1) {\n      const components = fieldName.split('.');\n      const first = components.shift();\n      object[first] = object[first] || {};\n      let currentObj = object[first];\n      let next;\n      let value = object[fieldName];\n      if (value && value.__op === 'Delete') {\n        value = undefined;\n      }\n      /* eslint-disable no-cond-assign */\n      while ((next = components.shift())) {\n        /* eslint-enable no-cond-assign */\n        currentObj[next] = currentObj[next] || {};\n        if (components.length === 0) {\n          currentObj[next] = value;\n        }\n        currentObj = currentObj[next];\n      }\n      delete object[fieldName];\n    }\n  });\n  return object;\n};\n\nconst transformDotFieldToComponents = fieldName => {\n  return fieldName.split('.').map((cmpt, index) => {\n    if (index === 0) {\n      return `\"${cmpt}\"`;\n    }\n    return `'${cmpt}'`;\n  });\n};\n\nconst transformDotField = fieldName => {\n  if (fieldName.indexOf('.') === -1) {\n    return `\"${fieldName}\"`;\n  }\n  const components = transformDotFieldToComponents(fieldName);\n  let name = components.slice(0, components.length - 1).join('->');\n  name += '->>' + components[components.length - 1];\n  return name;\n};\n\nconst transformAggregateField = fieldName => {\n  if (typeof fieldName !== 'string') {\n    return fieldName;\n  }\n  if (fieldName === '$_created_at') {\n    return 'createdAt';\n  }\n  if (fieldName === '$_updated_at') {\n    return 'updatedAt';\n  }\n  return fieldName.substr(1);\n};\n\nconst validateKeys = object => {\n  if (typeof object == 'object') {\n    for (const key in object) {\n      if (typeof object[key] == 'object') {\n        validateKeys(object[key]);\n      }\n\n      if (key.includes('$') || key.includes('.')) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_NESTED_KEY,\n          \"Nested keys should not contain the '$' or '.' characters\"\n        );\n      }\n    }\n  }\n};\n\n// Returns the list of join tables on a schema\nconst joinTablesForSchema = schema => {\n  const list = [];\n  if (schema) {\n    Object.keys(schema.fields).forEach(field => {\n      if (schema.fields[field].type === 'Relation') {\n        list.push(`_Join:${field}:${schema.className}`);\n      }\n    });\n  }\n  return list;\n};\n\ninterface WhereClause {\n  pattern: string;\n  values: Array<any>;\n  sorts: Array<any>;\n}\n\nconst buildWhereClause = ({ schema, query, index, caseInsensitive }): WhereClause => {\n  const patterns = [];\n  let values = [];\n  const sorts = [];\n\n  schema = toPostgresSchema(schema);\n  for (const fieldName in query) {\n    const isArrayField =\n      schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array';\n    const initialPatternsLength = patterns.length;\n    const fieldValue = query[fieldName];\n\n    // nothing in the schema, it's gonna blow up\n    if (!schema.fields[fieldName]) {\n      // as it won't exist\n      if (fieldValue && fieldValue.$exists === false) {\n        continue;\n      }\n    }\n    const authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);\n    if (authDataMatch) {\n      // TODO: Handle querying by _auth_data_provider, authData is stored in authData field\n      continue;\n    } else if (caseInsensitive && (fieldName === 'username' || fieldName === 'email')) {\n      patterns.push(`LOWER($${index}:name) = LOWER($${index + 1})`);\n      values.push(fieldName, fieldValue);\n      index += 2;\n    } else if (fieldName.indexOf('.') >= 0) {\n      let name = transformDotField(fieldName);\n      if (fieldValue === null) {\n        patterns.push(`$${index}:raw IS NULL`);\n        values.push(name);\n        index += 1;\n        continue;\n      } else {\n        if (fieldValue.$in) {\n          name = transformDotFieldToComponents(fieldName).join('->');\n          patterns.push(`($${index}:raw)::jsonb @> $${index + 1}::jsonb`);\n          values.push(name, JSON.stringify(fieldValue.$in));\n          index += 2;\n        } else if (fieldValue.$regex) {\n          // Handle later\n        } else if (typeof fieldValue !== 'object') {\n          patterns.push(`$${index}:raw = $${index + 1}::text`);\n          values.push(name, fieldValue);\n          index += 2;\n        }\n      }\n    } else if (fieldValue === null || fieldValue === undefined) {\n      patterns.push(`$${index}:name IS NULL`);\n      values.push(fieldName);\n      index += 1;\n      continue;\n    } else if (typeof fieldValue === 'string') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      values.push(fieldName, fieldValue);\n      index += 2;\n    } else if (typeof fieldValue === 'boolean') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      // Can't cast boolean to double precision\n      if (schema.fields[fieldName] && schema.fields[fieldName].type === 'Number') {\n        // Should always return zero results\n        const MAX_INT_PLUS_ONE = 9223372036854775808;\n        values.push(fieldName, MAX_INT_PLUS_ONE);\n      } else {\n        values.push(fieldName, fieldValue);\n      }\n      index += 2;\n    } else if (typeof fieldValue === 'number') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      values.push(fieldName, fieldValue);\n      index += 2;\n    } else if (['$or', '$nor', '$and'].includes(fieldName)) {\n      const clauses = [];\n      const clauseValues = [];\n      fieldValue.forEach(subQuery => {\n        const clause = buildWhereClause({\n          schema,\n          query: subQuery,\n          index,\n          caseInsensitive,\n        });\n        if (clause.pattern.length > 0) {\n          clauses.push(clause.pattern);\n          clauseValues.push(...clause.values);\n          index += clause.values.length;\n        }\n      });\n\n      const orOrAnd = fieldName === '$and' ? ' AND ' : ' OR ';\n      const not = fieldName === '$nor' ? ' NOT ' : '';\n\n      patterns.push(`${not}(${clauses.join(orOrAnd)})`);\n      values.push(...clauseValues);\n    }\n\n    if (fieldValue.$ne !== undefined) {\n      if (isArrayField) {\n        fieldValue.$ne = JSON.stringify([fieldValue.$ne]);\n        patterns.push(`NOT array_contains($${index}:name, $${index + 1})`);\n      } else {\n        if (fieldValue.$ne === null) {\n          patterns.push(`$${index}:name IS NOT NULL`);\n          values.push(fieldName);\n          index += 1;\n          continue;\n        } else {\n          // if not null, we need to manually exclude null\n          if (fieldValue.$ne.__type === 'GeoPoint') {\n            patterns.push(\n              `($${index}:name <> POINT($${index + 1}, $${index + 2}) OR $${index}:name IS NULL)`\n            );\n          } else {\n            if (fieldName.indexOf('.') >= 0) {\n              const castType = toPostgresValueCastType(fieldValue.$ne);\n              const constraintFieldName = castType\n                ? `CAST ((${transformDotField(fieldName)}) AS ${castType})`\n                : transformDotField(fieldName);\n              patterns.push(\n                `(${constraintFieldName} <> $${index + 1} OR ${constraintFieldName} IS NULL)`\n              );\n            } else if (typeof fieldValue.$ne === 'object' && fieldValue.$ne.$relativeTime) {\n              throw new Parse.Error(\n                Parse.Error.INVALID_JSON,\n                '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators'\n              );\n            } else {\n              patterns.push(`($${index}:name <> $${index + 1} OR $${index}:name IS NULL)`);\n            }\n          }\n        }\n      }\n      if (fieldValue.$ne.__type === 'GeoPoint') {\n        const point = fieldValue.$ne;\n        values.push(fieldName, point.longitude, point.latitude);\n        index += 3;\n      } else {\n        // TODO: support arrays\n        values.push(fieldName, fieldValue.$ne);\n        index += 2;\n      }\n    }\n    if (fieldValue.$eq !== undefined) {\n      if (fieldValue.$eq === null) {\n        patterns.push(`$${index}:name IS NULL`);\n        values.push(fieldName);\n        index += 1;\n      } else {\n        if (fieldName.indexOf('.') >= 0) {\n          const castType = toPostgresValueCastType(fieldValue.$eq);\n          const constraintFieldName = castType\n            ? `CAST ((${transformDotField(fieldName)}) AS ${castType})`\n            : transformDotField(fieldName);\n          values.push(fieldValue.$eq);\n          patterns.push(`${constraintFieldName} = $${index++}`);\n        } else if (typeof fieldValue.$eq === 'object' && fieldValue.$eq.$relativeTime) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators'\n          );\n        } else {\n          values.push(fieldName, fieldValue.$eq);\n          patterns.push(`$${index}:name = $${index + 1}`);\n          index += 2;\n        }\n      }\n    }\n    const isInOrNin = Array.isArray(fieldValue.$in) || Array.isArray(fieldValue.$nin);\n    if (\n      Array.isArray(fieldValue.$in) &&\n      isArrayField &&\n      schema.fields[fieldName].contents &&\n      schema.fields[fieldName].contents.type === 'String'\n    ) {\n      const inPatterns = [];\n      let allowNull = false;\n      values.push(fieldName);\n      fieldValue.$in.forEach((listElem, listIndex) => {\n        if (listElem === null) {\n          allowNull = true;\n        } else {\n          values.push(listElem);\n          inPatterns.push(`$${index + 1 + listIndex - (allowNull ? 1 : 0)}`);\n        }\n      });\n      if (allowNull) {\n        patterns.push(`($${index}:name IS NULL OR $${index}:name && ARRAY[${inPatterns.join()}])`);\n      } else {\n        patterns.push(`$${index}:name && ARRAY[${inPatterns.join()}]`);\n      }\n      index = index + 1 + inPatterns.length;\n    } else if (isInOrNin) {\n      var createConstraint = (baseArray, notIn) => {\n        const not = notIn ? ' NOT ' : '';\n        if (baseArray.length > 0) {\n          if (isArrayField) {\n            patterns.push(`${not} array_contains($${index}:name, $${index + 1})`);\n            values.push(fieldName, JSON.stringify(baseArray));\n            index += 2;\n          } else {\n            // Handle Nested Dot Notation Above\n            if (fieldName.indexOf('.') >= 0) {\n              return;\n            }\n            const inPatterns = [];\n            values.push(fieldName);\n            baseArray.forEach((listElem, listIndex) => {\n              if (listElem != null) {\n                values.push(listElem);\n                inPatterns.push(`$${index + 1 + listIndex}`);\n              }\n            });\n            patterns.push(`$${index}:name ${not} IN (${inPatterns.join()})`);\n            index = index + 1 + inPatterns.length;\n          }\n        } else if (!notIn) {\n          values.push(fieldName);\n          patterns.push(`$${index}:name IS NULL`);\n          index = index + 1;\n        } else {\n          // Handle empty array\n          if (notIn) {\n            patterns.push('1 = 1'); // Return all values\n          } else {\n            patterns.push('1 = 2'); // Return no values\n          }\n        }\n      };\n      if (fieldValue.$in) {\n        createConstraint(\n          _.flatMap(fieldValue.$in, elt => elt),\n          false\n        );\n      }\n      if (fieldValue.$nin) {\n        createConstraint(\n          _.flatMap(fieldValue.$nin, elt => elt),\n          true\n        );\n      }\n    } else if (typeof fieldValue.$in !== 'undefined') {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad $in value');\n    } else if (typeof fieldValue.$nin !== 'undefined') {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad $nin value');\n    }\n\n    if (Array.isArray(fieldValue.$all) && isArrayField) {\n      if (isAnyValueRegexStartsWith(fieldValue.$all)) {\n        if (!isAllValuesRegexOrNone(fieldValue.$all)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            'All $all values must be of regex type or none: ' + fieldValue.$all\n          );\n        }\n\n        for (let i = 0; i < fieldValue.$all.length; i += 1) {\n          const value = processRegexPattern(fieldValue.$all[i].$regex);\n          fieldValue.$all[i] = value.substring(1) + '%';\n        }\n        patterns.push(`array_contains_all_regex($${index}:name, $${index + 1}::jsonb)`);\n      } else {\n        patterns.push(`array_contains_all($${index}:name, $${index + 1}::jsonb)`);\n      }\n      values.push(fieldName, JSON.stringify(fieldValue.$all));\n      index += 2;\n    } else if (Array.isArray(fieldValue.$all)) {\n      if (fieldValue.$all.length === 1) {\n        patterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue.$all[0].objectId);\n        index += 2;\n      }\n    }\n\n    if (typeof fieldValue.$exists !== 'undefined') {\n      if (typeof fieldValue.$exists === 'object' && fieldValue.$exists.$relativeTime) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators'\n        );\n      } else if (fieldValue.$exists) {\n        patterns.push(`$${index}:name IS NOT NULL`);\n      } else {\n        patterns.push(`$${index}:name IS NULL`);\n      }\n      values.push(fieldName);\n      index += 1;\n    }\n\n    if (fieldValue.$containedBy) {\n      const arr = fieldValue.$containedBy;\n      if (!(arr instanceof Array)) {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $containedBy: should be an array`);\n      }\n\n      patterns.push(`$${index}:name <@ $${index + 1}::jsonb`);\n      values.push(fieldName, JSON.stringify(arr));\n      index += 2;\n    }\n\n    if (fieldValue.$text) {\n      const search = fieldValue.$text.$search;\n      let language = 'english';\n      if (typeof search !== 'object') {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $text: $search, should be object`);\n      }\n      if (!search.$term || typeof search.$term !== 'string') {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $text: $term, should be string`);\n      }\n      if (search.$language && typeof search.$language !== 'string') {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $text: $language, should be string`);\n      } else if (search.$language) {\n        language = search.$language;\n      }\n      if (search.$caseSensitive && typeof search.$caseSensitive !== 'boolean') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $caseSensitive, should be boolean`\n        );\n      } else if (search.$caseSensitive) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $caseSensitive not supported, please use $regex or create a separate lower case column.`\n        );\n      }\n      if (search.$diacriticSensitive && typeof search.$diacriticSensitive !== 'boolean') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $diacriticSensitive, should be boolean`\n        );\n      } else if (search.$diacriticSensitive === false) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $diacriticSensitive - false not supported, install Postgres Unaccent Extension`\n        );\n      }\n      patterns.push(\n        `to_tsvector($${index}, $${index + 1}:name) @@ to_tsquery($${index + 2}, $${index + 3})`\n      );\n      values.push(language, fieldName, language, search.$term);\n      index += 4;\n    }\n\n    if (fieldValue.$nearSphere) {\n      const point = fieldValue.$nearSphere;\n      const distance = fieldValue.$maxDistance;\n      const distanceInKM = distance * 6371 * 1000;\n      patterns.push(\n        `ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${\n          index + 2\n        })::geometry) <= $${index + 3}`\n      );\n      sorts.push(\n        `ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${\n          index + 2\n        })::geometry) ASC`\n      );\n      values.push(fieldName, point.longitude, point.latitude, distanceInKM);\n      index += 4;\n    }\n\n    if (fieldValue.$within && fieldValue.$within.$box) {\n      const box = fieldValue.$within.$box;\n      const left = box[0].longitude;\n      const bottom = box[0].latitude;\n      const right = box[1].longitude;\n      const top = box[1].latitude;\n\n      patterns.push(`$${index}:name::point <@ $${index + 1}::box`);\n      values.push(fieldName, `((${left}, ${bottom}), (${right}, ${top}))`);\n      index += 2;\n    }\n\n    if (fieldValue.$geoWithin && fieldValue.$geoWithin.$centerSphere) {\n      const centerSphere = fieldValue.$geoWithin.$centerSphere;\n      if (!(centerSphere instanceof Array) || centerSphere.length < 2) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoWithin value; $centerSphere should be an array of Parse.GeoPoint and distance'\n        );\n      }\n      // Get point, convert to geo point if necessary and validate\n      let point = centerSphere[0];\n      if (point instanceof Array && point.length === 2) {\n        point = new Parse.GeoPoint(point[1], point[0]);\n      } else if (!GeoPointCoder.isValidJSON(point)) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoWithin value; $centerSphere geo point invalid'\n        );\n      }\n      Parse.GeoPoint._validate(point.latitude, point.longitude);\n      // Get distance and validate\n      const distance = centerSphere[1];\n      if (isNaN(distance) || distance < 0) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoWithin value; $centerSphere distance invalid'\n        );\n      }\n      const distanceInKM = distance * 6371 * 1000;\n      patterns.push(\n        `ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${\n          index + 2\n        })::geometry) <= $${index + 3}`\n      );\n      values.push(fieldName, point.longitude, point.latitude, distanceInKM);\n      index += 4;\n    }\n\n    if (fieldValue.$geoWithin && fieldValue.$geoWithin.$polygon) {\n      const polygon = fieldValue.$geoWithin.$polygon;\n      let points;\n      if (typeof polygon === 'object' && polygon.__type === 'Polygon') {\n        if (!polygon.coordinates || polygon.coordinates.length < 3) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            'bad $geoWithin value; Polygon.coordinates should contain at least 3 lon/lat pairs'\n          );\n        }\n        points = polygon.coordinates;\n      } else if (polygon instanceof Array) {\n        if (polygon.length < 3) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            'bad $geoWithin value; $polygon should contain at least 3 GeoPoints'\n          );\n        }\n        points = polygon;\n      } else {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          \"bad $geoWithin value; $polygon should be Polygon object or Array of Parse.GeoPoint's\"\n        );\n      }\n      points = points\n        .map(point => {\n          if (point instanceof Array && point.length === 2) {\n            Parse.GeoPoint._validate(point[1], point[0]);\n            return `(${point[0]}, ${point[1]})`;\n          }\n          if (typeof point !== 'object' || point.__type !== 'GeoPoint') {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad $geoWithin value');\n          } else {\n            Parse.GeoPoint._validate(point.latitude, point.longitude);\n          }\n          return `(${point.longitude}, ${point.latitude})`;\n        })\n        .join(', ');\n\n      patterns.push(`$${index}:name::point <@ $${index + 1}::polygon`);\n      values.push(fieldName, `(${points})`);\n      index += 2;\n    }\n    if (fieldValue.$geoIntersects && fieldValue.$geoIntersects.$point) {\n      const point = fieldValue.$geoIntersects.$point;\n      if (typeof point !== 'object' || point.__type !== 'GeoPoint') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoIntersect value; $point should be GeoPoint'\n        );\n      } else {\n        Parse.GeoPoint._validate(point.latitude, point.longitude);\n      }\n      patterns.push(`$${index}:name::polygon @> $${index + 1}::point`);\n      values.push(fieldName, `(${point.longitude}, ${point.latitude})`);\n      index += 2;\n    }\n\n    if (fieldValue.$regex) {\n      let regex = fieldValue.$regex;\n      let operator = '~';\n      const opts = fieldValue.$options;\n      if (opts) {\n        if (opts.indexOf('i') >= 0) {\n          operator = '~*';\n        }\n        if (opts.indexOf('x') >= 0) {\n          regex = removeWhiteSpace(regex);\n        }\n      }\n\n      const name = transformDotField(fieldName);\n      regex = processRegexPattern(regex);\n\n      patterns.push(`$${index}:raw ${operator} '$${index + 1}:raw'`);\n      values.push(name, regex);\n      index += 2;\n    }\n\n    if (fieldValue.__type === 'Pointer') {\n      if (isArrayField) {\n        patterns.push(`array_contains($${index}:name, $${index + 1})`);\n        values.push(fieldName, JSON.stringify([fieldValue]));\n        index += 2;\n      } else {\n        patterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue.objectId);\n        index += 2;\n      }\n    }\n\n    if (fieldValue.__type === 'Date') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      values.push(fieldName, fieldValue.iso);\n      index += 2;\n    }\n\n    if (fieldValue.__type === 'GeoPoint') {\n      patterns.push(`$${index}:name ~= POINT($${index + 1}, $${index + 2})`);\n      values.push(fieldName, fieldValue.longitude, fieldValue.latitude);\n      index += 3;\n    }\n\n    if (fieldValue.__type === 'Polygon') {\n      const value = convertPolygonToSQL(fieldValue.coordinates);\n      patterns.push(`$${index}:name ~= $${index + 1}::polygon`);\n      values.push(fieldName, value);\n      index += 2;\n    }\n\n    Object.keys(ParseToPosgresComparator).forEach(cmp => {\n      if (fieldValue[cmp] || fieldValue[cmp] === 0) {\n        const pgComparator = ParseToPosgresComparator[cmp];\n        let constraintFieldName;\n        let postgresValue = toPostgresValue(fieldValue[cmp]);\n\n        if (fieldName.indexOf('.') >= 0) {\n          const castType = toPostgresValueCastType(fieldValue[cmp]);\n          constraintFieldName = castType\n            ? `CAST ((${transformDotField(fieldName)}) AS ${castType})`\n            : transformDotField(fieldName);\n        } else {\n          if (typeof postgresValue === 'object' && postgresValue.$relativeTime) {\n            if (schema.fields[fieldName].type !== 'Date') {\n              throw new Parse.Error(\n                Parse.Error.INVALID_JSON,\n                '$relativeTime can only be used with Date field'\n              );\n            }\n            const parserResult = Utils.relativeTimeToDate(postgresValue.$relativeTime);\n            if (parserResult.status === 'success') {\n              postgresValue = toPostgresValue(parserResult.result);\n            } else {\n              console.error('Error while parsing relative date', parserResult);\n              throw new Parse.Error(\n                Parse.Error.INVALID_JSON,\n                `bad $relativeTime (${postgresValue.$relativeTime}) value. ${parserResult.info}`\n              );\n            }\n          }\n          constraintFieldName = `$${index++}:name`;\n          values.push(fieldName);\n        }\n        values.push(postgresValue);\n        patterns.push(`${constraintFieldName} ${pgComparator} $${index++}`);\n      }\n    });\n\n    if (initialPatternsLength === patterns.length) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `Postgres doesn't support this query type yet ${JSON.stringify(fieldValue)}`\n      );\n    }\n  }\n  values = values.map(transformValue);\n  return { pattern: patterns.join(' AND '), values, sorts };\n};\n\nexport class PostgresStorageAdapter implements StorageAdapter {\n  canSortOnJoinTables: boolean;\n  enableSchemaHooks: boolean;\n\n  // Private\n  _collectionPrefix: string;\n  _client: any;\n  _onchange: any;\n  _pgp: any;\n  _stream: any;\n  _uuid: any;\n\n  constructor({ uri, collectionPrefix = '', databaseOptions = {} }: any) {\n    this._collectionPrefix = collectionPrefix;\n    this.enableSchemaHooks = !!databaseOptions.enableSchemaHooks;\n    delete databaseOptions.enableSchemaHooks;\n\n    const { client, pgp } = createClient(uri, databaseOptions);\n    this._client = client;\n    this._onchange = () => {};\n    this._pgp = pgp;\n    this._uuid = uuidv4();\n    this.canSortOnJoinTables = false;\n  }\n\n  watch(callback: () => void): void {\n    this._onchange = callback;\n  }\n\n  //Note that analyze=true will run the query, executing INSERTS, DELETES, etc.\n  createExplainableQuery(query: string, analyze: boolean = false) {\n    if (analyze) {\n      return 'EXPLAIN (ANALYZE, FORMAT JSON) ' + query;\n    } else {\n      return 'EXPLAIN (FORMAT JSON) ' + query;\n    }\n  }\n\n  handleShutdown() {\n    if (this._stream) {\n      this._stream.done();\n      delete this._stream;\n    }\n    if (!this._client) {\n      return;\n    }\n    this._client.$pool.end();\n  }\n\n  async _listenToSchema() {\n    if (!this._stream && this.enableSchemaHooks) {\n      this._stream = await this._client.connect({ direct: true });\n      this._stream.client.on('notification', data => {\n        const payload = JSON.parse(data.payload);\n        if (payload.senderId !== this._uuid) {\n          this._onchange();\n        }\n      });\n      await this._stream.none('LISTEN $1~', 'schema.change');\n    }\n  }\n\n  _notifySchemaChange() {\n    if (this._stream) {\n      this._stream\n        .none('NOTIFY $1~, $2', ['schema.change', { senderId: this._uuid }])\n        .catch(error => {\n          console.log('Failed to Notify:', error); // unlikely to ever happen\n        });\n    }\n  }\n\n  async _ensureSchemaCollectionExists(conn: any) {\n    conn = conn || this._client;\n    await conn\n      .none(\n        'CREATE TABLE IF NOT EXISTS \"_SCHEMA\" ( \"className\" varChar(120), \"schema\" jsonb, \"isParseClass\" bool, PRIMARY KEY (\"className\") )'\n      )\n      .catch(error => {\n        throw error;\n      });\n  }\n\n  async classExists(name: string) {\n    return this._client.one(\n      'SELECT EXISTS (SELECT 1 FROM information_schema.tables WHERE table_name = $1)',\n      [name],\n      a => a.exists\n    );\n  }\n\n  async setClassLevelPermissions(className: string, CLPs: any) {\n    await this._client.task('set-class-level-permissions', async t => {\n      const values = [className, 'schema', 'classLevelPermissions', JSON.stringify(CLPs)];\n      await t.none(\n        `UPDATE \"_SCHEMA\" SET $2:name = json_object_set_key($2:name, $3::text, $4::jsonb) WHERE \"className\" = $1`,\n        values\n      );\n    });\n    this._notifySchemaChange();\n  }\n\n  async setIndexesWithSchemaFormat(\n    className: string,\n    submittedIndexes: any,\n    existingIndexes: any = {},\n    fields: any,\n    conn: ?any\n  ): Promise<void> {\n    conn = conn || this._client;\n    const self = this;\n    if (submittedIndexes === undefined) {\n      return Promise.resolve();\n    }\n    if (Object.keys(existingIndexes).length === 0) {\n      existingIndexes = { _id_: { _id: 1 } };\n    }\n    const deletedIndexes = [];\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        deletedIndexes.push(name);\n        delete existingIndexes[name];\n      } else {\n        Object.keys(field).forEach(key => {\n          if (!Object.prototype.hasOwnProperty.call(fields, key)) {\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    await conn.tx('set-indexes-with-schema-format', async t => {\n      if (insertedIndexes.length > 0) {\n        await self.createIndexes(className, insertedIndexes, t);\n      }\n      if (deletedIndexes.length > 0) {\n        await self.dropIndexes(className, deletedIndexes, t);\n      }\n      await t.none(\n        'UPDATE \"_SCHEMA\" SET $2:name = json_object_set_key($2:name, $3::text, $4::jsonb) WHERE \"className\" = $1',\n        [className, 'schema', 'indexes', JSON.stringify(existingIndexes)]\n      );\n    });\n    this._notifySchemaChange();\n  }\n\n  async createClass(className: string, schema: SchemaType, conn: ?any) {\n    conn = conn || this._client;\n    const parseSchema = await conn\n      .tx('create-class', async t => {\n        await this.createTable(className, schema, t);\n        await t.none(\n          'INSERT INTO \"_SCHEMA\" (\"className\", \"schema\", \"isParseClass\") VALUES ($<className>, $<schema>, true)',\n          { className, schema }\n        );\n        await this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields, t);\n        return toParseSchema(schema);\n      })\n      .catch(err => {\n        if (err.code === PostgresUniqueIndexViolationError && err.detail.includes(className)) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, `Class ${className} already exists.`);\n        }\n        throw err;\n      });\n    this._notifySchemaChange();\n    return parseSchema;\n  }\n\n  // Just create a table, do not insert in schema\n  async createTable(className: string, schema: SchemaType, conn: any) {\n    conn = conn || this._client;\n    debug('createTable');\n    const valuesArray = [];\n    const patternsArray = [];\n    const fields = Object.assign({}, schema.fields);\n    if (className === '_User') {\n      fields._email_verify_token_expires_at = { type: 'Date' };\n      fields._email_verify_token = { type: 'String' };\n      fields._account_lockout_expires_at = { type: 'Date' };\n      fields._failed_login_count = { type: 'Number' };\n      fields._perishable_token = { type: 'String' };\n      fields._perishable_token_expires_at = { type: 'Date' };\n      fields._password_changed_at = { type: 'Date' };\n      fields._password_history = { type: 'Array' };\n    }\n    let index = 2;\n    const relations = [];\n    Object.keys(fields).forEach(fieldName => {\n      const parseType = fields[fieldName];\n      // Skip when it's a relation\n      // We'll create the tables later\n      if (parseType.type === 'Relation') {\n        relations.push(fieldName);\n        return;\n      }\n      if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {\n        parseType.contents = { type: 'String' };\n      }\n      valuesArray.push(fieldName);\n      valuesArray.push(parseTypeToPostgresType(parseType));\n      patternsArray.push(`$${index}:name $${index + 1}:raw`);\n      if (fieldName === 'objectId') {\n        patternsArray.push(`PRIMARY KEY ($${index}:name)`);\n      }\n      index = index + 2;\n    });\n    const qs = `CREATE TABLE IF NOT EXISTS $1:name (${patternsArray.join()})`;\n    const values = [className, ...valuesArray];\n\n    return conn.task('create-table', async t => {\n      try {\n        await t.none(qs, values);\n      } catch (error) {\n        if (error.code !== PostgresDuplicateRelationError) {\n          throw error;\n        }\n        // ELSE: Table already exists, must have been created by a different request. Ignore the error.\n      }\n      await t.tx('create-table-tx', tx => {\n        return tx.batch(\n          relations.map(fieldName => {\n            return tx.none(\n              'CREATE TABLE IF NOT EXISTS $<joinTable:name> (\"relatedId\" varChar(120), \"owningId\" varChar(120), PRIMARY KEY(\"relatedId\", \"owningId\") )',\n              { joinTable: `_Join:${fieldName}:${className}` }\n            );\n          })\n        );\n      });\n    });\n  }\n\n  async schemaUpgrade(className: string, schema: SchemaType, conn: any) {\n    debug('schemaUpgrade');\n    conn = conn || this._client;\n    const self = this;\n\n    await conn.task('schema-upgrade', async t => {\n      const columns = await t.map(\n        'SELECT column_name FROM information_schema.columns WHERE table_name = $<className>',\n        { className },\n        a => a.column_name\n      );\n      const newColumns = Object.keys(schema.fields)\n        .filter(item => columns.indexOf(item) === -1)\n        .map(fieldName => self.addFieldIfNotExists(className, fieldName, schema.fields[fieldName]));\n\n      await t.batch(newColumns);\n    });\n  }\n\n  async addFieldIfNotExists(className: string, fieldName: string, type: any) {\n    // TODO: Must be revised for invalid logic...\n    debug('addFieldIfNotExists');\n    const self = this;\n    await this._client.tx('add-field-if-not-exists', async t => {\n      if (type.type !== 'Relation') {\n        try {\n          await t.none(\n            'ALTER TABLE $<className:name> ADD COLUMN IF NOT EXISTS $<fieldName:name> $<postgresType:raw>',\n            {\n              className,\n              fieldName,\n              postgresType: parseTypeToPostgresType(type),\n            }\n          );\n        } catch (error) {\n          if (error.code === PostgresRelationDoesNotExistError) {\n            return self.createClass(className, { fields: { [fieldName]: type } }, t);\n          }\n          if (error.code !== PostgresDuplicateColumnError) {\n            throw error;\n          }\n          // Column already exists, created by other request. Carry on to see if it's the right type.\n        }\n      } else {\n        await t.none(\n          'CREATE TABLE IF NOT EXISTS $<joinTable:name> (\"relatedId\" varChar(120), \"owningId\" varChar(120), PRIMARY KEY(\"relatedId\", \"owningId\") )',\n          { joinTable: `_Join:${fieldName}:${className}` }\n        );\n      }\n\n      const result = await t.any(\n        'SELECT \"schema\" FROM \"_SCHEMA\" WHERE \"className\" = $<className> and (\"schema\"::json->\\'fields\\'->$<fieldName>) is not null',\n        { className, fieldName }\n      );\n\n      if (result[0]) {\n        throw 'Attempted to add a field that already exists';\n      } else {\n        const path = `{fields,${fieldName}}`;\n        await t.none(\n          'UPDATE \"_SCHEMA\" SET \"schema\"=jsonb_set(\"schema\", $<path>, $<type>)  WHERE \"className\"=$<className>',\n          { path, type, className }\n        );\n      }\n    });\n    this._notifySchemaChange();\n  }\n\n  async updateFieldOptions(className: string, fieldName: string, type: any) {\n    await this._client.tx('update-schema-field-options', async t => {\n      const path = `{fields,${fieldName}}`;\n      await t.none(\n        'UPDATE \"_SCHEMA\" SET \"schema\"=jsonb_set(\"schema\", $<path>, $<type>)  WHERE \"className\"=$<className>',\n        { path, type, className }\n      );\n    });\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  async deleteClass(className: string) {\n    const operations = [\n      { query: `DROP TABLE IF EXISTS $1:name`, values: [className] },\n      {\n        query: `DELETE FROM \"_SCHEMA\" WHERE \"className\" = $1`,\n        values: [className],\n      },\n    ];\n    const response = await this._client\n      .tx(t => t.none(this._pgp.helpers.concat(operations)))\n      .then(() => className.indexOf('_Join:') != 0); // resolves with false when _Join table\n\n    this._notifySchemaChange();\n    return response;\n  }\n\n  // Delete all data known to this adapter. Used for testing.\n  async deleteAllClasses() {\n    const now = new Date().getTime();\n    const helpers = this._pgp.helpers;\n    debug('deleteAllClasses');\n\n    await this._client\n      .task('delete-all-classes', async t => {\n        try {\n          const results = await t.any('SELECT * FROM \"_SCHEMA\"');\n          const joins = results.reduce((list: Array<string>, schema: any) => {\n            return list.concat(joinTablesForSchema(schema.schema));\n          }, []);\n          const classes = [\n            '_SCHEMA',\n            '_PushStatus',\n            '_JobStatus',\n            '_JobSchedule',\n            '_Hooks',\n            '_GlobalConfig',\n            '_GraphQLConfig',\n            '_Audience',\n            '_Idempotency',\n            ...results.map(result => result.className),\n            ...joins,\n          ];\n          const queries = classes.map(className => ({\n            query: 'DROP TABLE IF EXISTS $<className:name>',\n            values: { className },\n          }));\n          await t.tx(tx => tx.none(helpers.concat(queries)));\n        } catch (error) {\n          if (error.code !== PostgresRelationDoesNotExistError) {\n            throw error;\n          }\n          // No _SCHEMA collection. Don't delete anything.\n        }\n      })\n      .then(() => {\n        debug(`deleteAllClasses done in ${new Date().getTime() - now}`);\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  // 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  async deleteFields(className: string, schema: SchemaType, fieldNames: string[]): Promise<void> {\n    debug('deleteFields');\n    fieldNames = fieldNames.reduce((list: Array<string>, fieldName: string) => {\n      const field = schema.fields[fieldName];\n      if (field.type !== 'Relation') {\n        list.push(fieldName);\n      }\n      delete schema.fields[fieldName];\n      return list;\n    }, []);\n\n    const values = [className, ...fieldNames];\n    const columns = fieldNames\n      .map((name, idx) => {\n        return `$${idx + 2}:name`;\n      })\n      .join(', DROP COLUMN');\n\n    await this._client.tx('delete-fields', async t => {\n      await t.none('UPDATE \"_SCHEMA\" SET \"schema\" = $<schema> WHERE \"className\" = $<className>', {\n        schema,\n        className,\n      });\n      if (values.length > 1) {\n        await t.none(`ALTER TABLE $1:name DROP COLUMN IF EXISTS ${columns}`, values);\n      }\n    });\n    this._notifySchemaChange();\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  async getAllClasses() {\n    return this._client.task('get-all-classes', async t => {\n      return await t.map('SELECT * FROM \"_SCHEMA\"', null, row =>\n        toParseSchema({ className: row.className, ...row.schema })\n      );\n    });\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  async getClass(className: string) {\n    debug('getClass');\n    return this._client\n      .any('SELECT * FROM \"_SCHEMA\" WHERE \"className\" = $<className>', {\n        className,\n      })\n      .then(result => {\n        if (result.length !== 1) {\n          throw undefined;\n        }\n        return result[0].schema;\n      })\n      .then(toParseSchema);\n  }\n\n  // TODO: remove the mongo format dependency in the return value\n  async createObject(\n    className: string,\n    schema: SchemaType,\n    object: any,\n    transactionalSession: ?any\n  ) {\n    debug('createObject');\n    let columnsArray = [];\n    const valuesArray = [];\n    schema = toPostgresSchema(schema);\n    const geoPoints = {};\n\n    object = handleDotFields(object);\n\n    validateKeys(object);\n\n    Object.keys(object).forEach(fieldName => {\n      if (object[fieldName] === null) {\n        return;\n      }\n      var authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);\n      const authDataAlreadyExists = !!object.authData;\n      if (authDataMatch) {\n        var provider = authDataMatch[1];\n        object['authData'] = object['authData'] || {};\n        object['authData'][provider] = object[fieldName];\n        delete object[fieldName];\n        fieldName = 'authData';\n        // Avoid adding authData multiple times to the query\n        if (authDataAlreadyExists) {\n          return;\n        }\n      }\n\n      columnsArray.push(fieldName);\n      if (!schema.fields[fieldName] && className === '_User') {\n        if (\n          fieldName === '_email_verify_token' ||\n          fieldName === '_failed_login_count' ||\n          fieldName === '_perishable_token' ||\n          fieldName === '_password_history'\n        ) {\n          valuesArray.push(object[fieldName]);\n        }\n\n        if (fieldName === '_email_verify_token_expires_at') {\n          if (object[fieldName]) {\n            valuesArray.push(object[fieldName].iso);\n          } else {\n            valuesArray.push(null);\n          }\n        }\n\n        if (\n          fieldName === '_account_lockout_expires_at' ||\n          fieldName === '_perishable_token_expires_at' ||\n          fieldName === '_password_changed_at'\n        ) {\n          if (object[fieldName]) {\n            valuesArray.push(object[fieldName].iso);\n          } else {\n            valuesArray.push(null);\n          }\n        }\n        return;\n      }\n      switch (schema.fields[fieldName].type) {\n        case 'Date':\n          if (object[fieldName]) {\n            valuesArray.push(object[fieldName].iso);\n          } else {\n            valuesArray.push(null);\n          }\n          break;\n        case 'Pointer':\n          valuesArray.push(object[fieldName].objectId);\n          break;\n        case 'Array':\n          if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {\n            valuesArray.push(object[fieldName]);\n          } else {\n            valuesArray.push(JSON.stringify(object[fieldName]));\n          }\n          break;\n        case 'Object':\n        case 'Bytes':\n        case 'String':\n        case 'Number':\n        case 'Boolean':\n          valuesArray.push(object[fieldName]);\n          break;\n        case 'File':\n          valuesArray.push(object[fieldName].name);\n          break;\n        case 'Polygon': {\n          const value = convertPolygonToSQL(object[fieldName].coordinates);\n          valuesArray.push(value);\n          break;\n        }\n        case 'GeoPoint':\n          // pop the point and process later\n          geoPoints[fieldName] = object[fieldName];\n          columnsArray.pop();\n          break;\n        default:\n          throw `Type ${schema.fields[fieldName].type} not supported yet`;\n      }\n    });\n\n    columnsArray = columnsArray.concat(Object.keys(geoPoints));\n    const initialValues = valuesArray.map((val, index) => {\n      let termination = '';\n      const fieldName = columnsArray[index];\n      if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {\n        termination = '::text[]';\n      } else if (schema.fields[fieldName] && schema.fields[fieldName].type === 'Array') {\n        termination = '::jsonb';\n      }\n      return `$${index + 2 + columnsArray.length}${termination}`;\n    });\n    const geoPointsInjects = Object.keys(geoPoints).map(key => {\n      const value = geoPoints[key];\n      valuesArray.push(value.longitude, value.latitude);\n      const l = valuesArray.length + columnsArray.length;\n      return `POINT($${l}, $${l + 1})`;\n    });\n\n    const columnsPattern = columnsArray.map((col, index) => `$${index + 2}:name`).join();\n    const valuesPattern = initialValues.concat(geoPointsInjects).join();\n\n    const qs = `INSERT INTO $1:name (${columnsPattern}) VALUES (${valuesPattern})`;\n    const values = [className, ...columnsArray, ...valuesArray];\n    const promise = (transactionalSession ? transactionalSession.t : this._client)\n      .none(qs, values)\n      .then(() => ({ ops: [object] }))\n      .catch(error => {\n        if (error.code === PostgresUniqueIndexViolationError) {\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.constraint) {\n            const matches = error.constraint.match(/unique_([a-zA-Z]+)/);\n            if (matches && Array.isArray(matches)) {\n              err.userInfo = { duplicated_field: matches[1] };\n            }\n          }\n          error = err;\n        }\n        throw error;\n      });\n    if (transactionalSession) {\n      transactionalSession.batch.push(promise);\n    }\n    return promise;\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  async deleteObjectsByQuery(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    transactionalSession: ?any\n  ) {\n    debug('deleteObjectsByQuery');\n    const values = [className];\n    const index = 2;\n    const where = buildWhereClause({\n      schema,\n      index,\n      query,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n    if (Object.keys(query).length === 0) {\n      where.pattern = 'TRUE';\n    }\n    const qs = `WITH deleted AS (DELETE FROM $1:name WHERE ${where.pattern} RETURNING *) SELECT count(*) FROM deleted`;\n    const promise = (transactionalSession ? transactionalSession.t : this._client)\n      .one(qs, values, a => +a.count)\n      .then(count => {\n        if (count === 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        } else {\n          return count;\n        }\n      })\n      .catch(error => {\n        if (error.code !== PostgresRelationDoesNotExistError) {\n          throw error;\n        }\n        // ELSE: Don't delete anything if doesn't exist\n      });\n    if (transactionalSession) {\n      transactionalSession.batch.push(promise);\n    }\n    return promise;\n  }\n  // Return value not currently well specified.\n  async findOneAndUpdate(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ): Promise<any> {\n    debug('findOneAndUpdate');\n    return this.updateObjectsByQuery(className, schema, query, update, transactionalSession).then(\n      val => val[0]\n    );\n  }\n\n  // Apply the update to all objects that match the given Parse Query.\n  async updateObjectsByQuery(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ): Promise<[any]> {\n    debug('updateObjectsByQuery');\n    const updatePatterns = [];\n    const values = [className];\n    let index = 2;\n    schema = toPostgresSchema(schema);\n\n    const originalUpdate = { ...update };\n\n    // Set flag for dot notation fields\n    const dotNotationOptions = {};\n    Object.keys(update).forEach(fieldName => {\n      if (fieldName.indexOf('.') > -1) {\n        const components = fieldName.split('.');\n        const first = components.shift();\n        dotNotationOptions[first] = true;\n      } else {\n        dotNotationOptions[fieldName] = false;\n      }\n    });\n    update = handleDotFields(update);\n    // Resolve authData first,\n    // So we don't end up with multiple key updates\n    for (const fieldName in update) {\n      const authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);\n      if (authDataMatch) {\n        var provider = authDataMatch[1];\n        const value = update[fieldName];\n        delete update[fieldName];\n        update['authData'] = update['authData'] || {};\n        update['authData'][provider] = value;\n      }\n    }\n\n    for (const fieldName in update) {\n      const fieldValue = update[fieldName];\n      // Drop any undefined values.\n      if (typeof fieldValue === 'undefined') {\n        delete update[fieldName];\n      } else if (fieldValue === null) {\n        updatePatterns.push(`$${index}:name = NULL`);\n        values.push(fieldName);\n        index += 1;\n      } else if (fieldName == 'authData') {\n        // This recursively sets the json_object\n        // Only 1 level deep\n        const generate = (jsonb: string, key: string, value: any) => {\n          return `json_object_set_key(COALESCE(${jsonb}, '{}'::jsonb), ${key}, ${value})::jsonb`;\n        };\n        const lastKey = `$${index}:name`;\n        const fieldNameIndex = index;\n        index += 1;\n        values.push(fieldName);\n        const update = Object.keys(fieldValue).reduce((lastKey: string, key: string) => {\n          const str = generate(lastKey, `$${index}::text`, `$${index + 1}::jsonb`);\n          index += 2;\n          let value = fieldValue[key];\n          if (value) {\n            if (value.__op === 'Delete') {\n              value = null;\n            } else {\n              value = JSON.stringify(value);\n            }\n          }\n          values.push(key, value);\n          return str;\n        }, lastKey);\n        updatePatterns.push(`$${fieldNameIndex}:name = ${update}`);\n      } else if (fieldValue.__op === 'Increment') {\n        updatePatterns.push(`$${index}:name = COALESCE($${index}:name, 0) + $${index + 1}`);\n        values.push(fieldName, fieldValue.amount);\n        index += 2;\n      } else if (fieldValue.__op === 'Add') {\n        updatePatterns.push(\n          `$${index}:name = array_add(COALESCE($${index}:name, '[]'::jsonb), $${index + 1}::jsonb)`\n        );\n        values.push(fieldName, JSON.stringify(fieldValue.objects));\n        index += 2;\n      } else if (fieldValue.__op === 'Delete') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, null);\n        index += 2;\n      } else if (fieldValue.__op === 'Remove') {\n        updatePatterns.push(\n          `$${index}:name = array_remove(COALESCE($${index}:name, '[]'::jsonb), $${\n            index + 1\n          }::jsonb)`\n        );\n        values.push(fieldName, JSON.stringify(fieldValue.objects));\n        index += 2;\n      } else if (fieldValue.__op === 'AddUnique') {\n        updatePatterns.push(\n          `$${index}:name = array_add_unique(COALESCE($${index}:name, '[]'::jsonb), $${\n            index + 1\n          }::jsonb)`\n        );\n        values.push(fieldName, JSON.stringify(fieldValue.objects));\n        index += 2;\n      } else if (fieldName === 'updatedAt') {\n        //TODO: stop special casing this. It should check for __type === 'Date' and use .iso\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (typeof fieldValue === 'string') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (typeof fieldValue === 'boolean') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (fieldValue.__type === 'Pointer') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue.objectId);\n        index += 2;\n      } else if (fieldValue.__type === 'Date') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, toPostgresValue(fieldValue));\n        index += 2;\n      } else if (fieldValue instanceof Date) {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (fieldValue.__type === 'File') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, toPostgresValue(fieldValue));\n        index += 2;\n      } else if (fieldValue.__type === 'GeoPoint') {\n        updatePatterns.push(`$${index}:name = POINT($${index + 1}, $${index + 2})`);\n        values.push(fieldName, fieldValue.longitude, fieldValue.latitude);\n        index += 3;\n      } else if (fieldValue.__type === 'Polygon') {\n        const value = convertPolygonToSQL(fieldValue.coordinates);\n        updatePatterns.push(`$${index}:name = $${index + 1}::polygon`);\n        values.push(fieldName, value);\n        index += 2;\n      } else if (fieldValue.__type === 'Relation') {\n        // noop\n      } else if (typeof fieldValue === 'number') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (\n        typeof fieldValue === 'object' &&\n        schema.fields[fieldName] &&\n        schema.fields[fieldName].type === 'Object'\n      ) {\n        // Gather keys to increment\n        const keysToIncrement = Object.keys(originalUpdate)\n          .filter(k => {\n            // choose top level fields that have a delete operation set\n            // Note that Object.keys is iterating over the **original** update object\n            // and that some of the keys of the original update could be null or undefined:\n            // (See the above check `if (fieldValue === null || typeof fieldValue == \"undefined\")`)\n            const value = originalUpdate[k];\n            return (\n              value &&\n              value.__op === 'Increment' &&\n              k.split('.').length === 2 &&\n              k.split('.')[0] === fieldName\n            );\n          })\n          .map(k => k.split('.')[1]);\n\n        let incrementPatterns = '';\n        if (keysToIncrement.length > 0) {\n          incrementPatterns =\n            ' || ' +\n            keysToIncrement\n              .map(c => {\n                const amount = fieldValue[c].amount;\n                return `CONCAT('{\"${c}\":', COALESCE($${index}:name->>'${c}','0')::int + ${amount}, '}')::jsonb`;\n              })\n              .join(' || ');\n          // Strip the keys\n          keysToIncrement.forEach(key => {\n            delete fieldValue[key];\n          });\n        }\n\n        const keysToDelete: Array<string> = Object.keys(originalUpdate)\n          .filter(k => {\n            // choose top level fields that have a delete operation set.\n            const value = originalUpdate[k];\n            return (\n              value &&\n              value.__op === 'Delete' &&\n              k.split('.').length === 2 &&\n              k.split('.')[0] === fieldName\n            );\n          })\n          .map(k => k.split('.')[1]);\n\n        const deletePatterns = keysToDelete.reduce((p: string, c: string, i: number) => {\n          return p + ` - '$${index + 1 + i}:value'`;\n        }, '');\n        // Override Object\n        let updateObject = \"'{}'::jsonb\";\n\n        if (dotNotationOptions[fieldName]) {\n          // Merge Object\n          updateObject = `COALESCE($${index}:name, '{}'::jsonb)`;\n        }\n        updatePatterns.push(\n          `$${index}:name = (${updateObject} ${deletePatterns} ${incrementPatterns} || $${\n            index + 1 + keysToDelete.length\n          }::jsonb )`\n        );\n        values.push(fieldName, ...keysToDelete, JSON.stringify(fieldValue));\n        index += 2 + keysToDelete.length;\n      } else if (\n        Array.isArray(fieldValue) &&\n        schema.fields[fieldName] &&\n        schema.fields[fieldName].type === 'Array'\n      ) {\n        const expectedType = parseTypeToPostgresType(schema.fields[fieldName]);\n        if (expectedType === 'text[]') {\n          updatePatterns.push(`$${index}:name = $${index + 1}::text[]`);\n          values.push(fieldName, fieldValue);\n          index += 2;\n        } else {\n          updatePatterns.push(`$${index}:name = $${index + 1}::jsonb`);\n          values.push(fieldName, JSON.stringify(fieldValue));\n          index += 2;\n        }\n      } else {\n        debug('Not supported update', { fieldName, fieldValue });\n        return Promise.reject(\n          new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            `Postgres doesn't support update ${JSON.stringify(fieldValue)} yet`\n          )\n        );\n      }\n    }\n\n    const where = buildWhereClause({\n      schema,\n      index,\n      query,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n\n    const whereClause = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    const qs = `UPDATE $1:name SET ${updatePatterns.join()} ${whereClause} RETURNING *`;\n    const promise = (transactionalSession ? transactionalSession.t : this._client).any(qs, values);\n    if (transactionalSession) {\n      transactionalSession.batch.push(promise);\n    }\n    return promise;\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    debug('upsertOneObject');\n    const createValue = Object.assign({}, query, update);\n    return this.createObject(className, schema, createValue, transactionalSession).catch(error => {\n      // ignore duplicate value errors as it's upsert\n      if (error.code !== Parse.Error.DUPLICATE_VALUE) {\n        throw error;\n      }\n      return this.findOneAndUpdate(className, schema, query, update, transactionalSession);\n    });\n  }\n\n  find(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    { skip, limit, sort, keys, caseInsensitive, explain }: QueryOptions\n  ) {\n    debug('find');\n    const hasLimit = limit !== undefined;\n    const hasSkip = skip !== undefined;\n    let values = [className];\n    const where = buildWhereClause({\n      schema,\n      query,\n      index: 2,\n      caseInsensitive,\n    });\n    values.push(...where.values);\n    const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    const limitPattern = hasLimit ? `LIMIT $${values.length + 1}` : '';\n    if (hasLimit) {\n      values.push(limit);\n    }\n    const skipPattern = hasSkip ? `OFFSET $${values.length + 1}` : '';\n    if (hasSkip) {\n      values.push(skip);\n    }\n\n    let sortPattern = '';\n    if (sort) {\n      const sortCopy: any = sort;\n      const sorting = Object.keys(sort)\n        .map(key => {\n          const transformKey = transformDotFieldToComponents(key).join('->');\n          // Using $idx pattern gives:  non-integer constant in ORDER BY\n          if (sortCopy[key] === 1) {\n            return `${transformKey} ASC`;\n          }\n          return `${transformKey} DESC`;\n        })\n        .join();\n      sortPattern = sort !== undefined && Object.keys(sort).length > 0 ? `ORDER BY ${sorting}` : '';\n    }\n    if (where.sorts && Object.keys((where.sorts: any)).length > 0) {\n      sortPattern = `ORDER BY ${where.sorts.join()}`;\n    }\n\n    let columns = '*';\n    if (keys) {\n      // Exclude empty keys\n      // Replace ACL by it's keys\n      keys = keys.reduce((memo, key) => {\n        if (key === 'ACL') {\n          memo.push('_rperm');\n          memo.push('_wperm');\n        } else if (\n          key.length > 0 &&\n          // Remove selected field not referenced in the schema\n          // Relation is not a column in postgres\n          // $score is a Parse special field and is also not a column\n          ((schema.fields[key] && schema.fields[key].type !== 'Relation') || key === '$score')\n        ) {\n          memo.push(key);\n        }\n        return memo;\n      }, []);\n      columns = keys\n        .map((key, index) => {\n          if (key === '$score') {\n            return `ts_rank_cd(to_tsvector($${2}, $${3}:name), to_tsquery($${4}, $${5}), 32) as score`;\n          }\n          return `$${index + values.length + 1}:name`;\n        })\n        .join();\n      values = values.concat(keys);\n    }\n\n    const originalQuery = `SELECT ${columns} FROM $1:name ${wherePattern} ${sortPattern} ${limitPattern} ${skipPattern}`;\n    const qs = explain ? this.createExplainableQuery(originalQuery) : originalQuery;\n    return this._client\n      .any(qs, values)\n      .catch(error => {\n        // Query on non existing table, don't crash\n        if (error.code !== PostgresRelationDoesNotExistError) {\n          throw error;\n        }\n        return [];\n      })\n      .then(results => {\n        if (explain) {\n          return results;\n        }\n        return results.map(object => this.postgresObjectToParseObject(className, object, schema));\n      });\n  }\n\n  // Converts from a postgres-format object to a REST-format object.\n  // Does not strip out anything based on a lack of authentication.\n  postgresObjectToParseObject(className: string, object: any, schema: any) {\n    Object.keys(schema.fields).forEach(fieldName => {\n      if (schema.fields[fieldName].type === 'Pointer' && object[fieldName]) {\n        object[fieldName] = {\n          objectId: object[fieldName],\n          __type: 'Pointer',\n          className: schema.fields[fieldName].targetClass,\n        };\n      }\n      if (schema.fields[fieldName].type === 'Relation') {\n        object[fieldName] = {\n          __type: 'Relation',\n          className: schema.fields[fieldName].targetClass,\n        };\n      }\n      if (object[fieldName] && schema.fields[fieldName].type === 'GeoPoint') {\n        object[fieldName] = {\n          __type: 'GeoPoint',\n          latitude: object[fieldName].y,\n          longitude: object[fieldName].x,\n        };\n      }\n      if (object[fieldName] && schema.fields[fieldName].type === 'Polygon') {\n        let coords = object[fieldName];\n        coords = coords.substr(2, coords.length - 4).split('),(');\n        coords = coords.map(point => {\n          return [parseFloat(point.split(',')[1]), parseFloat(point.split(',')[0])];\n        });\n        object[fieldName] = {\n          __type: 'Polygon',\n          coordinates: coords,\n        };\n      }\n      if (object[fieldName] && schema.fields[fieldName].type === 'File') {\n        object[fieldName] = {\n          __type: 'File',\n          name: object[fieldName],\n        };\n      }\n    });\n    //TODO: remove this reliance on the mongo format. DB adapter shouldn't know there is a difference between created at and any other date field.\n    if (object.createdAt) {\n      object.createdAt = object.createdAt.toISOString();\n    }\n    if (object.updatedAt) {\n      object.updatedAt = object.updatedAt.toISOString();\n    }\n    if (object.expiresAt) {\n      object.expiresAt = {\n        __type: 'Date',\n        iso: object.expiresAt.toISOString(),\n      };\n    }\n    if (object._email_verify_token_expires_at) {\n      object._email_verify_token_expires_at = {\n        __type: 'Date',\n        iso: object._email_verify_token_expires_at.toISOString(),\n      };\n    }\n    if (object._account_lockout_expires_at) {\n      object._account_lockout_expires_at = {\n        __type: 'Date',\n        iso: object._account_lockout_expires_at.toISOString(),\n      };\n    }\n    if (object._perishable_token_expires_at) {\n      object._perishable_token_expires_at = {\n        __type: 'Date',\n        iso: object._perishable_token_expires_at.toISOString(),\n      };\n    }\n    if (object._password_changed_at) {\n      object._password_changed_at = {\n        __type: 'Date',\n        iso: object._password_changed_at.toISOString(),\n      };\n    }\n\n    for (const fieldName in object) {\n      if (object[fieldName] === null) {\n        delete object[fieldName];\n      }\n      if (object[fieldName] instanceof Date) {\n        object[fieldName] = {\n          __type: 'Date',\n          iso: object[fieldName].toISOString(),\n        };\n      }\n    }\n\n    return object;\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  async ensureUniqueness(className: string, schema: SchemaType, fieldNames: string[]) {\n    const constraintName = `${className}_unique_${fieldNames.sort().join('_')}`;\n    const constraintPatterns = fieldNames.map((fieldName, index) => `$${index + 3}:name`);\n    const qs = `CREATE UNIQUE INDEX IF NOT EXISTS $2:name ON $1:name(${constraintPatterns.join()})`;\n    return this._client.none(qs, [className, constraintName, ...fieldNames]).catch(error => {\n      if (error.code === PostgresDuplicateRelationError && error.message.includes(constraintName)) {\n        // Index already exists. Ignore error.\n      } else if (\n        error.code === PostgresUniqueIndexViolationError &&\n        error.message.includes(constraintName)\n      ) {\n        // Cast the error into the proper parse error\n        throw new Parse.Error(\n          Parse.Error.DUPLICATE_VALUE,\n          'A duplicate value for a field with unique values was provided'\n        );\n      } else {\n        throw error;\n      }\n    });\n  }\n\n  // Executes a count.\n  async count(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    readPreference?: string,\n    estimate?: boolean = true\n  ) {\n    debug('count');\n    const values = [className];\n    const where = buildWhereClause({\n      schema,\n      query,\n      index: 2,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n\n    const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    let qs = '';\n\n    if (where.pattern.length > 0 || !estimate) {\n      qs = `SELECT count(*) FROM $1:name ${wherePattern}`;\n    } else {\n      qs = 'SELECT reltuples AS approximate_row_count FROM pg_class WHERE relname = $1';\n    }\n\n    return this._client\n      .one(qs, values, a => {\n        if (a.approximate_row_count == null || a.approximate_row_count == -1) {\n          return !isNaN(+a.count) ? +a.count : 0;\n        } else {\n          return +a.approximate_row_count;\n        }\n      })\n      .catch(error => {\n        if (error.code !== PostgresRelationDoesNotExistError) {\n          throw error;\n        }\n        return 0;\n      });\n  }\n\n  async distinct(className: string, schema: SchemaType, query: QueryType, fieldName: string) {\n    debug('distinct');\n    let field = fieldName;\n    let column = fieldName;\n    const isNested = fieldName.indexOf('.') >= 0;\n    if (isNested) {\n      field = transformDotFieldToComponents(fieldName).join('->');\n      column = fieldName.split('.')[0];\n    }\n    const isArrayField =\n      schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array';\n    const isPointerField =\n      schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';\n    const values = [field, column, className];\n    const where = buildWhereClause({\n      schema,\n      query,\n      index: 4,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n\n    const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    const transformer = isArrayField ? 'jsonb_array_elements' : 'ON';\n    let qs = `SELECT DISTINCT ${transformer}($1:name) $2:name FROM $3:name ${wherePattern}`;\n    if (isNested) {\n      qs = `SELECT DISTINCT ${transformer}($1:raw) $2:raw FROM $3:name ${wherePattern}`;\n    }\n    return this._client\n      .any(qs, values)\n      .catch(error => {\n        if (error.code === PostgresMissingColumnError) {\n          return [];\n        }\n        throw error;\n      })\n      .then(results => {\n        if (!isNested) {\n          results = results.filter(object => object[field] !== null);\n          return results.map(object => {\n            if (!isPointerField) {\n              return object[field];\n            }\n            return {\n              __type: 'Pointer',\n              className: schema.fields[fieldName].targetClass,\n              objectId: object[field],\n            };\n          });\n        }\n        const child = fieldName.split('.')[1];\n        return results.map(object => object[column][child]);\n      })\n      .then(results =>\n        results.map(object => this.postgresObjectToParseObject(className, object, schema))\n      );\n  }\n\n  async aggregate(\n    className: string,\n    schema: any,\n    pipeline: any,\n    readPreference: ?string,\n    hint: ?mixed,\n    explain?: boolean\n  ) {\n    debug('aggregate');\n    const values = [className];\n    let index: number = 2;\n    let columns: string[] = [];\n    let countField = null;\n    let groupValues = null;\n    let wherePattern = '';\n    let limitPattern = '';\n    let skipPattern = '';\n    let sortPattern = '';\n    let groupPattern = '';\n    for (let i = 0; i < pipeline.length; i += 1) {\n      const stage = pipeline[i];\n      if (stage.$group) {\n        for (const field in stage.$group) {\n          const value = stage.$group[field];\n          if (value === null || value === undefined) {\n            continue;\n          }\n          if (field === '_id' && typeof value === 'string' && value !== '') {\n            columns.push(`$${index}:name AS \"objectId\"`);\n            groupPattern = `GROUP BY $${index}:name`;\n            values.push(transformAggregateField(value));\n            index += 1;\n            continue;\n          }\n          if (field === '_id' && typeof value === 'object' && Object.keys(value).length !== 0) {\n            groupValues = value;\n            const groupByFields = [];\n            for (const alias in value) {\n              if (typeof value[alias] === 'string' && value[alias]) {\n                const source = transformAggregateField(value[alias]);\n                if (!groupByFields.includes(`\"${source}\"`)) {\n                  groupByFields.push(`\"${source}\"`);\n                }\n                values.push(source, alias);\n                columns.push(`$${index}:name AS $${index + 1}:name`);\n                index += 2;\n              } else {\n                const operation = Object.keys(value[alias])[0];\n                const source = transformAggregateField(value[alias][operation]);\n                if (mongoAggregateToPostgres[operation]) {\n                  if (!groupByFields.includes(`\"${source}\"`)) {\n                    groupByFields.push(`\"${source}\"`);\n                  }\n                  columns.push(\n                    `EXTRACT(${\n                      mongoAggregateToPostgres[operation]\n                    } FROM $${index}:name AT TIME ZONE 'UTC')::integer AS $${index + 1}:name`\n                  );\n                  values.push(source, alias);\n                  index += 2;\n                }\n              }\n            }\n            groupPattern = `GROUP BY $${index}:raw`;\n            values.push(groupByFields.join());\n            index += 1;\n            continue;\n          }\n          if (typeof value === 'object') {\n            if (value.$sum) {\n              if (typeof value.$sum === 'string') {\n                columns.push(`SUM($${index}:name) AS $${index + 1}:name`);\n                values.push(transformAggregateField(value.$sum), field);\n                index += 2;\n              } else {\n                countField = field;\n                columns.push(`COUNT(*) AS $${index}:name`);\n                values.push(field);\n                index += 1;\n              }\n            }\n            if (value.$max) {\n              columns.push(`MAX($${index}:name) AS $${index + 1}:name`);\n              values.push(transformAggregateField(value.$max), field);\n              index += 2;\n            }\n            if (value.$min) {\n              columns.push(`MIN($${index}:name) AS $${index + 1}:name`);\n              values.push(transformAggregateField(value.$min), field);\n              index += 2;\n            }\n            if (value.$avg) {\n              columns.push(`AVG($${index}:name) AS $${index + 1}:name`);\n              values.push(transformAggregateField(value.$avg), field);\n              index += 2;\n            }\n          }\n        }\n      } else {\n        columns.push('*');\n      }\n      if (stage.$project) {\n        if (columns.includes('*')) {\n          columns = [];\n        }\n        for (const field in stage.$project) {\n          const value = stage.$project[field];\n          if (value === 1 || value === true) {\n            columns.push(`$${index}:name`);\n            values.push(field);\n            index += 1;\n          }\n        }\n      }\n      if (stage.$match) {\n        const patterns = [];\n        const orOrAnd = Object.prototype.hasOwnProperty.call(stage.$match, '$or')\n          ? ' OR '\n          : ' AND ';\n\n        if (stage.$match.$or) {\n          const collapse = {};\n          stage.$match.$or.forEach(element => {\n            for (const key in element) {\n              collapse[key] = element[key];\n            }\n          });\n          stage.$match = collapse;\n        }\n        for (const field in stage.$match) {\n          const value = stage.$match[field];\n          const matchPatterns = [];\n          Object.keys(ParseToPosgresComparator).forEach(cmp => {\n            if (value[cmp]) {\n              const pgComparator = ParseToPosgresComparator[cmp];\n              matchPatterns.push(`$${index}:name ${pgComparator} $${index + 1}`);\n              values.push(field, toPostgresValue(value[cmp]));\n              index += 2;\n            }\n          });\n          if (matchPatterns.length > 0) {\n            patterns.push(`(${matchPatterns.join(' AND ')})`);\n          }\n          if (schema.fields[field] && schema.fields[field].type && matchPatterns.length === 0) {\n            patterns.push(`$${index}:name = $${index + 1}`);\n            values.push(field, value);\n            index += 2;\n          }\n        }\n        wherePattern = patterns.length > 0 ? `WHERE ${patterns.join(` ${orOrAnd} `)}` : '';\n      }\n      if (stage.$limit) {\n        limitPattern = `LIMIT $${index}`;\n        values.push(stage.$limit);\n        index += 1;\n      }\n      if (stage.$skip) {\n        skipPattern = `OFFSET $${index}`;\n        values.push(stage.$skip);\n        index += 1;\n      }\n      if (stage.$sort) {\n        const sort = stage.$sort;\n        const keys = Object.keys(sort);\n        const sorting = keys\n          .map(key => {\n            const transformer = sort[key] === 1 ? 'ASC' : 'DESC';\n            const order = `$${index}:name ${transformer}`;\n            index += 1;\n            return order;\n          })\n          .join();\n        values.push(...keys);\n        sortPattern = sort !== undefined && sorting.length > 0 ? `ORDER BY ${sorting}` : '';\n      }\n    }\n\n    if (groupPattern) {\n      columns.forEach((e, i, a) => {\n        if (e && e.trim() === '*') {\n          a[i] = '';\n        }\n      });\n    }\n\n    const originalQuery = `SELECT ${columns\n      .filter(Boolean)\n      .join()} FROM $1:name ${wherePattern} ${skipPattern} ${groupPattern} ${sortPattern} ${limitPattern}`;\n    const qs = explain ? this.createExplainableQuery(originalQuery) : originalQuery;\n    return this._client.any(qs, values).then(a => {\n      if (explain) {\n        return a;\n      }\n      const results = a.map(object => this.postgresObjectToParseObject(className, object, schema));\n      results.forEach(result => {\n        if (!Object.prototype.hasOwnProperty.call(result, 'objectId')) {\n          result.objectId = null;\n        }\n        if (groupValues) {\n          result.objectId = {};\n          for (const key in groupValues) {\n            result.objectId[key] = result[key];\n            delete result[key];\n          }\n        }\n        if (countField) {\n          result[countField] = parseInt(result[countField], 10);\n        }\n      });\n      return results;\n    });\n  }\n\n  async performInitialization({ VolatileClassesSchemas }: any) {\n    // TODO: This method needs to be rewritten to make proper use of connections (@vitaly-t)\n    debug('performInitialization');\n    await this._ensureSchemaCollectionExists();\n    const promises = VolatileClassesSchemas.map(schema => {\n      return this.createTable(schema.className, schema)\n        .catch(err => {\n          if (\n            err.code === PostgresDuplicateRelationError ||\n            err.code === Parse.Error.INVALID_CLASS_NAME\n          ) {\n            return Promise.resolve();\n          }\n          throw err;\n        })\n        .then(() => this.schemaUpgrade(schema.className, schema));\n    });\n    promises.push(this._listenToSchema());\n    return Promise.all(promises)\n      .then(() => {\n        return this._client.tx('perform-initialization', async t => {\n          await t.none(sql.misc.jsonObjectSetKeys);\n          await t.none(sql.array.add);\n          await t.none(sql.array.addUnique);\n          await t.none(sql.array.remove);\n          await t.none(sql.array.containsAll);\n          await t.none(sql.array.containsAllRegex);\n          await t.none(sql.array.contains);\n          return t.ctx;\n        });\n      })\n      .then(ctx => {\n        debug(`initializationDone in ${ctx.duration}`);\n      })\n      .catch(error => {\n        /* eslint-disable no-console */\n        console.error(error);\n      });\n  }\n\n  async createIndexes(className: string, indexes: any, conn: ?any): Promise<void> {\n    return (conn || this._client).tx(t =>\n      t.batch(\n        indexes.map(i => {\n          return t.none('CREATE INDEX IF NOT EXISTS $1:name ON $2:name ($3:name)', [\n            i.name,\n            className,\n            i.key,\n          ]);\n        })\n      )\n    );\n  }\n\n  async createIndexesIfNeeded(\n    className: string,\n    fieldName: string,\n    type: any,\n    conn: ?any\n  ): Promise<void> {\n    await (conn || this._client).none('CREATE INDEX IF NOT EXISTS $1:name ON $2:name ($3:name)', [\n      fieldName,\n      className,\n      type,\n    ]);\n  }\n\n  async dropIndexes(className: string, indexes: any, conn: any): Promise<void> {\n    const queries = indexes.map(i => ({\n      query: 'DROP INDEX $1:name',\n      values: i,\n    }));\n    await (conn || this._client).tx(t => t.none(this._pgp.helpers.concat(queries)));\n  }\n\n  async getIndexes(className: string) {\n    const qs = 'SELECT * FROM pg_indexes WHERE tablename = ${className}';\n    return this._client.any(qs, { className });\n  }\n\n  async updateSchemaWithIndexes(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  // Used for testing purposes\n  async updateEstimatedCount(className: string) {\n    return this._client.none('ANALYZE $1:name', [className]);\n  }\n\n  async createTransactionalSession(): Promise<any> {\n    return new Promise(resolve => {\n      const transactionalSession = {};\n      transactionalSession.result = this._client.tx(t => {\n        transactionalSession.t = t;\n        transactionalSession.promise = new Promise(resolve => {\n          transactionalSession.resolve = resolve;\n        });\n        transactionalSession.batch = [];\n        resolve(transactionalSession);\n        return transactionalSession.promise;\n      });\n    });\n  }\n\n  commitTransactionalSession(transactionalSession: any): Promise<void> {\n    transactionalSession.resolve(transactionalSession.t.batch(transactionalSession.batch));\n    return transactionalSession.result;\n  }\n\n  abortTransactionalSession(transactionalSession: any): Promise<void> {\n    const result = transactionalSession.result.catch();\n    transactionalSession.batch.push(Promise.reject());\n    transactionalSession.resolve(transactionalSession.t.batch(transactionalSession.batch));\n    return result;\n  }\n\n  async ensureIndex(\n    className: string,\n    schema: SchemaType,\n    fieldNames: string[],\n    indexName: ?string,\n    caseInsensitive: boolean = false,\n    options?: Object = {}\n  ): Promise<any> {\n    const conn = options.conn !== undefined ? options.conn : this._client;\n    const defaultIndexName = `parse_default_${fieldNames.sort().join('_')}`;\n    const indexNameOptions: Object =\n      indexName != null ? { name: indexName } : { name: defaultIndexName };\n    const constraintPatterns = caseInsensitive\n      ? fieldNames.map((fieldName, index) => `lower($${index + 3}:name) varchar_pattern_ops`)\n      : fieldNames.map((fieldName, index) => `$${index + 3}:name`);\n    const qs = `CREATE INDEX IF NOT EXISTS $1:name ON $2:name (${constraintPatterns.join()})`;\n    const setIdempotencyFunction =\n      options.setIdempotencyFunction !== undefined ? options.setIdempotencyFunction : false;\n    if (setIdempotencyFunction) {\n      await this.ensureIdempotencyFunctionExists(options);\n    }\n    await conn.none(qs, [indexNameOptions.name, className, ...fieldNames]).catch(error => {\n      if (\n        error.code === PostgresDuplicateRelationError &&\n        error.message.includes(indexNameOptions.name)\n      ) {\n        // Index already exists. Ignore error.\n      } else if (\n        error.code === PostgresUniqueIndexViolationError &&\n        error.message.includes(indexNameOptions.name)\n      ) {\n        // Cast the error into the proper parse error\n        throw new Parse.Error(\n          Parse.Error.DUPLICATE_VALUE,\n          'A duplicate value for a field with unique values was provided'\n        );\n      } else {\n        throw error;\n      }\n    });\n  }\n\n  async deleteIdempotencyFunction(options?: Object = {}): Promise<any> {\n    const conn = options.conn !== undefined ? options.conn : this._client;\n    const qs = 'DROP FUNCTION IF EXISTS idempotency_delete_expired_records()';\n    return conn.none(qs).catch(error => {\n      throw error;\n    });\n  }\n\n  async ensureIdempotencyFunctionExists(options?: Object = {}): Promise<any> {\n    const conn = options.conn !== undefined ? options.conn : this._client;\n    const ttlOptions = options.ttl !== undefined ? `${options.ttl} seconds` : '60 seconds';\n    const qs =\n      'CREATE OR REPLACE FUNCTION idempotency_delete_expired_records() RETURNS void LANGUAGE plpgsql AS $$ BEGIN DELETE FROM \"_Idempotency\" WHERE expire < NOW() - INTERVAL $1; END; $$;';\n    return conn.none(qs, [ttlOptions]).catch(error => {\n      throw error;\n    });\n  }\n}\n\nfunction convertPolygonToSQL(polygon) {\n  if (polygon.length < 3) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, `Polygon must have at least 3 values`);\n  }\n  if (\n    polygon[0][0] !== polygon[polygon.length - 1][0] ||\n    polygon[0][1] !== polygon[polygon.length - 1][1]\n  ) {\n    polygon.push(polygon[0]);\n  }\n  const unique = polygon.filter((item, index, ar) => {\n    let foundIndex = -1;\n    for (let i = 0; i < ar.length; i += 1) {\n      const pt = ar[i];\n      if (pt[0] === item[0] && pt[1] === item[1]) {\n        foundIndex = i;\n        break;\n      }\n    }\n    return foundIndex === index;\n  });\n  if (unique.length < 3) {\n    throw new Parse.Error(\n      Parse.Error.INTERNAL_SERVER_ERROR,\n      'GeoJSON: Loop must have at least 3 different vertices'\n    );\n  }\n  const points = polygon\n    .map(point => {\n      Parse.GeoPoint._validate(parseFloat(point[1]), parseFloat(point[0]));\n      return `(${point[1]}, ${point[0]})`;\n    })\n    .join(', ');\n  return `(${points})`;\n}\n\nfunction removeWhiteSpace(regex) {\n  if (!regex.endsWith('\\n')) {\n    regex += '\\n';\n  }\n\n  // remove non escaped comments\n  return (\n    regex\n      .replace(/([^\\\\])#.*\\n/gim, '$1')\n      // remove lines starting with a comment\n      .replace(/^#.*\\n/gim, '')\n      // remove non escaped whitespace\n      .replace(/([^\\\\])\\s+/gim, '$1')\n      // remove whitespace at the beginning of a line\n      .replace(/^\\s+/, '')\n      .trim()\n  );\n}\n\nfunction processRegexPattern(s) {\n  if (s && s.startsWith('^')) {\n    // regex for startsWith\n    return '^' + literalizeRegexPart(s.slice(1));\n  } else if (s && s.endsWith('$')) {\n    // regex for endsWith\n    return literalizeRegexPart(s.slice(0, s.length - 1)) + '$';\n  }\n\n  // regex for contains\n  return literalizeRegexPart(s);\n}\n\nfunction isStartsWithRegex(value) {\n  if (!value || typeof value !== 'string' || !value.startsWith('^')) {\n    return false;\n  }\n\n  const matches = value.match(/\\^\\\\Q.*\\\\E/);\n  return !!matches;\n}\n\nfunction isAllValuesRegexOrNone(values) {\n  if (!values || !Array.isArray(values) || values.length === 0) {\n    return true;\n  }\n\n  const firstValuesIsRegex = isStartsWithRegex(values[0].$regex);\n  if (values.length === 1) {\n    return firstValuesIsRegex;\n  }\n\n  for (let i = 1, length = values.length; i < length; ++i) {\n    if (firstValuesIsRegex !== isStartsWithRegex(values[i].$regex)) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\nfunction isAnyValueRegexStartsWith(values) {\n  return values.some(function (value) {\n    return isStartsWithRegex(value.$regex);\n  });\n}\n\nfunction createLiteralRegex(remaining) {\n  return remaining\n    .split('')\n    .map(c => {\n      const regex = RegExp('[0-9 ]|\\\\p{L}', 'u'); // Support all unicode letter chars\n      if (c.match(regex) !== null) {\n        // don't escape alphanumeric characters\n        return c;\n      }\n      // escape everything else (single quotes with single quotes, everything else with a backslash)\n      return c === `'` ? `''` : `\\\\${c}`;\n    })\n    .join('');\n}\n\nfunction literalizeRegexPart(s: string) {\n  const matcher1 = /\\\\Q((?!\\\\E).*)\\\\E$/;\n  const result1: any = s.match(matcher1);\n  if (result1 && result1.length > 1 && result1.index > -1) {\n    // process regex that has a beginning and an end specified for the literal text\n    const prefix = s.substr(0, result1.index);\n    const remaining = result1[1];\n\n    return literalizeRegexPart(prefix) + createLiteralRegex(remaining);\n  }\n\n  // process regex that has a beginning specified for the literal text\n  const matcher2 = /\\\\Q((?!\\\\E).*)$/;\n  const result2: any = s.match(matcher2);\n  if (result2 && result2.length > 1 && result2.index > -1) {\n    const prefix = s.substr(0, result2.index);\n    const remaining = result2[1];\n\n    return literalizeRegexPart(prefix) + createLiteralRegex(remaining);\n  }\n\n  // remove all instances of \\Q and \\E from the remaining text & escape single quotes\n  return s\n    .replace(/([^\\\\])(\\\\E)/, '$1')\n    .replace(/([^\\\\])(\\\\Q)/, '$1')\n    .replace(/^\\\\E/, '')\n    .replace(/^\\\\Q/, '')\n    .replace(/([^'])'/, `$1''`)\n    .replace(/^'([^'])/, `''$1`);\n}\n\nvar GeoPointCoder = {\n  isValidJSON(value) {\n    return typeof value === 'object' && value !== null && value.__type === 'GeoPoint';\n  },\n};\n\nexport default PostgresStorageAdapter;\n"]}
|
|
2268
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Utils","require","PostgresRelationDoesNotExistError","PostgresDuplicateRelationError","PostgresDuplicateColumnError","PostgresMissingColumnError","PostgresUniqueIndexViolationError","logger","debug","args","arguments","concat","slice","length","log","getLogger","apply","parseTypeToPostgresType","type","contents","JSON","stringify","ParseToPosgresComparator","$gt","$lt","$gte","$lte","mongoAggregateToPostgres","$dayOfMonth","$dayOfWeek","$dayOfYear","$isoDayOfWeek","$isoWeekYear","$hour","$minute","$second","$millisecond","$month","$week","$year","toPostgresValue","value","__type","iso","name","toPostgresValueCastType","postgresValue","castType","undefined","transformValue","objectId","emptyCLPS","Object","freeze","find","get","count","create","update","delete","addField","protectedFields","defaultCLPS","toParseSchema","schema","className","fields","_hashed_password","_wperm","_rperm","clps","classLevelPermissions","indexes","toPostgresSchema","_password_history","handleDotFields","object","keys","forEach","fieldName","indexOf","components","split","first","shift","currentObj","next","__op","transformDotFieldToComponents","map","cmpt","index","transformDotField","join","transformAggregateField","substr","validateKeys","key","includes","Parse","Error","INVALID_NESTED_KEY","joinTablesForSchema","list","field","push","buildWhereClause","query","caseInsensitive","patterns","values","sorts","isArrayField","initialPatternsLength","fieldValue","$exists","authDataMatch","match","$in","$regex","MAX_INT_PLUS_ONE","clauses","clauseValues","subQuery","clause","pattern","orOrAnd","not","$ne","constraintFieldName","$relativeTime","INVALID_JSON","point","longitude","latitude","$eq","isInOrNin","Array","isArray","$nin","inPatterns","allowNull","listElem","listIndex","createConstraint","baseArray","notIn","_","flatMap","elt","$all","isAnyValueRegexStartsWith","isAllValuesRegexOrNone","i","processRegexPattern","substring","$containedBy","arr","$text","search","$search","language","$term","$language","$caseSensitive","$diacriticSensitive","$nearSphere","distance","$maxDistance","distanceInKM","$within","$box","box","left","bottom","right","top","$geoWithin","$centerSphere","centerSphere","GeoPoint","GeoPointCoder","isValidJSON","_validate","isNaN","$polygon","polygon","points","coordinates","$geoIntersects","$point","regex","operator","opts","$options","removeWhiteSpace","convertPolygonToSQL","cmp","pgComparator","parserResult","relativeTimeToDate","status","result","console","error","info","OPERATION_FORBIDDEN","PostgresStorageAdapter","constructor","uri","collectionPrefix","databaseOptions","_collectionPrefix","enableSchemaHooks","client","pgp","createClient","_client","_onchange","_pgp","_uuid","uuidv4","canSortOnJoinTables","watch","callback","createExplainableQuery","analyze","handleShutdown","_stream","done","$pool","end","_listenToSchema","connect","direct","on","data","payload","parse","senderId","none","_notifySchemaChange","catch","_ensureSchemaCollectionExists","conn","classExists","one","a","exists","setClassLevelPermissions","CLPs","task","t","setIndexesWithSchemaFormat","submittedIndexes","existingIndexes","self","Promise","resolve","_id_","_id","deletedIndexes","insertedIndexes","INVALID_QUERY","prototype","hasOwnProperty","call","tx","createIndexes","dropIndexes","createClass","parseSchema","createTable","err","code","detail","DUPLICATE_VALUE","valuesArray","patternsArray","assign","_email_verify_token_expires_at","_email_verify_token","_account_lockout_expires_at","_failed_login_count","_perishable_token","_perishable_token_expires_at","_password_changed_at","relations","parseType","qs","batch","joinTable","schemaUpgrade","columns","column_name","newColumns","filter","item","addFieldIfNotExists","postgresType","any","path","updateFieldOptions","deleteClass","operations","response","helpers","then","deleteAllClasses","now","Date","getTime","results","joins","reduce","classes","queries","deleteFields","fieldNames","idx","getAllClasses","row","getClass","createObject","transactionalSession","columnsArray","geoPoints","authDataAlreadyExists","authData","provider","pop","initialValues","val","termination","geoPointsInjects","l","columnsPattern","col","valuesPattern","promise","ops","underlyingError","constraint","matches","userInfo","duplicated_field","deleteObjectsByQuery","where","OBJECT_NOT_FOUND","findOneAndUpdate","updateObjectsByQuery","updatePatterns","originalUpdate","dotNotationOptions","generate","jsonb","lastKey","fieldNameIndex","str","amount","objects","keysToIncrement","k","incrementPatterns","c","keysToDelete","deletePatterns","p","updateObject","expectedType","reject","whereClause","upsertOneObject","createValue","skip","limit","sort","explain","hasLimit","hasSkip","wherePattern","limitPattern","skipPattern","sortPattern","sortCopy","sorting","transformKey","memo","originalQuery","postgresObjectToParseObject","targetClass","y","x","coords","parseFloat","createdAt","toISOString","updatedAt","expiresAt","ensureUniqueness","constraintName","constraintPatterns","message","readPreference","estimate","approximate_row_count","distinct","column","isNested","isPointerField","transformer","child","aggregate","pipeline","hint","countField","groupValues","groupPattern","stage","$group","groupByFields","alias","source","operation","$sum","$max","$min","$avg","$project","$match","$or","collapse","element","matchPatterns","$limit","$skip","$sort","order","e","trim","Boolean","parseInt","performInitialization","VolatileClassesSchemas","promises","INVALID_CLASS_NAME","all","sql","misc","jsonObjectSetKeys","array","add","addUnique","remove","containsAll","containsAllRegex","contains","ctx","duration","createIndexesIfNeeded","getIndexes","updateSchemaWithIndexes","updateEstimatedCount","createTransactionalSession","commitTransactionalSession","abortTransactionalSession","ensureIndex","indexName","options","defaultIndexName","indexNameOptions","setIdempotencyFunction","ensureIdempotencyFunctionExists","deleteIdempotencyFunction","ttlOptions","ttl","unique","ar","foundIndex","pt","INTERNAL_SERVER_ERROR","endsWith","replace","s","startsWith","literalizeRegexPart","isStartsWithRegex","firstValuesIsRegex","some","createLiteralRegex","remaining","RegExp","matcher1","result1","prefix","matcher2","result2"],"sources":["../../../../src/Adapters/Storage/Postgres/PostgresStorageAdapter.js"],"sourcesContent":["// @flow\nimport { createClient } from './PostgresClient';\n// @flow-disable-next\nimport Parse from 'parse/node';\n// @flow-disable-next\nimport _ from 'lodash';\n// @flow-disable-next\nimport { v4 as uuidv4 } from 'uuid';\nimport sql from './sql';\nimport { StorageAdapter } from '../StorageAdapter';\nimport type { SchemaType, QueryType, QueryOptions } from '../StorageAdapter';\nconst Utils = require('../../../Utils');\n\nconst PostgresRelationDoesNotExistError = '42P01';\nconst PostgresDuplicateRelationError = '42P07';\nconst PostgresDuplicateColumnError = '42701';\nconst PostgresMissingColumnError = '42703';\nconst PostgresUniqueIndexViolationError = '23505';\nconst logger = require('../../../logger');\n\nconst debug = function (...args: any) {\n  args = ['PG: ' + arguments[0]].concat(args.slice(1, args.length));\n  const log = logger.getLogger();\n  log.debug.apply(log, args);\n};\n\nconst parseTypeToPostgresType = type => {\n  switch (type.type) {\n    case 'String':\n      return 'text';\n    case 'Date':\n      return 'timestamp with time zone';\n    case 'Object':\n      return 'jsonb';\n    case 'File':\n      return 'text';\n    case 'Boolean':\n      return 'boolean';\n    case 'Pointer':\n      return 'text';\n    case 'Number':\n      return 'double precision';\n    case 'GeoPoint':\n      return 'point';\n    case 'Bytes':\n      return 'jsonb';\n    case 'Polygon':\n      return 'polygon';\n    case 'Array':\n      if (type.contents && type.contents.type === 'String') {\n        return 'text[]';\n      } else {\n        return 'jsonb';\n      }\n    default:\n      throw `no type for ${JSON.stringify(type)} yet`;\n  }\n};\n\nconst ParseToPosgresComparator = {\n  $gt: '>',\n  $lt: '<',\n  $gte: '>=',\n  $lte: '<=',\n};\n\nconst mongoAggregateToPostgres = {\n  $dayOfMonth: 'DAY',\n  $dayOfWeek: 'DOW',\n  $dayOfYear: 'DOY',\n  $isoDayOfWeek: 'ISODOW',\n  $isoWeekYear: 'ISOYEAR',\n  $hour: 'HOUR',\n  $minute: 'MINUTE',\n  $second: 'SECOND',\n  $millisecond: 'MILLISECONDS',\n  $month: 'MONTH',\n  $week: 'WEEK',\n  $year: 'YEAR',\n};\n\nconst toPostgresValue = value => {\n  if (typeof value === 'object') {\n    if (value.__type === 'Date') {\n      return value.iso;\n    }\n    if (value.__type === 'File') {\n      return value.name;\n    }\n  }\n  return value;\n};\n\nconst toPostgresValueCastType = value => {\n  const postgresValue = toPostgresValue(value);\n  let castType;\n  switch (typeof postgresValue) {\n    case 'number':\n      castType = 'double precision';\n      break;\n    case 'boolean':\n      castType = 'boolean';\n      break;\n    default:\n      castType = undefined;\n  }\n  return castType;\n};\n\nconst transformValue = value => {\n  if (typeof value === 'object' && value.__type === 'Pointer') {\n    return value.objectId;\n  }\n  return value;\n};\n\n// Duplicate from then mongo adapter...\nconst emptyCLPS = Object.freeze({\n  find: {},\n  get: {},\n  count: {},\n  create: {},\n  update: {},\n  delete: {},\n  addField: {},\n  protectedFields: {},\n});\n\nconst defaultCLPS = Object.freeze({\n  find: { '*': true },\n  get: { '*': true },\n  count: { '*': true },\n  create: { '*': true },\n  update: { '*': true },\n  delete: { '*': true },\n  addField: { '*': true },\n  protectedFields: { '*': [] },\n});\n\nconst toParseSchema = schema => {\n  if (schema.className === '_User') {\n    delete schema.fields._hashed_password;\n  }\n  if (schema.fields) {\n    delete schema.fields._wperm;\n    delete schema.fields._rperm;\n  }\n  let clps = defaultCLPS;\n  if (schema.classLevelPermissions) {\n    clps = { ...emptyCLPS, ...schema.classLevelPermissions };\n  }\n  let indexes = {};\n  if (schema.indexes) {\n    indexes = { ...schema.indexes };\n  }\n  return {\n    className: schema.className,\n    fields: schema.fields,\n    classLevelPermissions: clps,\n    indexes,\n  };\n};\n\nconst toPostgresSchema = schema => {\n  if (!schema) {\n    return schema;\n  }\n  schema.fields = schema.fields || {};\n  schema.fields._wperm = { type: 'Array', contents: { type: 'String' } };\n  schema.fields._rperm = { type: 'Array', contents: { type: 'String' } };\n  if (schema.className === '_User') {\n    schema.fields._hashed_password = { type: 'String' };\n    schema.fields._password_history = { type: 'Array' };\n  }\n  return schema;\n};\n\nconst handleDotFields = object => {\n  Object.keys(object).forEach(fieldName => {\n    if (fieldName.indexOf('.') > -1) {\n      const components = fieldName.split('.');\n      const first = components.shift();\n      object[first] = object[first] || {};\n      let currentObj = object[first];\n      let next;\n      let value = object[fieldName];\n      if (value && value.__op === 'Delete') {\n        value = undefined;\n      }\n      /* eslint-disable no-cond-assign */\n      while ((next = components.shift())) {\n        /* eslint-enable no-cond-assign */\n        currentObj[next] = currentObj[next] || {};\n        if (components.length === 0) {\n          currentObj[next] = value;\n        }\n        currentObj = currentObj[next];\n      }\n      delete object[fieldName];\n    }\n  });\n  return object;\n};\n\nconst transformDotFieldToComponents = fieldName => {\n  return fieldName.split('.').map((cmpt, index) => {\n    if (index === 0) {\n      return `\"${cmpt}\"`;\n    }\n    return `'${cmpt}'`;\n  });\n};\n\nconst transformDotField = fieldName => {\n  if (fieldName.indexOf('.') === -1) {\n    return `\"${fieldName}\"`;\n  }\n  const components = transformDotFieldToComponents(fieldName);\n  let name = components.slice(0, components.length - 1).join('->');\n  name += '->>' + components[components.length - 1];\n  return name;\n};\n\nconst transformAggregateField = fieldName => {\n  if (typeof fieldName !== 'string') {\n    return fieldName;\n  }\n  if (fieldName === '$_created_at') {\n    return 'createdAt';\n  }\n  if (fieldName === '$_updated_at') {\n    return 'updatedAt';\n  }\n  return fieldName.substr(1);\n};\n\nconst validateKeys = object => {\n  if (typeof object == 'object') {\n    for (const key in object) {\n      if (typeof object[key] == 'object') {\n        validateKeys(object[key]);\n      }\n\n      if (key.includes('$') || key.includes('.')) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_NESTED_KEY,\n          \"Nested keys should not contain the '$' or '.' characters\"\n        );\n      }\n    }\n  }\n};\n\n// Returns the list of join tables on a schema\nconst joinTablesForSchema = schema => {\n  const list = [];\n  if (schema) {\n    Object.keys(schema.fields).forEach(field => {\n      if (schema.fields[field].type === 'Relation') {\n        list.push(`_Join:${field}:${schema.className}`);\n      }\n    });\n  }\n  return list;\n};\n\ninterface WhereClause {\n  pattern: string;\n  values: Array<any>;\n  sorts: Array<any>;\n}\n\nconst buildWhereClause = ({ schema, query, index, caseInsensitive }): WhereClause => {\n  const patterns = [];\n  let values = [];\n  const sorts = [];\n\n  schema = toPostgresSchema(schema);\n  for (const fieldName in query) {\n    const isArrayField =\n      schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array';\n    const initialPatternsLength = patterns.length;\n    const fieldValue = query[fieldName];\n\n    // nothing in the schema, it's gonna blow up\n    if (!schema.fields[fieldName]) {\n      // as it won't exist\n      if (fieldValue && fieldValue.$exists === false) {\n        continue;\n      }\n    }\n    const authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);\n    if (authDataMatch) {\n      // TODO: Handle querying by _auth_data_provider, authData is stored in authData field\n      continue;\n    } else if (caseInsensitive && (fieldName === 'username' || fieldName === 'email')) {\n      patterns.push(`LOWER($${index}:name) = LOWER($${index + 1})`);\n      values.push(fieldName, fieldValue);\n      index += 2;\n    } else if (fieldName.indexOf('.') >= 0) {\n      let name = transformDotField(fieldName);\n      if (fieldValue === null) {\n        patterns.push(`$${index}:raw IS NULL`);\n        values.push(name);\n        index += 1;\n        continue;\n      } else {\n        if (fieldValue.$in) {\n          name = transformDotFieldToComponents(fieldName).join('->');\n          patterns.push(`($${index}:raw)::jsonb @> $${index + 1}::jsonb`);\n          values.push(name, JSON.stringify(fieldValue.$in));\n          index += 2;\n        } else if (fieldValue.$regex) {\n          // Handle later\n        } else if (typeof fieldValue !== 'object') {\n          patterns.push(`$${index}:raw = $${index + 1}::text`);\n          values.push(name, fieldValue);\n          index += 2;\n        }\n      }\n    } else if (fieldValue === null || fieldValue === undefined) {\n      patterns.push(`$${index}:name IS NULL`);\n      values.push(fieldName);\n      index += 1;\n      continue;\n    } else if (typeof fieldValue === 'string') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      values.push(fieldName, fieldValue);\n      index += 2;\n    } else if (typeof fieldValue === 'boolean') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      // Can't cast boolean to double precision\n      if (schema.fields[fieldName] && schema.fields[fieldName].type === 'Number') {\n        // Should always return zero results\n        const MAX_INT_PLUS_ONE = 9223372036854775808;\n        values.push(fieldName, MAX_INT_PLUS_ONE);\n      } else {\n        values.push(fieldName, fieldValue);\n      }\n      index += 2;\n    } else if (typeof fieldValue === 'number') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      values.push(fieldName, fieldValue);\n      index += 2;\n    } else if (['$or', '$nor', '$and'].includes(fieldName)) {\n      const clauses = [];\n      const clauseValues = [];\n      fieldValue.forEach(subQuery => {\n        const clause = buildWhereClause({\n          schema,\n          query: subQuery,\n          index,\n          caseInsensitive,\n        });\n        if (clause.pattern.length > 0) {\n          clauses.push(clause.pattern);\n          clauseValues.push(...clause.values);\n          index += clause.values.length;\n        }\n      });\n\n      const orOrAnd = fieldName === '$and' ? ' AND ' : ' OR ';\n      const not = fieldName === '$nor' ? ' NOT ' : '';\n\n      patterns.push(`${not}(${clauses.join(orOrAnd)})`);\n      values.push(...clauseValues);\n    }\n\n    if (fieldValue.$ne !== undefined) {\n      if (isArrayField) {\n        fieldValue.$ne = JSON.stringify([fieldValue.$ne]);\n        patterns.push(`NOT array_contains($${index}:name, $${index + 1})`);\n      } else {\n        if (fieldValue.$ne === null) {\n          patterns.push(`$${index}:name IS NOT NULL`);\n          values.push(fieldName);\n          index += 1;\n          continue;\n        } else {\n          // if not null, we need to manually exclude null\n          if (fieldValue.$ne.__type === 'GeoPoint') {\n            patterns.push(\n              `($${index}:name <> POINT($${index + 1}, $${index + 2}) OR $${index}:name IS NULL)`\n            );\n          } else {\n            if (fieldName.indexOf('.') >= 0) {\n              const castType = toPostgresValueCastType(fieldValue.$ne);\n              const constraintFieldName = castType\n                ? `CAST ((${transformDotField(fieldName)}) AS ${castType})`\n                : transformDotField(fieldName);\n              patterns.push(\n                `(${constraintFieldName} <> $${index + 1} OR ${constraintFieldName} IS NULL)`\n              );\n            } else if (typeof fieldValue.$ne === 'object' && fieldValue.$ne.$relativeTime) {\n              throw new Parse.Error(\n                Parse.Error.INVALID_JSON,\n                '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators'\n              );\n            } else {\n              patterns.push(`($${index}:name <> $${index + 1} OR $${index}:name IS NULL)`);\n            }\n          }\n        }\n      }\n      if (fieldValue.$ne.__type === 'GeoPoint') {\n        const point = fieldValue.$ne;\n        values.push(fieldName, point.longitude, point.latitude);\n        index += 3;\n      } else {\n        // TODO: support arrays\n        values.push(fieldName, fieldValue.$ne);\n        index += 2;\n      }\n    }\n    if (fieldValue.$eq !== undefined) {\n      if (fieldValue.$eq === null) {\n        patterns.push(`$${index}:name IS NULL`);\n        values.push(fieldName);\n        index += 1;\n      } else {\n        if (fieldName.indexOf('.') >= 0) {\n          const castType = toPostgresValueCastType(fieldValue.$eq);\n          const constraintFieldName = castType\n            ? `CAST ((${transformDotField(fieldName)}) AS ${castType})`\n            : transformDotField(fieldName);\n          values.push(fieldValue.$eq);\n          patterns.push(`${constraintFieldName} = $${index++}`);\n        } else if (typeof fieldValue.$eq === 'object' && fieldValue.$eq.$relativeTime) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators'\n          );\n        } else {\n          values.push(fieldName, fieldValue.$eq);\n          patterns.push(`$${index}:name = $${index + 1}`);\n          index += 2;\n        }\n      }\n    }\n    const isInOrNin = Array.isArray(fieldValue.$in) || Array.isArray(fieldValue.$nin);\n    if (\n      Array.isArray(fieldValue.$in) &&\n      isArrayField &&\n      schema.fields[fieldName].contents &&\n      schema.fields[fieldName].contents.type === 'String'\n    ) {\n      const inPatterns = [];\n      let allowNull = false;\n      values.push(fieldName);\n      fieldValue.$in.forEach((listElem, listIndex) => {\n        if (listElem === null) {\n          allowNull = true;\n        } else {\n          values.push(listElem);\n          inPatterns.push(`$${index + 1 + listIndex - (allowNull ? 1 : 0)}`);\n        }\n      });\n      if (allowNull) {\n        patterns.push(`($${index}:name IS NULL OR $${index}:name && ARRAY[${inPatterns.join()}])`);\n      } else {\n        patterns.push(`$${index}:name && ARRAY[${inPatterns.join()}]`);\n      }\n      index = index + 1 + inPatterns.length;\n    } else if (isInOrNin) {\n      var createConstraint = (baseArray, notIn) => {\n        const not = notIn ? ' NOT ' : '';\n        if (baseArray.length > 0) {\n          if (isArrayField) {\n            patterns.push(`${not} array_contains($${index}:name, $${index + 1})`);\n            values.push(fieldName, JSON.stringify(baseArray));\n            index += 2;\n          } else {\n            // Handle Nested Dot Notation Above\n            if (fieldName.indexOf('.') >= 0) {\n              return;\n            }\n            const inPatterns = [];\n            values.push(fieldName);\n            baseArray.forEach((listElem, listIndex) => {\n              if (listElem != null) {\n                values.push(listElem);\n                inPatterns.push(`$${index + 1 + listIndex}`);\n              }\n            });\n            patterns.push(`$${index}:name ${not} IN (${inPatterns.join()})`);\n            index = index + 1 + inPatterns.length;\n          }\n        } else if (!notIn) {\n          values.push(fieldName);\n          patterns.push(`$${index}:name IS NULL`);\n          index = index + 1;\n        } else {\n          // Handle empty array\n          if (notIn) {\n            patterns.push('1 = 1'); // Return all values\n          } else {\n            patterns.push('1 = 2'); // Return no values\n          }\n        }\n      };\n      if (fieldValue.$in) {\n        createConstraint(\n          _.flatMap(fieldValue.$in, elt => elt),\n          false\n        );\n      }\n      if (fieldValue.$nin) {\n        createConstraint(\n          _.flatMap(fieldValue.$nin, elt => elt),\n          true\n        );\n      }\n    } else if (typeof fieldValue.$in !== 'undefined') {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad $in value');\n    } else if (typeof fieldValue.$nin !== 'undefined') {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad $nin value');\n    }\n\n    if (Array.isArray(fieldValue.$all) && isArrayField) {\n      if (isAnyValueRegexStartsWith(fieldValue.$all)) {\n        if (!isAllValuesRegexOrNone(fieldValue.$all)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            'All $all values must be of regex type or none: ' + fieldValue.$all\n          );\n        }\n\n        for (let i = 0; i < fieldValue.$all.length; i += 1) {\n          const value = processRegexPattern(fieldValue.$all[i].$regex);\n          fieldValue.$all[i] = value.substring(1) + '%';\n        }\n        patterns.push(`array_contains_all_regex($${index}:name, $${index + 1}::jsonb)`);\n      } else {\n        patterns.push(`array_contains_all($${index}:name, $${index + 1}::jsonb)`);\n      }\n      values.push(fieldName, JSON.stringify(fieldValue.$all));\n      index += 2;\n    } else if (Array.isArray(fieldValue.$all)) {\n      if (fieldValue.$all.length === 1) {\n        patterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue.$all[0].objectId);\n        index += 2;\n      }\n    }\n\n    if (typeof fieldValue.$exists !== 'undefined') {\n      if (typeof fieldValue.$exists === 'object' && fieldValue.$exists.$relativeTime) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          '$relativeTime can only be used with the $lt, $lte, $gt, and $gte operators'\n        );\n      } else if (fieldValue.$exists) {\n        patterns.push(`$${index}:name IS NOT NULL`);\n      } else {\n        patterns.push(`$${index}:name IS NULL`);\n      }\n      values.push(fieldName);\n      index += 1;\n    }\n\n    if (fieldValue.$containedBy) {\n      const arr = fieldValue.$containedBy;\n      if (!(arr instanceof Array)) {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $containedBy: should be an array`);\n      }\n\n      patterns.push(`$${index}:name <@ $${index + 1}::jsonb`);\n      values.push(fieldName, JSON.stringify(arr));\n      index += 2;\n    }\n\n    if (fieldValue.$text) {\n      const search = fieldValue.$text.$search;\n      let language = 'english';\n      if (typeof search !== 'object') {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $text: $search, should be object`);\n      }\n      if (!search.$term || typeof search.$term !== 'string') {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $text: $term, should be string`);\n      }\n      if (search.$language && typeof search.$language !== 'string') {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `bad $text: $language, should be string`);\n      } else if (search.$language) {\n        language = search.$language;\n      }\n      if (search.$caseSensitive && typeof search.$caseSensitive !== 'boolean') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $caseSensitive, should be boolean`\n        );\n      } else if (search.$caseSensitive) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $caseSensitive not supported, please use $regex or create a separate lower case column.`\n        );\n      }\n      if (search.$diacriticSensitive && typeof search.$diacriticSensitive !== 'boolean') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $diacriticSensitive, should be boolean`\n        );\n      } else if (search.$diacriticSensitive === false) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `bad $text: $diacriticSensitive - false not supported, install Postgres Unaccent Extension`\n        );\n      }\n      patterns.push(\n        `to_tsvector($${index}, $${index + 1}:name) @@ to_tsquery($${index + 2}, $${index + 3})`\n      );\n      values.push(language, fieldName, language, search.$term);\n      index += 4;\n    }\n\n    if (fieldValue.$nearSphere) {\n      const point = fieldValue.$nearSphere;\n      const distance = fieldValue.$maxDistance;\n      const distanceInKM = distance * 6371 * 1000;\n      patterns.push(\n        `ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${\n          index + 2\n        })::geometry) <= $${index + 3}`\n      );\n      sorts.push(\n        `ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${\n          index + 2\n        })::geometry) ASC`\n      );\n      values.push(fieldName, point.longitude, point.latitude, distanceInKM);\n      index += 4;\n    }\n\n    if (fieldValue.$within && fieldValue.$within.$box) {\n      const box = fieldValue.$within.$box;\n      const left = box[0].longitude;\n      const bottom = box[0].latitude;\n      const right = box[1].longitude;\n      const top = box[1].latitude;\n\n      patterns.push(`$${index}:name::point <@ $${index + 1}::box`);\n      values.push(fieldName, `((${left}, ${bottom}), (${right}, ${top}))`);\n      index += 2;\n    }\n\n    if (fieldValue.$geoWithin && fieldValue.$geoWithin.$centerSphere) {\n      const centerSphere = fieldValue.$geoWithin.$centerSphere;\n      if (!(centerSphere instanceof Array) || centerSphere.length < 2) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoWithin value; $centerSphere should be an array of Parse.GeoPoint and distance'\n        );\n      }\n      // Get point, convert to geo point if necessary and validate\n      let point = centerSphere[0];\n      if (point instanceof Array && point.length === 2) {\n        point = new Parse.GeoPoint(point[1], point[0]);\n      } else if (!GeoPointCoder.isValidJSON(point)) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoWithin value; $centerSphere geo point invalid'\n        );\n      }\n      Parse.GeoPoint._validate(point.latitude, point.longitude);\n      // Get distance and validate\n      const distance = centerSphere[1];\n      if (isNaN(distance) || distance < 0) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoWithin value; $centerSphere distance invalid'\n        );\n      }\n      const distanceInKM = distance * 6371 * 1000;\n      patterns.push(\n        `ST_DistanceSphere($${index}:name::geometry, POINT($${index + 1}, $${\n          index + 2\n        })::geometry) <= $${index + 3}`\n      );\n      values.push(fieldName, point.longitude, point.latitude, distanceInKM);\n      index += 4;\n    }\n\n    if (fieldValue.$geoWithin && fieldValue.$geoWithin.$polygon) {\n      const polygon = fieldValue.$geoWithin.$polygon;\n      let points;\n      if (typeof polygon === 'object' && polygon.__type === 'Polygon') {\n        if (!polygon.coordinates || polygon.coordinates.length < 3) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            'bad $geoWithin value; Polygon.coordinates should contain at least 3 lon/lat pairs'\n          );\n        }\n        points = polygon.coordinates;\n      } else if (polygon instanceof Array) {\n        if (polygon.length < 3) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            'bad $geoWithin value; $polygon should contain at least 3 GeoPoints'\n          );\n        }\n        points = polygon;\n      } else {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          \"bad $geoWithin value; $polygon should be Polygon object or Array of Parse.GeoPoint's\"\n        );\n      }\n      points = points\n        .map(point => {\n          if (point instanceof Array && point.length === 2) {\n            Parse.GeoPoint._validate(point[1], point[0]);\n            return `(${point[0]}, ${point[1]})`;\n          }\n          if (typeof point !== 'object' || point.__type !== 'GeoPoint') {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad $geoWithin value');\n          } else {\n            Parse.GeoPoint._validate(point.latitude, point.longitude);\n          }\n          return `(${point.longitude}, ${point.latitude})`;\n        })\n        .join(', ');\n\n      patterns.push(`$${index}:name::point <@ $${index + 1}::polygon`);\n      values.push(fieldName, `(${points})`);\n      index += 2;\n    }\n    if (fieldValue.$geoIntersects && fieldValue.$geoIntersects.$point) {\n      const point = fieldValue.$geoIntersects.$point;\n      if (typeof point !== 'object' || point.__type !== 'GeoPoint') {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          'bad $geoIntersect value; $point should be GeoPoint'\n        );\n      } else {\n        Parse.GeoPoint._validate(point.latitude, point.longitude);\n      }\n      patterns.push(`$${index}:name::polygon @> $${index + 1}::point`);\n      values.push(fieldName, `(${point.longitude}, ${point.latitude})`);\n      index += 2;\n    }\n\n    if (fieldValue.$regex) {\n      let regex = fieldValue.$regex;\n      let operator = '~';\n      const opts = fieldValue.$options;\n      if (opts) {\n        if (opts.indexOf('i') >= 0) {\n          operator = '~*';\n        }\n        if (opts.indexOf('x') >= 0) {\n          regex = removeWhiteSpace(regex);\n        }\n      }\n\n      const name = transformDotField(fieldName);\n      regex = processRegexPattern(regex);\n\n      patterns.push(`$${index}:raw ${operator} '$${index + 1}:raw'`);\n      values.push(name, regex);\n      index += 2;\n    }\n\n    if (fieldValue.__type === 'Pointer') {\n      if (isArrayField) {\n        patterns.push(`array_contains($${index}:name, $${index + 1})`);\n        values.push(fieldName, JSON.stringify([fieldValue]));\n        index += 2;\n      } else {\n        patterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue.objectId);\n        index += 2;\n      }\n    }\n\n    if (fieldValue.__type === 'Date') {\n      patterns.push(`$${index}:name = $${index + 1}`);\n      values.push(fieldName, fieldValue.iso);\n      index += 2;\n    }\n\n    if (fieldValue.__type === 'GeoPoint') {\n      patterns.push(`$${index}:name ~= POINT($${index + 1}, $${index + 2})`);\n      values.push(fieldName, fieldValue.longitude, fieldValue.latitude);\n      index += 3;\n    }\n\n    if (fieldValue.__type === 'Polygon') {\n      const value = convertPolygonToSQL(fieldValue.coordinates);\n      patterns.push(`$${index}:name ~= $${index + 1}::polygon`);\n      values.push(fieldName, value);\n      index += 2;\n    }\n\n    Object.keys(ParseToPosgresComparator).forEach(cmp => {\n      if (fieldValue[cmp] || fieldValue[cmp] === 0) {\n        const pgComparator = ParseToPosgresComparator[cmp];\n        let constraintFieldName;\n        let postgresValue = toPostgresValue(fieldValue[cmp]);\n\n        if (fieldName.indexOf('.') >= 0) {\n          const castType = toPostgresValueCastType(fieldValue[cmp]);\n          constraintFieldName = castType\n            ? `CAST ((${transformDotField(fieldName)}) AS ${castType})`\n            : transformDotField(fieldName);\n        } else {\n          if (typeof postgresValue === 'object' && postgresValue.$relativeTime) {\n            if (schema.fields[fieldName].type !== 'Date') {\n              throw new Parse.Error(\n                Parse.Error.INVALID_JSON,\n                '$relativeTime can only be used with Date field'\n              );\n            }\n            const parserResult = Utils.relativeTimeToDate(postgresValue.$relativeTime);\n            if (parserResult.status === 'success') {\n              postgresValue = toPostgresValue(parserResult.result);\n            } else {\n              console.error('Error while parsing relative date', parserResult);\n              throw new Parse.Error(\n                Parse.Error.INVALID_JSON,\n                `bad $relativeTime (${postgresValue.$relativeTime}) value. ${parserResult.info}`\n              );\n            }\n          }\n          constraintFieldName = `$${index++}:name`;\n          values.push(fieldName);\n        }\n        values.push(postgresValue);\n        patterns.push(`${constraintFieldName} ${pgComparator} $${index++}`);\n      }\n    });\n\n    if (initialPatternsLength === patterns.length) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `Postgres doesn't support this query type yet ${JSON.stringify(fieldValue)}`\n      );\n    }\n  }\n  values = values.map(transformValue);\n  return { pattern: patterns.join(' AND '), values, sorts };\n};\n\nexport class PostgresStorageAdapter implements StorageAdapter {\n  canSortOnJoinTables: boolean;\n  enableSchemaHooks: boolean;\n\n  // Private\n  _collectionPrefix: string;\n  _client: any;\n  _onchange: any;\n  _pgp: any;\n  _stream: any;\n  _uuid: any;\n\n  constructor({ uri, collectionPrefix = '', databaseOptions = {} }: any) {\n    this._collectionPrefix = collectionPrefix;\n    this.enableSchemaHooks = !!databaseOptions.enableSchemaHooks;\n    delete databaseOptions.enableSchemaHooks;\n\n    const { client, pgp } = createClient(uri, databaseOptions);\n    this._client = client;\n    this._onchange = () => {};\n    this._pgp = pgp;\n    this._uuid = uuidv4();\n    this.canSortOnJoinTables = false;\n  }\n\n  watch(callback: () => void): void {\n    this._onchange = callback;\n  }\n\n  //Note that analyze=true will run the query, executing INSERTS, DELETES, etc.\n  createExplainableQuery(query: string, analyze: boolean = false) {\n    if (analyze) {\n      return 'EXPLAIN (ANALYZE, FORMAT JSON) ' + query;\n    } else {\n      return 'EXPLAIN (FORMAT JSON) ' + query;\n    }\n  }\n\n  handleShutdown() {\n    if (this._stream) {\n      this._stream.done();\n      delete this._stream;\n    }\n    if (!this._client) {\n      return;\n    }\n    this._client.$pool.end();\n  }\n\n  async _listenToSchema() {\n    if (!this._stream && this.enableSchemaHooks) {\n      this._stream = await this._client.connect({ direct: true });\n      this._stream.client.on('notification', data => {\n        const payload = JSON.parse(data.payload);\n        if (payload.senderId !== this._uuid) {\n          this._onchange();\n        }\n      });\n      await this._stream.none('LISTEN $1~', 'schema.change');\n    }\n  }\n\n  _notifySchemaChange() {\n    if (this._stream) {\n      this._stream\n        .none('NOTIFY $1~, $2', ['schema.change', { senderId: this._uuid }])\n        .catch(error => {\n          console.log('Failed to Notify:', error); // unlikely to ever happen\n        });\n    }\n  }\n\n  async _ensureSchemaCollectionExists(conn: any) {\n    conn = conn || this._client;\n    await conn\n      .none(\n        'CREATE TABLE IF NOT EXISTS \"_SCHEMA\" ( \"className\" varChar(120), \"schema\" jsonb, \"isParseClass\" bool, PRIMARY KEY (\"className\") )'\n      )\n      .catch(error => {\n        throw error;\n      });\n  }\n\n  async classExists(name: string) {\n    return this._client.one(\n      'SELECT EXISTS (SELECT 1 FROM information_schema.tables WHERE table_name = $1)',\n      [name],\n      a => a.exists\n    );\n  }\n\n  async setClassLevelPermissions(className: string, CLPs: any) {\n    await this._client.task('set-class-level-permissions', async t => {\n      const values = [className, 'schema', 'classLevelPermissions', JSON.stringify(CLPs)];\n      await t.none(\n        `UPDATE \"_SCHEMA\" SET $2:name = json_object_set_key($2:name, $3::text, $4::jsonb) WHERE \"className\" = $1`,\n        values\n      );\n    });\n    this._notifySchemaChange();\n  }\n\n  async setIndexesWithSchemaFormat(\n    className: string,\n    submittedIndexes: any,\n    existingIndexes: any = {},\n    fields: any,\n    conn: ?any\n  ): Promise<void> {\n    conn = conn || this._client;\n    const self = this;\n    if (submittedIndexes === undefined) {\n      return Promise.resolve();\n    }\n    if (Object.keys(existingIndexes).length === 0) {\n      existingIndexes = { _id_: { _id: 1 } };\n    }\n    const deletedIndexes = [];\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        deletedIndexes.push(name);\n        delete existingIndexes[name];\n      } else {\n        Object.keys(field).forEach(key => {\n          if (!Object.prototype.hasOwnProperty.call(fields, key)) {\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    await conn.tx('set-indexes-with-schema-format', async t => {\n      if (insertedIndexes.length > 0) {\n        await self.createIndexes(className, insertedIndexes, t);\n      }\n      if (deletedIndexes.length > 0) {\n        await self.dropIndexes(className, deletedIndexes, t);\n      }\n      await t.none(\n        'UPDATE \"_SCHEMA\" SET $2:name = json_object_set_key($2:name, $3::text, $4::jsonb) WHERE \"className\" = $1',\n        [className, 'schema', 'indexes', JSON.stringify(existingIndexes)]\n      );\n    });\n    this._notifySchemaChange();\n  }\n\n  async createClass(className: string, schema: SchemaType, conn: ?any) {\n    conn = conn || this._client;\n    const parseSchema = await conn\n      .tx('create-class', async t => {\n        await this.createTable(className, schema, t);\n        await t.none(\n          'INSERT INTO \"_SCHEMA\" (\"className\", \"schema\", \"isParseClass\") VALUES ($<className>, $<schema>, true)',\n          { className, schema }\n        );\n        await this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields, t);\n        return toParseSchema(schema);\n      })\n      .catch(err => {\n        if (err.code === PostgresUniqueIndexViolationError && err.detail.includes(className)) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, `Class ${className} already exists.`);\n        }\n        throw err;\n      });\n    this._notifySchemaChange();\n    return parseSchema;\n  }\n\n  // Just create a table, do not insert in schema\n  async createTable(className: string, schema: SchemaType, conn: any) {\n    conn = conn || this._client;\n    debug('createTable');\n    const valuesArray = [];\n    const patternsArray = [];\n    const fields = Object.assign({}, schema.fields);\n    if (className === '_User') {\n      fields._email_verify_token_expires_at = { type: 'Date' };\n      fields._email_verify_token = { type: 'String' };\n      fields._account_lockout_expires_at = { type: 'Date' };\n      fields._failed_login_count = { type: 'Number' };\n      fields._perishable_token = { type: 'String' };\n      fields._perishable_token_expires_at = { type: 'Date' };\n      fields._password_changed_at = { type: 'Date' };\n      fields._password_history = { type: 'Array' };\n    }\n    let index = 2;\n    const relations = [];\n    Object.keys(fields).forEach(fieldName => {\n      const parseType = fields[fieldName];\n      // Skip when it's a relation\n      // We'll create the tables later\n      if (parseType.type === 'Relation') {\n        relations.push(fieldName);\n        return;\n      }\n      if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {\n        parseType.contents = { type: 'String' };\n      }\n      valuesArray.push(fieldName);\n      valuesArray.push(parseTypeToPostgresType(parseType));\n      patternsArray.push(`$${index}:name $${index + 1}:raw`);\n      if (fieldName === 'objectId') {\n        patternsArray.push(`PRIMARY KEY ($${index}:name)`);\n      }\n      index = index + 2;\n    });\n    const qs = `CREATE TABLE IF NOT EXISTS $1:name (${patternsArray.join()})`;\n    const values = [className, ...valuesArray];\n\n    return conn.task('create-table', async t => {\n      try {\n        await t.none(qs, values);\n      } catch (error) {\n        if (error.code !== PostgresDuplicateRelationError) {\n          throw error;\n        }\n        // ELSE: Table already exists, must have been created by a different request. Ignore the error.\n      }\n      await t.tx('create-table-tx', tx => {\n        return tx.batch(\n          relations.map(fieldName => {\n            return tx.none(\n              'CREATE TABLE IF NOT EXISTS $<joinTable:name> (\"relatedId\" varChar(120), \"owningId\" varChar(120), PRIMARY KEY(\"relatedId\", \"owningId\") )',\n              { joinTable: `_Join:${fieldName}:${className}` }\n            );\n          })\n        );\n      });\n    });\n  }\n\n  async schemaUpgrade(className: string, schema: SchemaType, conn: any) {\n    debug('schemaUpgrade');\n    conn = conn || this._client;\n    const self = this;\n\n    await conn.task('schema-upgrade', async t => {\n      const columns = await t.map(\n        'SELECT column_name FROM information_schema.columns WHERE table_name = $<className>',\n        { className },\n        a => a.column_name\n      );\n      const newColumns = Object.keys(schema.fields)\n        .filter(item => columns.indexOf(item) === -1)\n        .map(fieldName => self.addFieldIfNotExists(className, fieldName, schema.fields[fieldName]));\n\n      await t.batch(newColumns);\n    });\n  }\n\n  async addFieldIfNotExists(className: string, fieldName: string, type: any) {\n    // TODO: Must be revised for invalid logic...\n    debug('addFieldIfNotExists');\n    const self = this;\n    await this._client.tx('add-field-if-not-exists', async t => {\n      if (type.type !== 'Relation') {\n        try {\n          await t.none(\n            'ALTER TABLE $<className:name> ADD COLUMN IF NOT EXISTS $<fieldName:name> $<postgresType:raw>',\n            {\n              className,\n              fieldName,\n              postgresType: parseTypeToPostgresType(type),\n            }\n          );\n        } catch (error) {\n          if (error.code === PostgresRelationDoesNotExistError) {\n            return self.createClass(className, { fields: { [fieldName]: type } }, t);\n          }\n          if (error.code !== PostgresDuplicateColumnError) {\n            throw error;\n          }\n          // Column already exists, created by other request. Carry on to see if it's the right type.\n        }\n      } else {\n        await t.none(\n          'CREATE TABLE IF NOT EXISTS $<joinTable:name> (\"relatedId\" varChar(120), \"owningId\" varChar(120), PRIMARY KEY(\"relatedId\", \"owningId\") )',\n          { joinTable: `_Join:${fieldName}:${className}` }\n        );\n      }\n\n      const result = await t.any(\n        'SELECT \"schema\" FROM \"_SCHEMA\" WHERE \"className\" = $<className> and (\"schema\"::json->\\'fields\\'->$<fieldName>) is not null',\n        { className, fieldName }\n      );\n\n      if (result[0]) {\n        throw 'Attempted to add a field that already exists';\n      } else {\n        const path = `{fields,${fieldName}}`;\n        await t.none(\n          'UPDATE \"_SCHEMA\" SET \"schema\"=jsonb_set(\"schema\", $<path>, $<type>)  WHERE \"className\"=$<className>',\n          { path, type, className }\n        );\n      }\n    });\n    this._notifySchemaChange();\n  }\n\n  async updateFieldOptions(className: string, fieldName: string, type: any) {\n    await this._client.tx('update-schema-field-options', async t => {\n      const path = `{fields,${fieldName}}`;\n      await t.none(\n        'UPDATE \"_SCHEMA\" SET \"schema\"=jsonb_set(\"schema\", $<path>, $<type>)  WHERE \"className\"=$<className>',\n        { path, type, className }\n      );\n    });\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  async deleteClass(className: string) {\n    const operations = [\n      { query: `DROP TABLE IF EXISTS $1:name`, values: [className] },\n      {\n        query: `DELETE FROM \"_SCHEMA\" WHERE \"className\" = $1`,\n        values: [className],\n      },\n    ];\n    const response = await this._client\n      .tx(t => t.none(this._pgp.helpers.concat(operations)))\n      .then(() => className.indexOf('_Join:') != 0); // resolves with false when _Join table\n\n    this._notifySchemaChange();\n    return response;\n  }\n\n  // Delete all data known to this adapter. Used for testing.\n  async deleteAllClasses() {\n    const now = new Date().getTime();\n    const helpers = this._pgp.helpers;\n    debug('deleteAllClasses');\n\n    await this._client\n      .task('delete-all-classes', async t => {\n        try {\n          const results = await t.any('SELECT * FROM \"_SCHEMA\"');\n          const joins = results.reduce((list: Array<string>, schema: any) => {\n            return list.concat(joinTablesForSchema(schema.schema));\n          }, []);\n          const classes = [\n            '_SCHEMA',\n            '_PushStatus',\n            '_JobStatus',\n            '_JobSchedule',\n            '_Hooks',\n            '_GlobalConfig',\n            '_GraphQLConfig',\n            '_Audience',\n            '_Idempotency',\n            ...results.map(result => result.className),\n            ...joins,\n          ];\n          const queries = classes.map(className => ({\n            query: 'DROP TABLE IF EXISTS $<className:name>',\n            values: { className },\n          }));\n          await t.tx(tx => tx.none(helpers.concat(queries)));\n        } catch (error) {\n          if (error.code !== PostgresRelationDoesNotExistError) {\n            throw error;\n          }\n          // No _SCHEMA collection. Don't delete anything.\n        }\n      })\n      .then(() => {\n        debug(`deleteAllClasses done in ${new Date().getTime() - now}`);\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  // 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  async deleteFields(className: string, schema: SchemaType, fieldNames: string[]): Promise<void> {\n    debug('deleteFields');\n    fieldNames = fieldNames.reduce((list: Array<string>, fieldName: string) => {\n      const field = schema.fields[fieldName];\n      if (field.type !== 'Relation') {\n        list.push(fieldName);\n      }\n      delete schema.fields[fieldName];\n      return list;\n    }, []);\n\n    const values = [className, ...fieldNames];\n    const columns = fieldNames\n      .map((name, idx) => {\n        return `$${idx + 2}:name`;\n      })\n      .join(', DROP COLUMN');\n\n    await this._client.tx('delete-fields', async t => {\n      await t.none('UPDATE \"_SCHEMA\" SET \"schema\" = $<schema> WHERE \"className\" = $<className>', {\n        schema,\n        className,\n      });\n      if (values.length > 1) {\n        await t.none(`ALTER TABLE $1:name DROP COLUMN IF EXISTS ${columns}`, values);\n      }\n    });\n    this._notifySchemaChange();\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  async getAllClasses() {\n    return this._client.task('get-all-classes', async t => {\n      return await t.map('SELECT * FROM \"_SCHEMA\"', null, row =>\n        toParseSchema({ className: row.className, ...row.schema })\n      );\n    });\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  async getClass(className: string) {\n    debug('getClass');\n    return this._client\n      .any('SELECT * FROM \"_SCHEMA\" WHERE \"className\" = $<className>', {\n        className,\n      })\n      .then(result => {\n        if (result.length !== 1) {\n          throw undefined;\n        }\n        return result[0].schema;\n      })\n      .then(toParseSchema);\n  }\n\n  // TODO: remove the mongo format dependency in the return value\n  async createObject(\n    className: string,\n    schema: SchemaType,\n    object: any,\n    transactionalSession: ?any\n  ) {\n    debug('createObject');\n    let columnsArray = [];\n    const valuesArray = [];\n    schema = toPostgresSchema(schema);\n    const geoPoints = {};\n\n    object = handleDotFields(object);\n\n    validateKeys(object);\n\n    Object.keys(object).forEach(fieldName => {\n      if (object[fieldName] === null) {\n        return;\n      }\n      var authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);\n      const authDataAlreadyExists = !!object.authData;\n      if (authDataMatch) {\n        var provider = authDataMatch[1];\n        object['authData'] = object['authData'] || {};\n        object['authData'][provider] = object[fieldName];\n        delete object[fieldName];\n        fieldName = 'authData';\n        // Avoid adding authData multiple times to the query\n        if (authDataAlreadyExists) {\n          return;\n        }\n      }\n\n      columnsArray.push(fieldName);\n      if (!schema.fields[fieldName] && className === '_User') {\n        if (\n          fieldName === '_email_verify_token' ||\n          fieldName === '_failed_login_count' ||\n          fieldName === '_perishable_token' ||\n          fieldName === '_password_history'\n        ) {\n          valuesArray.push(object[fieldName]);\n        }\n\n        if (fieldName === '_email_verify_token_expires_at') {\n          if (object[fieldName]) {\n            valuesArray.push(object[fieldName].iso);\n          } else {\n            valuesArray.push(null);\n          }\n        }\n\n        if (\n          fieldName === '_account_lockout_expires_at' ||\n          fieldName === '_perishable_token_expires_at' ||\n          fieldName === '_password_changed_at'\n        ) {\n          if (object[fieldName]) {\n            valuesArray.push(object[fieldName].iso);\n          } else {\n            valuesArray.push(null);\n          }\n        }\n        return;\n      }\n      switch (schema.fields[fieldName].type) {\n        case 'Date':\n          if (object[fieldName]) {\n            valuesArray.push(object[fieldName].iso);\n          } else {\n            valuesArray.push(null);\n          }\n          break;\n        case 'Pointer':\n          valuesArray.push(object[fieldName].objectId);\n          break;\n        case 'Array':\n          if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {\n            valuesArray.push(object[fieldName]);\n          } else {\n            valuesArray.push(JSON.stringify(object[fieldName]));\n          }\n          break;\n        case 'Object':\n        case 'Bytes':\n        case 'String':\n        case 'Number':\n        case 'Boolean':\n          valuesArray.push(object[fieldName]);\n          break;\n        case 'File':\n          valuesArray.push(object[fieldName].name);\n          break;\n        case 'Polygon': {\n          const value = convertPolygonToSQL(object[fieldName].coordinates);\n          valuesArray.push(value);\n          break;\n        }\n        case 'GeoPoint':\n          // pop the point and process later\n          geoPoints[fieldName] = object[fieldName];\n          columnsArray.pop();\n          break;\n        default:\n          throw `Type ${schema.fields[fieldName].type} not supported yet`;\n      }\n    });\n\n    columnsArray = columnsArray.concat(Object.keys(geoPoints));\n    const initialValues = valuesArray.map((val, index) => {\n      let termination = '';\n      const fieldName = columnsArray[index];\n      if (['_rperm', '_wperm'].indexOf(fieldName) >= 0) {\n        termination = '::text[]';\n      } else if (schema.fields[fieldName] && schema.fields[fieldName].type === 'Array') {\n        termination = '::jsonb';\n      }\n      return `$${index + 2 + columnsArray.length}${termination}`;\n    });\n    const geoPointsInjects = Object.keys(geoPoints).map(key => {\n      const value = geoPoints[key];\n      valuesArray.push(value.longitude, value.latitude);\n      const l = valuesArray.length + columnsArray.length;\n      return `POINT($${l}, $${l + 1})`;\n    });\n\n    const columnsPattern = columnsArray.map((col, index) => `$${index + 2}:name`).join();\n    const valuesPattern = initialValues.concat(geoPointsInjects).join();\n\n    const qs = `INSERT INTO $1:name (${columnsPattern}) VALUES (${valuesPattern})`;\n    const values = [className, ...columnsArray, ...valuesArray];\n    const promise = (transactionalSession ? transactionalSession.t : this._client)\n      .none(qs, values)\n      .then(() => ({ ops: [object] }))\n      .catch(error => {\n        if (error.code === PostgresUniqueIndexViolationError) {\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.constraint) {\n            const matches = error.constraint.match(/unique_([a-zA-Z]+)/);\n            if (matches && Array.isArray(matches)) {\n              err.userInfo = { duplicated_field: matches[1] };\n            }\n          }\n          error = err;\n        }\n        throw error;\n      });\n    if (transactionalSession) {\n      transactionalSession.batch.push(promise);\n    }\n    return promise;\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  async deleteObjectsByQuery(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    transactionalSession: ?any\n  ) {\n    debug('deleteObjectsByQuery');\n    const values = [className];\n    const index = 2;\n    const where = buildWhereClause({\n      schema,\n      index,\n      query,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n    if (Object.keys(query).length === 0) {\n      where.pattern = 'TRUE';\n    }\n    const qs = `WITH deleted AS (DELETE FROM $1:name WHERE ${where.pattern} RETURNING *) SELECT count(*) FROM deleted`;\n    const promise = (transactionalSession ? transactionalSession.t : this._client)\n      .one(qs, values, a => +a.count)\n      .then(count => {\n        if (count === 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        } else {\n          return count;\n        }\n      })\n      .catch(error => {\n        if (error.code !== PostgresRelationDoesNotExistError) {\n          throw error;\n        }\n        // ELSE: Don't delete anything if doesn't exist\n      });\n    if (transactionalSession) {\n      transactionalSession.batch.push(promise);\n    }\n    return promise;\n  }\n  // Return value not currently well specified.\n  async findOneAndUpdate(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ): Promise<any> {\n    debug('findOneAndUpdate');\n    return this.updateObjectsByQuery(className, schema, query, update, transactionalSession).then(\n      val => val[0]\n    );\n  }\n\n  // Apply the update to all objects that match the given Parse Query.\n  async updateObjectsByQuery(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    update: any,\n    transactionalSession: ?any\n  ): Promise<[any]> {\n    debug('updateObjectsByQuery');\n    const updatePatterns = [];\n    const values = [className];\n    let index = 2;\n    schema = toPostgresSchema(schema);\n\n    const originalUpdate = { ...update };\n\n    // Set flag for dot notation fields\n    const dotNotationOptions = {};\n    Object.keys(update).forEach(fieldName => {\n      if (fieldName.indexOf('.') > -1) {\n        const components = fieldName.split('.');\n        const first = components.shift();\n        dotNotationOptions[first] = true;\n      } else {\n        dotNotationOptions[fieldName] = false;\n      }\n    });\n    update = handleDotFields(update);\n    // Resolve authData first,\n    // So we don't end up with multiple key updates\n    for (const fieldName in update) {\n      const authDataMatch = fieldName.match(/^_auth_data_([a-zA-Z0-9_]+)$/);\n      if (authDataMatch) {\n        var provider = authDataMatch[1];\n        const value = update[fieldName];\n        delete update[fieldName];\n        update['authData'] = update['authData'] || {};\n        update['authData'][provider] = value;\n      }\n    }\n\n    for (const fieldName in update) {\n      const fieldValue = update[fieldName];\n      // Drop any undefined values.\n      if (typeof fieldValue === 'undefined') {\n        delete update[fieldName];\n      } else if (fieldValue === null) {\n        updatePatterns.push(`$${index}:name = NULL`);\n        values.push(fieldName);\n        index += 1;\n      } else if (fieldName == 'authData') {\n        // This recursively sets the json_object\n        // Only 1 level deep\n        const generate = (jsonb: string, key: string, value: any) => {\n          return `json_object_set_key(COALESCE(${jsonb}, '{}'::jsonb), ${key}, ${value})::jsonb`;\n        };\n        const lastKey = `$${index}:name`;\n        const fieldNameIndex = index;\n        index += 1;\n        values.push(fieldName);\n        const update = Object.keys(fieldValue).reduce((lastKey: string, key: string) => {\n          const str = generate(lastKey, `$${index}::text`, `$${index + 1}::jsonb`);\n          index += 2;\n          let value = fieldValue[key];\n          if (value) {\n            if (value.__op === 'Delete') {\n              value = null;\n            } else {\n              value = JSON.stringify(value);\n            }\n          }\n          values.push(key, value);\n          return str;\n        }, lastKey);\n        updatePatterns.push(`$${fieldNameIndex}:name = ${update}`);\n      } else if (fieldValue.__op === 'Increment') {\n        updatePatterns.push(`$${index}:name = COALESCE($${index}:name, 0) + $${index + 1}`);\n        values.push(fieldName, fieldValue.amount);\n        index += 2;\n      } else if (fieldValue.__op === 'Add') {\n        updatePatterns.push(\n          `$${index}:name = array_add(COALESCE($${index}:name, '[]'::jsonb), $${index + 1}::jsonb)`\n        );\n        values.push(fieldName, JSON.stringify(fieldValue.objects));\n        index += 2;\n      } else if (fieldValue.__op === 'Delete') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, null);\n        index += 2;\n      } else if (fieldValue.__op === 'Remove') {\n        updatePatterns.push(\n          `$${index}:name = array_remove(COALESCE($${index}:name, '[]'::jsonb), $${\n            index + 1\n          }::jsonb)`\n        );\n        values.push(fieldName, JSON.stringify(fieldValue.objects));\n        index += 2;\n      } else if (fieldValue.__op === 'AddUnique') {\n        updatePatterns.push(\n          `$${index}:name = array_add_unique(COALESCE($${index}:name, '[]'::jsonb), $${\n            index + 1\n          }::jsonb)`\n        );\n        values.push(fieldName, JSON.stringify(fieldValue.objects));\n        index += 2;\n      } else if (fieldName === 'updatedAt') {\n        //TODO: stop special casing this. It should check for __type === 'Date' and use .iso\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (typeof fieldValue === 'string') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (typeof fieldValue === 'boolean') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (fieldValue.__type === 'Pointer') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue.objectId);\n        index += 2;\n      } else if (fieldValue.__type === 'Date') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, toPostgresValue(fieldValue));\n        index += 2;\n      } else if (fieldValue instanceof Date) {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (fieldValue.__type === 'File') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, toPostgresValue(fieldValue));\n        index += 2;\n      } else if (fieldValue.__type === 'GeoPoint') {\n        updatePatterns.push(`$${index}:name = POINT($${index + 1}, $${index + 2})`);\n        values.push(fieldName, fieldValue.longitude, fieldValue.latitude);\n        index += 3;\n      } else if (fieldValue.__type === 'Polygon') {\n        const value = convertPolygonToSQL(fieldValue.coordinates);\n        updatePatterns.push(`$${index}:name = $${index + 1}::polygon`);\n        values.push(fieldName, value);\n        index += 2;\n      } else if (fieldValue.__type === 'Relation') {\n        // noop\n      } else if (typeof fieldValue === 'number') {\n        updatePatterns.push(`$${index}:name = $${index + 1}`);\n        values.push(fieldName, fieldValue);\n        index += 2;\n      } else if (\n        typeof fieldValue === 'object' &&\n        schema.fields[fieldName] &&\n        schema.fields[fieldName].type === 'Object'\n      ) {\n        // Gather keys to increment\n        const keysToIncrement = Object.keys(originalUpdate)\n          .filter(k => {\n            // choose top level fields that have a delete operation set\n            // Note that Object.keys is iterating over the **original** update object\n            // and that some of the keys of the original update could be null or undefined:\n            // (See the above check `if (fieldValue === null || typeof fieldValue == \"undefined\")`)\n            const value = originalUpdate[k];\n            return (\n              value &&\n              value.__op === 'Increment' &&\n              k.split('.').length === 2 &&\n              k.split('.')[0] === fieldName\n            );\n          })\n          .map(k => k.split('.')[1]);\n\n        let incrementPatterns = '';\n        if (keysToIncrement.length > 0) {\n          incrementPatterns =\n            ' || ' +\n            keysToIncrement\n              .map(c => {\n                const amount = fieldValue[c].amount;\n                return `CONCAT('{\"${c}\":', COALESCE($${index}:name->>'${c}','0')::int + ${amount}, '}')::jsonb`;\n              })\n              .join(' || ');\n          // Strip the keys\n          keysToIncrement.forEach(key => {\n            delete fieldValue[key];\n          });\n        }\n\n        const keysToDelete: Array<string> = Object.keys(originalUpdate)\n          .filter(k => {\n            // choose top level fields that have a delete operation set.\n            const value = originalUpdate[k];\n            return (\n              value &&\n              value.__op === 'Delete' &&\n              k.split('.').length === 2 &&\n              k.split('.')[0] === fieldName\n            );\n          })\n          .map(k => k.split('.')[1]);\n\n        const deletePatterns = keysToDelete.reduce((p: string, c: string, i: number) => {\n          return p + ` - '$${index + 1 + i}:value'`;\n        }, '');\n        // Override Object\n        let updateObject = \"'{}'::jsonb\";\n\n        if (dotNotationOptions[fieldName]) {\n          // Merge Object\n          updateObject = `COALESCE($${index}:name, '{}'::jsonb)`;\n        }\n        updatePatterns.push(\n          `$${index}:name = (${updateObject} ${deletePatterns} ${incrementPatterns} || $${\n            index + 1 + keysToDelete.length\n          }::jsonb )`\n        );\n        values.push(fieldName, ...keysToDelete, JSON.stringify(fieldValue));\n        index += 2 + keysToDelete.length;\n      } else if (\n        Array.isArray(fieldValue) &&\n        schema.fields[fieldName] &&\n        schema.fields[fieldName].type === 'Array'\n      ) {\n        const expectedType = parseTypeToPostgresType(schema.fields[fieldName]);\n        if (expectedType === 'text[]') {\n          updatePatterns.push(`$${index}:name = $${index + 1}::text[]`);\n          values.push(fieldName, fieldValue);\n          index += 2;\n        } else {\n          updatePatterns.push(`$${index}:name = $${index + 1}::jsonb`);\n          values.push(fieldName, JSON.stringify(fieldValue));\n          index += 2;\n        }\n      } else {\n        debug('Not supported update', { fieldName, fieldValue });\n        return Promise.reject(\n          new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            `Postgres doesn't support update ${JSON.stringify(fieldValue)} yet`\n          )\n        );\n      }\n    }\n\n    const where = buildWhereClause({\n      schema,\n      index,\n      query,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n\n    const whereClause = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    const qs = `UPDATE $1:name SET ${updatePatterns.join()} ${whereClause} RETURNING *`;\n    const promise = (transactionalSession ? transactionalSession.t : this._client).any(qs, values);\n    if (transactionalSession) {\n      transactionalSession.batch.push(promise);\n    }\n    return promise;\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    debug('upsertOneObject');\n    const createValue = Object.assign({}, query, update);\n    return this.createObject(className, schema, createValue, transactionalSession).catch(error => {\n      // ignore duplicate value errors as it's upsert\n      if (error.code !== Parse.Error.DUPLICATE_VALUE) {\n        throw error;\n      }\n      return this.findOneAndUpdate(className, schema, query, update, transactionalSession);\n    });\n  }\n\n  find(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    { skip, limit, sort, keys, caseInsensitive, explain }: QueryOptions\n  ) {\n    debug('find');\n    const hasLimit = limit !== undefined;\n    const hasSkip = skip !== undefined;\n    let values = [className];\n    const where = buildWhereClause({\n      schema,\n      query,\n      index: 2,\n      caseInsensitive,\n    });\n    values.push(...where.values);\n    const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    const limitPattern = hasLimit ? `LIMIT $${values.length + 1}` : '';\n    if (hasLimit) {\n      values.push(limit);\n    }\n    const skipPattern = hasSkip ? `OFFSET $${values.length + 1}` : '';\n    if (hasSkip) {\n      values.push(skip);\n    }\n\n    let sortPattern = '';\n    if (sort) {\n      const sortCopy: any = sort;\n      const sorting = Object.keys(sort)\n        .map(key => {\n          const transformKey = transformDotFieldToComponents(key).join('->');\n          // Using $idx pattern gives:  non-integer constant in ORDER BY\n          if (sortCopy[key] === 1) {\n            return `${transformKey} ASC`;\n          }\n          return `${transformKey} DESC`;\n        })\n        .join();\n      sortPattern = sort !== undefined && Object.keys(sort).length > 0 ? `ORDER BY ${sorting}` : '';\n    }\n    if (where.sorts && Object.keys((where.sorts: any)).length > 0) {\n      sortPattern = `ORDER BY ${where.sorts.join()}`;\n    }\n\n    let columns = '*';\n    if (keys) {\n      // Exclude empty keys\n      // Replace ACL by it's keys\n      keys = keys.reduce((memo, key) => {\n        if (key === 'ACL') {\n          memo.push('_rperm');\n          memo.push('_wperm');\n        } else if (\n          key.length > 0 &&\n          // Remove selected field not referenced in the schema\n          // Relation is not a column in postgres\n          // $score is a Parse special field and is also not a column\n          ((schema.fields[key] && schema.fields[key].type !== 'Relation') || key === '$score')\n        ) {\n          memo.push(key);\n        }\n        return memo;\n      }, []);\n      columns = keys\n        .map((key, index) => {\n          if (key === '$score') {\n            return `ts_rank_cd(to_tsvector($${2}, $${3}:name), to_tsquery($${4}, $${5}), 32) as score`;\n          }\n          return `$${index + values.length + 1}:name`;\n        })\n        .join();\n      values = values.concat(keys);\n    }\n\n    const originalQuery = `SELECT ${columns} FROM $1:name ${wherePattern} ${sortPattern} ${limitPattern} ${skipPattern}`;\n    const qs = explain ? this.createExplainableQuery(originalQuery) : originalQuery;\n    return this._client\n      .any(qs, values)\n      .catch(error => {\n        // Query on non existing table, don't crash\n        if (error.code !== PostgresRelationDoesNotExistError) {\n          throw error;\n        }\n        return [];\n      })\n      .then(results => {\n        if (explain) {\n          return results;\n        }\n        return results.map(object => this.postgresObjectToParseObject(className, object, schema));\n      });\n  }\n\n  // Converts from a postgres-format object to a REST-format object.\n  // Does not strip out anything based on a lack of authentication.\n  postgresObjectToParseObject(className: string, object: any, schema: any) {\n    Object.keys(schema.fields).forEach(fieldName => {\n      if (schema.fields[fieldName].type === 'Pointer' && object[fieldName]) {\n        object[fieldName] = {\n          objectId: object[fieldName],\n          __type: 'Pointer',\n          className: schema.fields[fieldName].targetClass,\n        };\n      }\n      if (schema.fields[fieldName].type === 'Relation') {\n        object[fieldName] = {\n          __type: 'Relation',\n          className: schema.fields[fieldName].targetClass,\n        };\n      }\n      if (object[fieldName] && schema.fields[fieldName].type === 'GeoPoint') {\n        object[fieldName] = {\n          __type: 'GeoPoint',\n          latitude: object[fieldName].y,\n          longitude: object[fieldName].x,\n        };\n      }\n      if (object[fieldName] && schema.fields[fieldName].type === 'Polygon') {\n        let coords = object[fieldName];\n        coords = coords.substr(2, coords.length - 4).split('),(');\n        coords = coords.map(point => {\n          return [parseFloat(point.split(',')[1]), parseFloat(point.split(',')[0])];\n        });\n        object[fieldName] = {\n          __type: 'Polygon',\n          coordinates: coords,\n        };\n      }\n      if (object[fieldName] && schema.fields[fieldName].type === 'File') {\n        object[fieldName] = {\n          __type: 'File',\n          name: object[fieldName],\n        };\n      }\n    });\n    //TODO: remove this reliance on the mongo format. DB adapter shouldn't know there is a difference between created at and any other date field.\n    if (object.createdAt) {\n      object.createdAt = object.createdAt.toISOString();\n    }\n    if (object.updatedAt) {\n      object.updatedAt = object.updatedAt.toISOString();\n    }\n    if (object.expiresAt) {\n      object.expiresAt = {\n        __type: 'Date',\n        iso: object.expiresAt.toISOString(),\n      };\n    }\n    if (object._email_verify_token_expires_at) {\n      object._email_verify_token_expires_at = {\n        __type: 'Date',\n        iso: object._email_verify_token_expires_at.toISOString(),\n      };\n    }\n    if (object._account_lockout_expires_at) {\n      object._account_lockout_expires_at = {\n        __type: 'Date',\n        iso: object._account_lockout_expires_at.toISOString(),\n      };\n    }\n    if (object._perishable_token_expires_at) {\n      object._perishable_token_expires_at = {\n        __type: 'Date',\n        iso: object._perishable_token_expires_at.toISOString(),\n      };\n    }\n    if (object._password_changed_at) {\n      object._password_changed_at = {\n        __type: 'Date',\n        iso: object._password_changed_at.toISOString(),\n      };\n    }\n\n    for (const fieldName in object) {\n      if (object[fieldName] === null) {\n        delete object[fieldName];\n      }\n      if (object[fieldName] instanceof Date) {\n        object[fieldName] = {\n          __type: 'Date',\n          iso: object[fieldName].toISOString(),\n        };\n      }\n    }\n\n    return object;\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  async ensureUniqueness(className: string, schema: SchemaType, fieldNames: string[]) {\n    const constraintName = `${className}_unique_${fieldNames.sort().join('_')}`;\n    const constraintPatterns = fieldNames.map((fieldName, index) => `$${index + 3}:name`);\n    const qs = `CREATE UNIQUE INDEX IF NOT EXISTS $2:name ON $1:name(${constraintPatterns.join()})`;\n    return this._client.none(qs, [className, constraintName, ...fieldNames]).catch(error => {\n      if (error.code === PostgresDuplicateRelationError && error.message.includes(constraintName)) {\n        // Index already exists. Ignore error.\n      } else if (\n        error.code === PostgresUniqueIndexViolationError &&\n        error.message.includes(constraintName)\n      ) {\n        // Cast the error into the proper parse error\n        throw new Parse.Error(\n          Parse.Error.DUPLICATE_VALUE,\n          'A duplicate value for a field with unique values was provided'\n        );\n      } else {\n        throw error;\n      }\n    });\n  }\n\n  // Executes a count.\n  async count(\n    className: string,\n    schema: SchemaType,\n    query: QueryType,\n    readPreference?: string,\n    estimate?: boolean = true\n  ) {\n    debug('count');\n    const values = [className];\n    const where = buildWhereClause({\n      schema,\n      query,\n      index: 2,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n\n    const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    let qs = '';\n\n    if (where.pattern.length > 0 || !estimate) {\n      qs = `SELECT count(*) FROM $1:name ${wherePattern}`;\n    } else {\n      qs = 'SELECT reltuples AS approximate_row_count FROM pg_class WHERE relname = $1';\n    }\n\n    return this._client\n      .one(qs, values, a => {\n        if (a.approximate_row_count == null || a.approximate_row_count == -1) {\n          return !isNaN(+a.count) ? +a.count : 0;\n        } else {\n          return +a.approximate_row_count;\n        }\n      })\n      .catch(error => {\n        if (error.code !== PostgresRelationDoesNotExistError) {\n          throw error;\n        }\n        return 0;\n      });\n  }\n\n  async distinct(className: string, schema: SchemaType, query: QueryType, fieldName: string) {\n    debug('distinct');\n    let field = fieldName;\n    let column = fieldName;\n    const isNested = fieldName.indexOf('.') >= 0;\n    if (isNested) {\n      field = transformDotFieldToComponents(fieldName).join('->');\n      column = fieldName.split('.')[0];\n    }\n    const isArrayField =\n      schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Array';\n    const isPointerField =\n      schema.fields && schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';\n    const values = [field, column, className];\n    const where = buildWhereClause({\n      schema,\n      query,\n      index: 4,\n      caseInsensitive: false,\n    });\n    values.push(...where.values);\n\n    const wherePattern = where.pattern.length > 0 ? `WHERE ${where.pattern}` : '';\n    const transformer = isArrayField ? 'jsonb_array_elements' : 'ON';\n    let qs = `SELECT DISTINCT ${transformer}($1:name) $2:name FROM $3:name ${wherePattern}`;\n    if (isNested) {\n      qs = `SELECT DISTINCT ${transformer}($1:raw) $2:raw FROM $3:name ${wherePattern}`;\n    }\n    return this._client\n      .any(qs, values)\n      .catch(error => {\n        if (error.code === PostgresMissingColumnError) {\n          return [];\n        }\n        throw error;\n      })\n      .then(results => {\n        if (!isNested) {\n          results = results.filter(object => object[field] !== null);\n          return results.map(object => {\n            if (!isPointerField) {\n              return object[field];\n            }\n            return {\n              __type: 'Pointer',\n              className: schema.fields[fieldName].targetClass,\n              objectId: object[field],\n            };\n          });\n        }\n        const child = fieldName.split('.')[1];\n        return results.map(object => object[column][child]);\n      })\n      .then(results =>\n        results.map(object => this.postgresObjectToParseObject(className, object, schema))\n      );\n  }\n\n  async aggregate(\n    className: string,\n    schema: any,\n    pipeline: any,\n    readPreference: ?string,\n    hint: ?mixed,\n    explain?: boolean\n  ) {\n    debug('aggregate');\n    const values = [className];\n    let index: number = 2;\n    let columns: string[] = [];\n    let countField = null;\n    let groupValues = null;\n    let wherePattern = '';\n    let limitPattern = '';\n    let skipPattern = '';\n    let sortPattern = '';\n    let groupPattern = '';\n    for (let i = 0; i < pipeline.length; i += 1) {\n      const stage = pipeline[i];\n      if (stage.$group) {\n        for (const field in stage.$group) {\n          const value = stage.$group[field];\n          if (value === null || value === undefined) {\n            continue;\n          }\n          if (field === '_id' && typeof value === 'string' && value !== '') {\n            columns.push(`$${index}:name AS \"objectId\"`);\n            groupPattern = `GROUP BY $${index}:name`;\n            values.push(transformAggregateField(value));\n            index += 1;\n            continue;\n          }\n          if (field === '_id' && typeof value === 'object' && Object.keys(value).length !== 0) {\n            groupValues = value;\n            const groupByFields = [];\n            for (const alias in value) {\n              if (typeof value[alias] === 'string' && value[alias]) {\n                const source = transformAggregateField(value[alias]);\n                if (!groupByFields.includes(`\"${source}\"`)) {\n                  groupByFields.push(`\"${source}\"`);\n                }\n                values.push(source, alias);\n                columns.push(`$${index}:name AS $${index + 1}:name`);\n                index += 2;\n              } else {\n                const operation = Object.keys(value[alias])[0];\n                const source = transformAggregateField(value[alias][operation]);\n                if (mongoAggregateToPostgres[operation]) {\n                  if (!groupByFields.includes(`\"${source}\"`)) {\n                    groupByFields.push(`\"${source}\"`);\n                  }\n                  columns.push(\n                    `EXTRACT(${\n                      mongoAggregateToPostgres[operation]\n                    } FROM $${index}:name AT TIME ZONE 'UTC')::integer AS $${index + 1}:name`\n                  );\n                  values.push(source, alias);\n                  index += 2;\n                }\n              }\n            }\n            groupPattern = `GROUP BY $${index}:raw`;\n            values.push(groupByFields.join());\n            index += 1;\n            continue;\n          }\n          if (typeof value === 'object') {\n            if (value.$sum) {\n              if (typeof value.$sum === 'string') {\n                columns.push(`SUM($${index}:name) AS $${index + 1}:name`);\n                values.push(transformAggregateField(value.$sum), field);\n                index += 2;\n              } else {\n                countField = field;\n                columns.push(`COUNT(*) AS $${index}:name`);\n                values.push(field);\n                index += 1;\n              }\n            }\n            if (value.$max) {\n              columns.push(`MAX($${index}:name) AS $${index + 1}:name`);\n              values.push(transformAggregateField(value.$max), field);\n              index += 2;\n            }\n            if (value.$min) {\n              columns.push(`MIN($${index}:name) AS $${index + 1}:name`);\n              values.push(transformAggregateField(value.$min), field);\n              index += 2;\n            }\n            if (value.$avg) {\n              columns.push(`AVG($${index}:name) AS $${index + 1}:name`);\n              values.push(transformAggregateField(value.$avg), field);\n              index += 2;\n            }\n          }\n        }\n      } else {\n        columns.push('*');\n      }\n      if (stage.$project) {\n        if (columns.includes('*')) {\n          columns = [];\n        }\n        for (const field in stage.$project) {\n          const value = stage.$project[field];\n          if (value === 1 || value === true) {\n            columns.push(`$${index}:name`);\n            values.push(field);\n            index += 1;\n          }\n        }\n      }\n      if (stage.$match) {\n        const patterns = [];\n        const orOrAnd = Object.prototype.hasOwnProperty.call(stage.$match, '$or')\n          ? ' OR '\n          : ' AND ';\n\n        if (stage.$match.$or) {\n          const collapse = {};\n          stage.$match.$or.forEach(element => {\n            for (const key in element) {\n              collapse[key] = element[key];\n            }\n          });\n          stage.$match = collapse;\n        }\n        for (let field in stage.$match) {\n          const value = stage.$match[field];\n          if (field === '_id') {\n            field = 'objectId';\n          }\n          const matchPatterns = [];\n          Object.keys(ParseToPosgresComparator).forEach(cmp => {\n            if (value[cmp]) {\n              const pgComparator = ParseToPosgresComparator[cmp];\n              matchPatterns.push(`$${index}:name ${pgComparator} $${index + 1}`);\n              values.push(field, toPostgresValue(value[cmp]));\n              index += 2;\n            }\n          });\n          if (matchPatterns.length > 0) {\n            patterns.push(`(${matchPatterns.join(' AND ')})`);\n          }\n          if (schema.fields[field] && schema.fields[field].type && matchPatterns.length === 0) {\n            patterns.push(`$${index}:name = $${index + 1}`);\n            values.push(field, value);\n            index += 2;\n          }\n        }\n        wherePattern = patterns.length > 0 ? `WHERE ${patterns.join(` ${orOrAnd} `)}` : '';\n      }\n      if (stage.$limit) {\n        limitPattern = `LIMIT $${index}`;\n        values.push(stage.$limit);\n        index += 1;\n      }\n      if (stage.$skip) {\n        skipPattern = `OFFSET $${index}`;\n        values.push(stage.$skip);\n        index += 1;\n      }\n      if (stage.$sort) {\n        const sort = stage.$sort;\n        const keys = Object.keys(sort);\n        const sorting = keys\n          .map(key => {\n            const transformer = sort[key] === 1 ? 'ASC' : 'DESC';\n            const order = `$${index}:name ${transformer}`;\n            index += 1;\n            return order;\n          })\n          .join();\n        values.push(...keys);\n        sortPattern = sort !== undefined && sorting.length > 0 ? `ORDER BY ${sorting}` : '';\n      }\n    }\n\n    if (groupPattern) {\n      columns.forEach((e, i, a) => {\n        if (e && e.trim() === '*') {\n          a[i] = '';\n        }\n      });\n    }\n\n    const originalQuery = `SELECT ${columns\n      .filter(Boolean)\n      .join()} FROM $1:name ${wherePattern} ${skipPattern} ${groupPattern} ${sortPattern} ${limitPattern}`;\n    const qs = explain ? this.createExplainableQuery(originalQuery) : originalQuery;\n    return this._client.any(qs, values).then(a => {\n      if (explain) {\n        return a;\n      }\n      const results = a.map(object => this.postgresObjectToParseObject(className, object, schema));\n      results.forEach(result => {\n        if (!Object.prototype.hasOwnProperty.call(result, 'objectId')) {\n          result.objectId = null;\n        }\n        if (groupValues) {\n          result.objectId = {};\n          for (const key in groupValues) {\n            result.objectId[key] = result[key];\n            delete result[key];\n          }\n        }\n        if (countField) {\n          result[countField] = parseInt(result[countField], 10);\n        }\n      });\n      return results;\n    });\n  }\n\n  async performInitialization({ VolatileClassesSchemas }: any) {\n    // TODO: This method needs to be rewritten to make proper use of connections (@vitaly-t)\n    debug('performInitialization');\n    await this._ensureSchemaCollectionExists();\n    const promises = VolatileClassesSchemas.map(schema => {\n      return this.createTable(schema.className, schema)\n        .catch(err => {\n          if (\n            err.code === PostgresDuplicateRelationError ||\n            err.code === Parse.Error.INVALID_CLASS_NAME\n          ) {\n            return Promise.resolve();\n          }\n          throw err;\n        })\n        .then(() => this.schemaUpgrade(schema.className, schema));\n    });\n    promises.push(this._listenToSchema());\n    return Promise.all(promises)\n      .then(() => {\n        return this._client.tx('perform-initialization', async t => {\n          await t.none(sql.misc.jsonObjectSetKeys);\n          await t.none(sql.array.add);\n          await t.none(sql.array.addUnique);\n          await t.none(sql.array.remove);\n          await t.none(sql.array.containsAll);\n          await t.none(sql.array.containsAllRegex);\n          await t.none(sql.array.contains);\n          return t.ctx;\n        });\n      })\n      .then(ctx => {\n        debug(`initializationDone in ${ctx.duration}`);\n      })\n      .catch(error => {\n        /* eslint-disable no-console */\n        console.error(error);\n      });\n  }\n\n  async createIndexes(className: string, indexes: any, conn: ?any): Promise<void> {\n    return (conn || this._client).tx(t =>\n      t.batch(\n        indexes.map(i => {\n          return t.none('CREATE INDEX IF NOT EXISTS $1:name ON $2:name ($3:name)', [\n            i.name,\n            className,\n            i.key,\n          ]);\n        })\n      )\n    );\n  }\n\n  async createIndexesIfNeeded(\n    className: string,\n    fieldName: string,\n    type: any,\n    conn: ?any\n  ): Promise<void> {\n    await (conn || this._client).none('CREATE INDEX IF NOT EXISTS $1:name ON $2:name ($3:name)', [\n      fieldName,\n      className,\n      type,\n    ]);\n  }\n\n  async dropIndexes(className: string, indexes: any, conn: any): Promise<void> {\n    const queries = indexes.map(i => ({\n      query: 'DROP INDEX $1:name',\n      values: i,\n    }));\n    await (conn || this._client).tx(t => t.none(this._pgp.helpers.concat(queries)));\n  }\n\n  async getIndexes(className: string) {\n    const qs = 'SELECT * FROM pg_indexes WHERE tablename = ${className}';\n    return this._client.any(qs, { className });\n  }\n\n  async updateSchemaWithIndexes(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  // Used for testing purposes\n  async updateEstimatedCount(className: string) {\n    return this._client.none('ANALYZE $1:name', [className]);\n  }\n\n  async createTransactionalSession(): Promise<any> {\n    return new Promise(resolve => {\n      const transactionalSession = {};\n      transactionalSession.result = this._client.tx(t => {\n        transactionalSession.t = t;\n        transactionalSession.promise = new Promise(resolve => {\n          transactionalSession.resolve = resolve;\n        });\n        transactionalSession.batch = [];\n        resolve(transactionalSession);\n        return transactionalSession.promise;\n      });\n    });\n  }\n\n  commitTransactionalSession(transactionalSession: any): Promise<void> {\n    transactionalSession.resolve(transactionalSession.t.batch(transactionalSession.batch));\n    return transactionalSession.result;\n  }\n\n  abortTransactionalSession(transactionalSession: any): Promise<void> {\n    const result = transactionalSession.result.catch();\n    transactionalSession.batch.push(Promise.reject());\n    transactionalSession.resolve(transactionalSession.t.batch(transactionalSession.batch));\n    return result;\n  }\n\n  async ensureIndex(\n    className: string,\n    schema: SchemaType,\n    fieldNames: string[],\n    indexName: ?string,\n    caseInsensitive: boolean = false,\n    options?: Object = {}\n  ): Promise<any> {\n    const conn = options.conn !== undefined ? options.conn : this._client;\n    const defaultIndexName = `parse_default_${fieldNames.sort().join('_')}`;\n    const indexNameOptions: Object =\n      indexName != null ? { name: indexName } : { name: defaultIndexName };\n    const constraintPatterns = caseInsensitive\n      ? fieldNames.map((fieldName, index) => `lower($${index + 3}:name) varchar_pattern_ops`)\n      : fieldNames.map((fieldName, index) => `$${index + 3}:name`);\n    const qs = `CREATE INDEX IF NOT EXISTS $1:name ON $2:name (${constraintPatterns.join()})`;\n    const setIdempotencyFunction =\n      options.setIdempotencyFunction !== undefined ? options.setIdempotencyFunction : false;\n    if (setIdempotencyFunction) {\n      await this.ensureIdempotencyFunctionExists(options);\n    }\n    await conn.none(qs, [indexNameOptions.name, className, ...fieldNames]).catch(error => {\n      if (\n        error.code === PostgresDuplicateRelationError &&\n        error.message.includes(indexNameOptions.name)\n      ) {\n        // Index already exists. Ignore error.\n      } else if (\n        error.code === PostgresUniqueIndexViolationError &&\n        error.message.includes(indexNameOptions.name)\n      ) {\n        // Cast the error into the proper parse error\n        throw new Parse.Error(\n          Parse.Error.DUPLICATE_VALUE,\n          'A duplicate value for a field with unique values was provided'\n        );\n      } else {\n        throw error;\n      }\n    });\n  }\n\n  async deleteIdempotencyFunction(options?: Object = {}): Promise<any> {\n    const conn = options.conn !== undefined ? options.conn : this._client;\n    const qs = 'DROP FUNCTION IF EXISTS idempotency_delete_expired_records()';\n    return conn.none(qs).catch(error => {\n      throw error;\n    });\n  }\n\n  async ensureIdempotencyFunctionExists(options?: Object = {}): Promise<any> {\n    const conn = options.conn !== undefined ? options.conn : this._client;\n    const ttlOptions = options.ttl !== undefined ? `${options.ttl} seconds` : '60 seconds';\n    const qs =\n      'CREATE OR REPLACE FUNCTION idempotency_delete_expired_records() RETURNS void LANGUAGE plpgsql AS $$ BEGIN DELETE FROM \"_Idempotency\" WHERE expire < NOW() - INTERVAL $1; END; $$;';\n    return conn.none(qs, [ttlOptions]).catch(error => {\n      throw error;\n    });\n  }\n}\n\nfunction convertPolygonToSQL(polygon) {\n  if (polygon.length < 3) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, `Polygon must have at least 3 values`);\n  }\n  if (\n    polygon[0][0] !== polygon[polygon.length - 1][0] ||\n    polygon[0][1] !== polygon[polygon.length - 1][1]\n  ) {\n    polygon.push(polygon[0]);\n  }\n  const unique = polygon.filter((item, index, ar) => {\n    let foundIndex = -1;\n    for (let i = 0; i < ar.length; i += 1) {\n      const pt = ar[i];\n      if (pt[0] === item[0] && pt[1] === item[1]) {\n        foundIndex = i;\n        break;\n      }\n    }\n    return foundIndex === index;\n  });\n  if (unique.length < 3) {\n    throw new Parse.Error(\n      Parse.Error.INTERNAL_SERVER_ERROR,\n      'GeoJSON: Loop must have at least 3 different vertices'\n    );\n  }\n  const points = polygon\n    .map(point => {\n      Parse.GeoPoint._validate(parseFloat(point[1]), parseFloat(point[0]));\n      return `(${point[1]}, ${point[0]})`;\n    })\n    .join(', ');\n  return `(${points})`;\n}\n\nfunction removeWhiteSpace(regex) {\n  if (!regex.endsWith('\\n')) {\n    regex += '\\n';\n  }\n\n  // remove non escaped comments\n  return (\n    regex\n      .replace(/([^\\\\])#.*\\n/gim, '$1')\n      // remove lines starting with a comment\n      .replace(/^#.*\\n/gim, '')\n      // remove non escaped whitespace\n      .replace(/([^\\\\])\\s+/gim, '$1')\n      // remove whitespace at the beginning of a line\n      .replace(/^\\s+/, '')\n      .trim()\n  );\n}\n\nfunction processRegexPattern(s) {\n  if (s && s.startsWith('^')) {\n    // regex for startsWith\n    return '^' + literalizeRegexPart(s.slice(1));\n  } else if (s && s.endsWith('$')) {\n    // regex for endsWith\n    return literalizeRegexPart(s.slice(0, s.length - 1)) + '$';\n  }\n\n  // regex for contains\n  return literalizeRegexPart(s);\n}\n\nfunction isStartsWithRegex(value) {\n  if (!value || typeof value !== 'string' || !value.startsWith('^')) {\n    return false;\n  }\n\n  const matches = value.match(/\\^\\\\Q.*\\\\E/);\n  return !!matches;\n}\n\nfunction isAllValuesRegexOrNone(values) {\n  if (!values || !Array.isArray(values) || values.length === 0) {\n    return true;\n  }\n\n  const firstValuesIsRegex = isStartsWithRegex(values[0].$regex);\n  if (values.length === 1) {\n    return firstValuesIsRegex;\n  }\n\n  for (let i = 1, length = values.length; i < length; ++i) {\n    if (firstValuesIsRegex !== isStartsWithRegex(values[i].$regex)) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\nfunction isAnyValueRegexStartsWith(values) {\n  return values.some(function (value) {\n    return isStartsWithRegex(value.$regex);\n  });\n}\n\nfunction createLiteralRegex(remaining) {\n  return remaining\n    .split('')\n    .map(c => {\n      const regex = RegExp('[0-9 ]|\\\\p{L}', 'u'); // Support all unicode letter chars\n      if (c.match(regex) !== null) {\n        // don't escape alphanumeric characters\n        return c;\n      }\n      // escape everything else (single quotes with single quotes, everything else with a backslash)\n      return c === `'` ? `''` : `\\\\${c}`;\n    })\n    .join('');\n}\n\nfunction literalizeRegexPart(s: string) {\n  const matcher1 = /\\\\Q((?!\\\\E).*)\\\\E$/;\n  const result1: any = s.match(matcher1);\n  if (result1 && result1.length > 1 && result1.index > -1) {\n    // process regex that has a beginning and an end specified for the literal text\n    const prefix = s.substr(0, result1.index);\n    const remaining = result1[1];\n\n    return literalizeRegexPart(prefix) + createLiteralRegex(remaining);\n  }\n\n  // process regex that has a beginning specified for the literal text\n  const matcher2 = /\\\\Q((?!\\\\E).*)$/;\n  const result2: any = s.match(matcher2);\n  if (result2 && result2.length > 1 && result2.index > -1) {\n    const prefix = s.substr(0, result2.index);\n    const remaining = result2[1];\n\n    return literalizeRegexPart(prefix) + createLiteralRegex(remaining);\n  }\n\n  // remove all instances of \\Q and \\E from the remaining text & escape single quotes\n  return s\n    .replace(/([^\\\\])(\\\\E)/, '$1')\n    .replace(/([^\\\\])(\\\\Q)/, '$1')\n    .replace(/^\\\\E/, '')\n    .replace(/^\\\\Q/, '')\n    .replace(/([^'])'/, `$1''`)\n    .replace(/^'([^'])/, `''$1`);\n}\n\nvar GeoPointCoder = {\n  isValidJSON(value) {\n    return typeof value === 'object' && value !== null && value.__type === 'GeoPoint';\n  },\n};\n\nexport default PostgresStorageAdapter;\n"],"mappings":"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"}
|