parse-server 2.8.4 → 8.6.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (240) hide show
  1. package/LICENSE +167 -25
  2. package/NOTICE +10 -0
  3. package/README.md +929 -278
  4. package/lib/AccountLockout.js +47 -30
  5. package/lib/Adapters/AdapterLoader.js +21 -6
  6. package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
  7. package/lib/Adapters/Auth/AuthAdapter.js +116 -13
  8. package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
  9. package/lib/Adapters/Auth/OAuth1Client.js +27 -46
  10. package/lib/Adapters/Auth/apple.js +123 -0
  11. package/lib/Adapters/Auth/facebook.js +162 -35
  12. package/lib/Adapters/Auth/gcenter.js +217 -0
  13. package/lib/Adapters/Auth/github.js +118 -48
  14. package/lib/Adapters/Auth/google.js +160 -51
  15. package/lib/Adapters/Auth/gpgames.js +125 -0
  16. package/lib/Adapters/Auth/httpsRequest.js +6 -7
  17. package/lib/Adapters/Auth/index.js +170 -62
  18. package/lib/Adapters/Auth/instagram.js +114 -40
  19. package/lib/Adapters/Auth/janraincapture.js +52 -23
  20. package/lib/Adapters/Auth/janrainengage.js +19 -36
  21. package/lib/Adapters/Auth/keycloak.js +148 -0
  22. package/lib/Adapters/Auth/ldap.js +167 -0
  23. package/lib/Adapters/Auth/line.js +125 -0
  24. package/lib/Adapters/Auth/linkedin.js +111 -55
  25. package/lib/Adapters/Auth/meetup.js +24 -34
  26. package/lib/Adapters/Auth/mfa.js +324 -0
  27. package/lib/Adapters/Auth/microsoft.js +111 -0
  28. package/lib/Adapters/Auth/oauth2.js +97 -162
  29. package/lib/Adapters/Auth/phantauth.js +53 -0
  30. package/lib/Adapters/Auth/qq.js +108 -49
  31. package/lib/Adapters/Auth/spotify.js +107 -55
  32. package/lib/Adapters/Auth/twitter.js +188 -48
  33. package/lib/Adapters/Auth/utils.js +28 -0
  34. package/lib/Adapters/Auth/vkontakte.js +26 -39
  35. package/lib/Adapters/Auth/wechat.js +106 -44
  36. package/lib/Adapters/Auth/weibo.js +132 -58
  37. package/lib/Adapters/Cache/CacheAdapter.js +13 -8
  38. package/lib/Adapters/Cache/InMemoryCache.js +3 -13
  39. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
  40. package/lib/Adapters/Cache/LRUCache.js +13 -27
  41. package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
  42. package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
  43. package/lib/Adapters/Cache/SchemaCache.js +25 -0
  44. package/lib/Adapters/Email/MailAdapter.js +10 -8
  45. package/lib/Adapters/Files/FilesAdapter.js +83 -25
  46. package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
  47. package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
  48. package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
  49. package/lib/Adapters/Logger/WinstonLogger.js +69 -88
  50. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
  51. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
  52. package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
  53. package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
  54. package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
  55. package/lib/Adapters/Push/PushAdapter.js +14 -7
  56. package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
  57. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
  58. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
  59. package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
  60. package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
  61. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
  62. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
  63. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
  64. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  65. package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
  66. package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
  67. package/lib/Auth.js +488 -125
  68. package/lib/ClientSDK.js +2 -6
  69. package/lib/Config.js +525 -94
  70. package/lib/Controllers/AdaptableController.js +5 -25
  71. package/lib/Controllers/AnalyticsController.js +22 -23
  72. package/lib/Controllers/CacheController.js +10 -31
  73. package/lib/Controllers/DatabaseController.js +767 -313
  74. package/lib/Controllers/FilesController.js +49 -54
  75. package/lib/Controllers/HooksController.js +80 -84
  76. package/lib/Controllers/LiveQueryController.js +35 -22
  77. package/lib/Controllers/LoggerController.js +22 -58
  78. package/lib/Controllers/ParseGraphQLController.js +293 -0
  79. package/lib/Controllers/PushController.js +58 -49
  80. package/lib/Controllers/SchemaController.js +916 -422
  81. package/lib/Controllers/UserController.js +265 -180
  82. package/lib/Controllers/index.js +90 -125
  83. package/lib/Controllers/types.js +1 -1
  84. package/lib/Deprecator/Deprecations.js +30 -0
  85. package/lib/Deprecator/Deprecator.js +127 -0
  86. package/lib/Error.js +48 -0
  87. package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
  88. package/lib/GraphQL/ParseGraphQLServer.js +214 -0
  89. package/lib/GraphQL/helpers/objectsMutations.js +30 -0
  90. package/lib/GraphQL/helpers/objectsQueries.js +246 -0
  91. package/lib/GraphQL/loaders/configMutations.js +87 -0
  92. package/lib/GraphQL/loaders/configQueries.js +79 -0
  93. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
  94. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
  95. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
  96. package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
  97. package/lib/GraphQL/loaders/filesMutations.js +107 -0
  98. package/lib/GraphQL/loaders/functionsMutations.js +78 -0
  99. package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
  100. package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
  101. package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
  102. package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
  103. package/lib/GraphQL/loaders/schemaMutations.js +162 -0
  104. package/lib/GraphQL/loaders/schemaQueries.js +81 -0
  105. package/lib/GraphQL/loaders/schemaTypes.js +341 -0
  106. package/lib/GraphQL/loaders/usersMutations.js +433 -0
  107. package/lib/GraphQL/loaders/usersQueries.js +90 -0
  108. package/lib/GraphQL/parseGraphQLUtils.js +63 -0
  109. package/lib/GraphQL/transformers/className.js +14 -0
  110. package/lib/GraphQL/transformers/constraintType.js +53 -0
  111. package/lib/GraphQL/transformers/inputType.js +51 -0
  112. package/lib/GraphQL/transformers/mutation.js +274 -0
  113. package/lib/GraphQL/transformers/outputType.js +51 -0
  114. package/lib/GraphQL/transformers/query.js +237 -0
  115. package/lib/GraphQL/transformers/schemaFields.js +99 -0
  116. package/lib/KeyPromiseQueue.js +48 -0
  117. package/lib/LiveQuery/Client.js +25 -33
  118. package/lib/LiveQuery/Id.js +2 -5
  119. package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
  120. package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
  121. package/lib/LiveQuery/ParsePubSub.js +7 -16
  122. package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
  123. package/lib/LiveQuery/QueryTools.js +76 -15
  124. package/lib/LiveQuery/RequestSchema.js +111 -97
  125. package/lib/LiveQuery/SessionTokenCache.js +23 -36
  126. package/lib/LiveQuery/Subscription.js +8 -17
  127. package/lib/LiveQuery/equalObjects.js +2 -3
  128. package/lib/Options/Definitions.js +1355 -382
  129. package/lib/Options/docs.js +301 -62
  130. package/lib/Options/index.js +11 -1
  131. package/lib/Options/parsers.js +14 -10
  132. package/lib/Page.js +44 -0
  133. package/lib/ParseMessageQueue.js +6 -13
  134. package/lib/ParseServer.js +474 -235
  135. package/lib/ParseServerRESTController.js +102 -40
  136. package/lib/PromiseRouter.js +39 -50
  137. package/lib/Push/PushQueue.js +24 -30
  138. package/lib/Push/PushWorker.js +32 -56
  139. package/lib/Push/utils.js +22 -35
  140. package/lib/RestQuery.js +361 -139
  141. package/lib/RestWrite.js +713 -344
  142. package/lib/Routers/AggregateRouter.js +97 -71
  143. package/lib/Routers/AnalyticsRouter.js +8 -14
  144. package/lib/Routers/AudiencesRouter.js +16 -35
  145. package/lib/Routers/ClassesRouter.js +86 -72
  146. package/lib/Routers/CloudCodeRouter.js +28 -37
  147. package/lib/Routers/FeaturesRouter.js +22 -25
  148. package/lib/Routers/FilesRouter.js +266 -171
  149. package/lib/Routers/FunctionsRouter.js +87 -103
  150. package/lib/Routers/GlobalConfigRouter.js +94 -33
  151. package/lib/Routers/GraphQLRouter.js +41 -0
  152. package/lib/Routers/HooksRouter.js +43 -47
  153. package/lib/Routers/IAPValidationRouter.js +57 -70
  154. package/lib/Routers/InstallationsRouter.js +17 -25
  155. package/lib/Routers/LogsRouter.js +10 -25
  156. package/lib/Routers/PagesRouter.js +647 -0
  157. package/lib/Routers/PublicAPIRouter.js +104 -112
  158. package/lib/Routers/PurgeRouter.js +19 -29
  159. package/lib/Routers/PushRouter.js +14 -28
  160. package/lib/Routers/RolesRouter.js +7 -14
  161. package/lib/Routers/SchemasRouter.js +63 -42
  162. package/lib/Routers/SecurityRouter.js +34 -0
  163. package/lib/Routers/SessionsRouter.js +25 -38
  164. package/lib/Routers/UsersRouter.js +463 -190
  165. package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
  166. package/lib/SchemaMigrations/Migrations.js +30 -0
  167. package/lib/Security/Check.js +109 -0
  168. package/lib/Security/CheckGroup.js +44 -0
  169. package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
  170. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
  171. package/lib/Security/CheckGroups/CheckGroups.js +21 -0
  172. package/lib/Security/CheckRunner.js +213 -0
  173. package/lib/SharedRest.js +29 -0
  174. package/lib/StatusHandler.js +96 -93
  175. package/lib/TestUtils.js +70 -14
  176. package/lib/Utils.js +468 -0
  177. package/lib/batch.js +74 -40
  178. package/lib/cache.js +8 -8
  179. package/lib/cli/definitions/parse-live-query-server.js +4 -3
  180. package/lib/cli/definitions/parse-server.js +4 -3
  181. package/lib/cli/parse-live-query-server.js +9 -17
  182. package/lib/cli/parse-server.js +49 -47
  183. package/lib/cli/utils/commander.js +20 -29
  184. package/lib/cli/utils/runner.js +31 -32
  185. package/lib/cloud-code/Parse.Cloud.js +711 -36
  186. package/lib/cloud-code/Parse.Server.js +21 -0
  187. package/lib/cryptoUtils.js +6 -11
  188. package/lib/defaults.js +21 -15
  189. package/lib/deprecated.js +1 -1
  190. package/lib/index.js +78 -67
  191. package/lib/logger.js +12 -20
  192. package/lib/middlewares.js +484 -160
  193. package/lib/password.js +10 -6
  194. package/lib/request.js +175 -0
  195. package/lib/requiredParameter.js +4 -3
  196. package/lib/rest.js +157 -82
  197. package/lib/triggers.js +627 -185
  198. package/lib/vendor/README.md +3 -3
  199. package/lib/vendor/mongodbUrl.js +224 -137
  200. package/package.json +135 -57
  201. package/postinstall.js +38 -50
  202. package/public_html/invalid_verification_link.html +3 -3
  203. package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
  204. package/types/@types/deepcopy/index.d.ts +5 -0
  205. package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
  206. package/types/Options/index.d.ts +301 -0
  207. package/types/ParseServer.d.ts +65 -0
  208. package/types/eslint.config.mjs +30 -0
  209. package/types/index.d.ts +21 -0
  210. package/types/logger.d.ts +2 -0
  211. package/types/tests.ts +44 -0
  212. package/types/tsconfig.json +24 -0
  213. package/CHANGELOG.md +0 -1246
  214. package/PATENTS +0 -37
  215. package/bin/dev +0 -37
  216. package/lib/.DS_Store +0 -0
  217. package/lib/Adapters/Auth/common.js +0 -2
  218. package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
  219. package/lib/Controllers/SchemaCache.js +0 -97
  220. package/lib/LiveQuery/.DS_Store +0 -0
  221. package/lib/cli/utils/parsers.js +0 -77
  222. package/lib/cloud-code/.DS_Store +0 -0
  223. package/lib/cloud-code/HTTPResponse.js +0 -57
  224. package/lib/cloud-code/Untitled-1 +0 -123
  225. package/lib/cloud-code/httpRequest.js +0 -102
  226. package/lib/cloud-code/team.html +0 -123
  227. package/lib/graphql/ParseClass.js +0 -234
  228. package/lib/graphql/Schema.js +0 -197
  229. package/lib/graphql/index.js +0 -1
  230. package/lib/graphql/types/ACL.js +0 -35
  231. package/lib/graphql/types/Date.js +0 -25
  232. package/lib/graphql/types/File.js +0 -24
  233. package/lib/graphql/types/GeoPoint.js +0 -35
  234. package/lib/graphql/types/JSONObject.js +0 -30
  235. package/lib/graphql/types/NumberInput.js +0 -43
  236. package/lib/graphql/types/NumberQuery.js +0 -42
  237. package/lib/graphql/types/Pointer.js +0 -35
  238. package/lib/graphql/types/QueryConstraint.js +0 -61
  239. package/lib/graphql/types/StringQuery.js +0 -39
  240. package/lib/graphql/types/index.js +0 -110
@@ -1,22 +1,22 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- exports.SchemaController = exports.VolatileClassesSchemas = exports.convertSchemaToAdapterSchema = exports.defaultColumns = exports.systemClasses = exports.buildMergedSchemaObject = exports.invalidClassNameMessage = exports.fieldNameIsValid = exports.classNameIsValid = exports.load = undefined;
7
-
8
- var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
9
-
10
- var _StorageAdapter = require('../Adapters/Storage/StorageAdapter');
11
-
12
- var _DatabaseController = require('./DatabaseController');
13
-
14
- var _DatabaseController2 = _interopRequireDefault(_DatabaseController);
15
-
16
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
17
-
18
- function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; }
19
-
6
+ exports.VolatileClassesSchemas = exports.SchemaController = void 0;
7
+ exports.buildMergedSchemaObject = buildMergedSchemaObject;
8
+ exports.classNameIsValid = classNameIsValid;
9
+ exports.defaultColumns = exports.default = exports.convertSchemaToAdapterSchema = void 0;
10
+ exports.fieldNameIsValid = fieldNameIsValid;
11
+ exports.invalidClassNameMessage = invalidClassNameMessage;
12
+ exports.systemClasses = exports.requiredColumns = exports.load = void 0;
13
+ var _StorageAdapter = require("../Adapters/Storage/StorageAdapter");
14
+ var _SchemaCache = _interopRequireDefault(require("../Adapters/Cache/SchemaCache"));
15
+ var _DatabaseController = _interopRequireDefault(require("./DatabaseController"));
16
+ var _Config = _interopRequireDefault(require("../Config"));
17
+ var _Error = require("../Error");
18
+ var _deepcopy = _interopRequireDefault(require("deepcopy"));
19
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
20
20
  // This class handles schema validation, persistence, and modification.
21
21
  //
22
22
  // Each individual Schema object should be immutable. The helpers to
@@ -35,187 +35,485 @@ function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in ob
35
35
  // -disable-next
36
36
  const Parse = require('parse/node').Parse;
37
37
 
38
+ // -disable-next
38
39
 
39
- const defaultColumns = Object.freeze({
40
+ const defaultColumns = exports.defaultColumns = Object.freeze({
40
41
  // Contain the default columns for every parse object type (except _Join collection)
41
42
  _Default: {
42
- "objectId": { type: 'String' },
43
- "createdAt": { type: 'Date' },
44
- "updatedAt": { type: 'Date' },
45
- "ACL": { type: 'ACL' }
43
+ objectId: {
44
+ type: 'String'
45
+ },
46
+ createdAt: {
47
+ type: 'Date'
48
+ },
49
+ updatedAt: {
50
+ type: 'Date'
51
+ },
52
+ ACL: {
53
+ type: 'ACL'
54
+ }
46
55
  },
47
56
  // The additional default columns for the _User collection (in addition to DefaultCols)
48
57
  _User: {
49
- "username": { type: 'String' },
50
- "password": { type: 'String' },
51
- "email": { type: 'String' },
52
- "emailVerified": { type: 'Boolean' },
53
- "authData": { type: 'Object' }
58
+ username: {
59
+ type: 'String'
60
+ },
61
+ password: {
62
+ type: 'String'
63
+ },
64
+ email: {
65
+ type: 'String'
66
+ },
67
+ emailVerified: {
68
+ type: 'Boolean'
69
+ },
70
+ authData: {
71
+ type: 'Object'
72
+ }
54
73
  },
55
74
  // The additional default columns for the _Installation collection (in addition to DefaultCols)
56
75
  _Installation: {
57
- "installationId": { type: 'String' },
58
- "deviceToken": { type: 'String' },
59
- "channels": { type: 'Array' },
60
- "deviceType": { type: 'String' },
61
- "pushType": { type: 'String' },
62
- "GCMSenderId": { type: 'String' },
63
- "timeZone": { type: 'String' },
64
- "localeIdentifier": { type: 'String' },
65
- "badge": { type: 'Number' },
66
- "appVersion": { type: 'String' },
67
- "appName": { type: 'String' },
68
- "appIdentifier": { type: 'String' },
69
- "parseVersion": { type: 'String' }
76
+ installationId: {
77
+ type: 'String'
78
+ },
79
+ deviceToken: {
80
+ type: 'String'
81
+ },
82
+ channels: {
83
+ type: 'Array'
84
+ },
85
+ deviceType: {
86
+ type: 'String'
87
+ },
88
+ pushType: {
89
+ type: 'String'
90
+ },
91
+ GCMSenderId: {
92
+ type: 'String'
93
+ },
94
+ timeZone: {
95
+ type: 'String'
96
+ },
97
+ localeIdentifier: {
98
+ type: 'String'
99
+ },
100
+ badge: {
101
+ type: 'Number'
102
+ },
103
+ appVersion: {
104
+ type: 'String'
105
+ },
106
+ appName: {
107
+ type: 'String'
108
+ },
109
+ appIdentifier: {
110
+ type: 'String'
111
+ },
112
+ parseVersion: {
113
+ type: 'String'
114
+ }
70
115
  },
71
116
  // The additional default columns for the _Role collection (in addition to DefaultCols)
72
117
  _Role: {
73
- "name": { type: 'String' },
74
- "users": { type: 'Relation', targetClass: '_User' },
75
- "roles": { type: 'Relation', targetClass: '_Role' }
118
+ name: {
119
+ type: 'String'
120
+ },
121
+ users: {
122
+ type: 'Relation',
123
+ targetClass: '_User'
124
+ },
125
+ roles: {
126
+ type: 'Relation',
127
+ targetClass: '_Role'
128
+ }
76
129
  },
77
130
  // The additional default columns for the _Session collection (in addition to DefaultCols)
78
131
  _Session: {
79
- "restricted": { type: 'Boolean' },
80
- "user": { type: 'Pointer', targetClass: '_User' },
81
- "installationId": { type: 'String' },
82
- "sessionToken": { type: 'String' },
83
- "expiresAt": { type: 'Date' },
84
- "createdWith": { type: 'Object' }
132
+ user: {
133
+ type: 'Pointer',
134
+ targetClass: '_User'
135
+ },
136
+ installationId: {
137
+ type: 'String'
138
+ },
139
+ sessionToken: {
140
+ type: 'String'
141
+ },
142
+ expiresAt: {
143
+ type: 'Date'
144
+ },
145
+ createdWith: {
146
+ type: 'Object'
147
+ }
85
148
  },
86
149
  _Product: {
87
- "productIdentifier": { type: 'String' },
88
- "download": { type: 'File' },
89
- "downloadName": { type: 'String' },
90
- "icon": { type: 'File' },
91
- "order": { type: 'Number' },
92
- "title": { type: 'String' },
93
- "subtitle": { type: 'String' }
150
+ productIdentifier: {
151
+ type: 'String'
152
+ },
153
+ download: {
154
+ type: 'File'
155
+ },
156
+ downloadName: {
157
+ type: 'String'
158
+ },
159
+ icon: {
160
+ type: 'File'
161
+ },
162
+ order: {
163
+ type: 'Number'
164
+ },
165
+ title: {
166
+ type: 'String'
167
+ },
168
+ subtitle: {
169
+ type: 'String'
170
+ }
94
171
  },
95
172
  _PushStatus: {
96
- "pushTime": { type: 'String' },
97
- "source": { type: 'String' }, // rest or webui
98
- "query": { type: 'String' }, // the stringified JSON query
99
- "payload": { type: 'String' }, // the stringified JSON payload,
100
- "title": { type: 'String' },
101
- "expiry": { type: 'Number' },
102
- "expiration_interval": { type: 'Number' },
103
- "status": { type: 'String' },
104
- "numSent": { type: 'Number' },
105
- "numFailed": { type: 'Number' },
106
- "pushHash": { type: 'String' },
107
- "errorMessage": { type: 'Object' },
108
- "sentPerType": { type: 'Object' },
109
- "failedPerType": { type: 'Object' },
110
- "sentPerUTCOffset": { type: 'Object' },
111
- "failedPerUTCOffset": { type: 'Object' },
112
- "count": { type: 'Number' // tracks # of batches queued and pending
113
- } },
173
+ pushTime: {
174
+ type: 'String'
175
+ },
176
+ source: {
177
+ type: 'String'
178
+ },
179
+ // rest or webui
180
+ query: {
181
+ type: 'String'
182
+ },
183
+ // the stringified JSON query
184
+ payload: {
185
+ type: 'String'
186
+ },
187
+ // the stringified JSON payload,
188
+ title: {
189
+ type: 'String'
190
+ },
191
+ expiry: {
192
+ type: 'Number'
193
+ },
194
+ expiration_interval: {
195
+ type: 'Number'
196
+ },
197
+ status: {
198
+ type: 'String'
199
+ },
200
+ numSent: {
201
+ type: 'Number'
202
+ },
203
+ numFailed: {
204
+ type: 'Number'
205
+ },
206
+ pushHash: {
207
+ type: 'String'
208
+ },
209
+ errorMessage: {
210
+ type: 'Object'
211
+ },
212
+ sentPerType: {
213
+ type: 'Object'
214
+ },
215
+ failedPerType: {
216
+ type: 'Object'
217
+ },
218
+ sentPerUTCOffset: {
219
+ type: 'Object'
220
+ },
221
+ failedPerUTCOffset: {
222
+ type: 'Object'
223
+ },
224
+ count: {
225
+ type: 'Number'
226
+ } // tracks # of batches queued and pending
227
+ },
114
228
  _JobStatus: {
115
- "jobName": { type: 'String' },
116
- "source": { type: 'String' },
117
- "status": { type: 'String' },
118
- "message": { type: 'String' },
119
- "params": { type: 'Object' }, // params received when calling the job
120
- "finishedAt": { type: 'Date' }
229
+ jobName: {
230
+ type: 'String'
231
+ },
232
+ source: {
233
+ type: 'String'
234
+ },
235
+ status: {
236
+ type: 'String'
237
+ },
238
+ message: {
239
+ type: 'String'
240
+ },
241
+ params: {
242
+ type: 'Object'
243
+ },
244
+ // params received when calling the job
245
+ finishedAt: {
246
+ type: 'Date'
247
+ }
121
248
  },
122
249
  _JobSchedule: {
123
- "jobName": { type: 'String' },
124
- "description": { type: 'String' },
125
- "params": { type: 'String' },
126
- "startAfter": { type: 'String' },
127
- "daysOfWeek": { type: 'Array' },
128
- "timeOfDay": { type: 'String' },
129
- "lastRun": { type: 'Number' },
130
- "repeatMinutes": { type: 'Number' }
250
+ jobName: {
251
+ type: 'String'
252
+ },
253
+ description: {
254
+ type: 'String'
255
+ },
256
+ params: {
257
+ type: 'String'
258
+ },
259
+ startAfter: {
260
+ type: 'String'
261
+ },
262
+ daysOfWeek: {
263
+ type: 'Array'
264
+ },
265
+ timeOfDay: {
266
+ type: 'String'
267
+ },
268
+ lastRun: {
269
+ type: 'Number'
270
+ },
271
+ repeatMinutes: {
272
+ type: 'Number'
273
+ }
131
274
  },
132
275
  _Hooks: {
133
- "functionName": { type: 'String' },
134
- "className": { type: 'String' },
135
- "triggerName": { type: 'String' },
136
- "url": { type: 'String' }
276
+ functionName: {
277
+ type: 'String'
278
+ },
279
+ className: {
280
+ type: 'String'
281
+ },
282
+ triggerName: {
283
+ type: 'String'
284
+ },
285
+ url: {
286
+ type: 'String'
287
+ }
137
288
  },
138
289
  _GlobalConfig: {
139
- "objectId": { type: 'String' },
140
- "params": { type: 'Object' }
290
+ objectId: {
291
+ type: 'String'
292
+ },
293
+ params: {
294
+ type: 'Object'
295
+ },
296
+ masterKeyOnly: {
297
+ type: 'Object'
298
+ }
299
+ },
300
+ _GraphQLConfig: {
301
+ objectId: {
302
+ type: 'String'
303
+ },
304
+ config: {
305
+ type: 'Object'
306
+ }
141
307
  },
142
308
  _Audience: {
143
- "objectId": { type: 'String' },
144
- "name": { type: 'String' },
145
- "query": { type: 'String' }, //storing query as JSON string to prevent "Nested keys should not contain the '$' or '.' characters" error
146
- "lastUsed": { type: 'Date' },
147
- "timesUsed": { type: 'Number' }
309
+ objectId: {
310
+ type: 'String'
311
+ },
312
+ name: {
313
+ type: 'String'
314
+ },
315
+ query: {
316
+ type: 'String'
317
+ },
318
+ //storing query as JSON string to prevent "Nested keys should not contain the '$' or '.' characters" error
319
+ lastUsed: {
320
+ type: 'Date'
321
+ },
322
+ timesUsed: {
323
+ type: 'Number'
324
+ }
325
+ },
326
+ _Idempotency: {
327
+ reqId: {
328
+ type: 'String'
329
+ },
330
+ expire: {
331
+ type: 'Date'
332
+ }
148
333
  }
149
334
  });
150
335
 
151
- const requiredColumns = Object.freeze({
152
- _Product: ["productIdentifier", "icon", "order", "title", "subtitle"],
153
- _Role: ["name", "ACL"]
336
+ // fields required for read or write operations on their respective classes.
337
+ const requiredColumns = exports.requiredColumns = Object.freeze({
338
+ read: {
339
+ _User: ['username']
340
+ },
341
+ write: {
342
+ _Product: ['productIdentifier', 'icon', 'order', 'title', 'subtitle'],
343
+ _Role: ['name', 'ACL']
344
+ }
154
345
  });
346
+ const invalidColumns = ['length'];
347
+ const systemClasses = exports.systemClasses = Object.freeze(['_User', '_Installation', '_Role', '_Session', '_Product', '_PushStatus', '_JobStatus', '_JobSchedule', '_Audience', '_Idempotency']);
348
+ const volatileClasses = Object.freeze(['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_GraphQLConfig', '_JobSchedule', '_Audience', '_Idempotency']);
155
349
 
156
- const systemClasses = Object.freeze(['_User', '_Installation', '_Role', '_Session', '_Product', '_PushStatus', '_JobStatus', '_JobSchedule', '_Audience']);
157
-
158
- const volatileClasses = Object.freeze(['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_JobSchedule', '_Audience']);
159
-
160
- // 10 alpha numberic chars + uppercase
161
- const userIdRegex = /^[a-zA-Z0-9]{10}$/;
162
350
  // Anything that start with role
163
351
  const roleRegex = /^role:.*/;
352
+ // Anything that starts with userField (allowed for protected fields only)
353
+ const protectedFieldsPointerRegex = /^userField:.*/;
164
354
  // * permission
165
355
  const publicRegex = /^\*$/;
356
+ const authenticatedRegex = /^authenticated$/;
357
+ const requiresAuthenticationRegex = /^requiresAuthentication$/;
358
+ const clpPointerRegex = /^pointerFields$/;
359
+
360
+ // regex for validating entities in protectedFields object
361
+ const protectedFieldsRegex = Object.freeze([protectedFieldsPointerRegex, publicRegex, authenticatedRegex, roleRegex]);
362
+
363
+ // clp regex
364
+ const clpFieldsRegex = Object.freeze([clpPointerRegex, publicRegex, requiresAuthenticationRegex, roleRegex]);
365
+ function validatePermissionKey(key, userIdRegExp) {
366
+ let matchesSome = false;
367
+ for (const regEx of clpFieldsRegex) {
368
+ if (key.match(regEx) !== null) {
369
+ matchesSome = true;
370
+ break;
371
+ }
372
+ }
166
373
 
167
- const requireAuthenticationRegex = /^requiresAuthentication$/;
168
-
169
- const permissionKeyRegex = Object.freeze([userIdRegex, roleRegex, publicRegex, requireAuthenticationRegex]);
374
+ // userId depends on startup options so it's dynamic
375
+ const valid = matchesSome || key.match(userIdRegExp) !== null;
376
+ if (!valid) {
377
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);
378
+ }
379
+ }
380
+ function validateProtectedFieldsKey(key, userIdRegExp) {
381
+ let matchesSome = false;
382
+ for (const regEx of protectedFieldsRegex) {
383
+ if (key.match(regEx) !== null) {
384
+ matchesSome = true;
385
+ break;
386
+ }
387
+ }
170
388
 
171
- function verifyPermissionKey(key) {
172
- const result = permissionKeyRegex.reduce((isGood, regEx) => {
173
- isGood = isGood || key.match(regEx) != null;
174
- return isGood;
175
- }, false);
176
- if (!result) {
389
+ // userId regex depends on launch options so it's dynamic
390
+ const valid = matchesSome || key.match(userIdRegExp) !== null;
391
+ if (!valid) {
177
392
  throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);
178
393
  }
179
394
  }
395
+ const CLPValidKeys = Object.freeze(['ACL', 'find', 'count', 'get', 'create', 'update', 'delete', 'addField', 'readUserFields', 'writeUserFields', 'protectedFields']);
180
396
 
181
- const CLPValidKeys = Object.freeze(['find', 'count', 'get', 'create', 'update', 'delete', 'addField', 'readUserFields', 'writeUserFields']);
182
- function validateCLP(perms, fields) {
397
+ // validation before setting class-level permissions on collection
398
+ function validateCLP(perms, fields, userIdRegExp) {
183
399
  if (!perms) {
184
400
  return;
185
401
  }
186
- Object.keys(perms).forEach(operation => {
187
- if (CLPValidKeys.indexOf(operation) == -1) {
188
- throw new Parse.Error(Parse.Error.INVALID_JSON, `${operation} is not a valid operation for class level permissions`);
402
+ for (const operationKey in perms) {
403
+ if (CLPValidKeys.indexOf(operationKey) == -1) {
404
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `${operationKey} is not a valid operation for class level permissions`);
189
405
  }
190
- if (!perms[operation]) {
191
- return;
406
+ const operation = perms[operationKey];
407
+ // proceed with next operationKey
408
+
409
+ // throws when root fields are of wrong type
410
+ validateCLPjson(operation, operationKey);
411
+ if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {
412
+ // validate grouped pointer permissions
413
+ // must be an array with field names
414
+ for (const fieldName of operation) {
415
+ validatePointerPermission(fieldName, fields, operationKey);
416
+ }
417
+ // readUserFields and writerUserFields do not have nesdted fields
418
+ // proceed with next operationKey
419
+ continue;
192
420
  }
193
421
 
194
- if (operation === 'readUserFields' || operation === 'writeUserFields') {
195
- if (!Array.isArray(perms[operation])) {
196
- // -disable-next
197
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${perms[operation]}' is not a valid value for class level permissions ${operation}`);
198
- } else {
199
- perms[operation].forEach(key => {
200
- if (!fields[key] || fields[key].type != 'Pointer' || fields[key].targetClass != '_User') {
201
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid column for class level pointer permissions ${operation}`);
422
+ // validate protected fields
423
+ if (operationKey === 'protectedFields') {
424
+ for (const entity in operation) {
425
+ // throws on unexpected key
426
+ validateProtectedFieldsKey(entity, userIdRegExp);
427
+ const protectedFields = operation[entity];
428
+ if (!Array.isArray(protectedFields)) {
429
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`);
430
+ }
431
+
432
+ // if the field is in form of array
433
+ for (const field of protectedFields) {
434
+ // do not alloow to protect default fields
435
+ if (defaultColumns._Default[field]) {
436
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `Default field '${field}' can not be protected`);
202
437
  }
203
- });
438
+ // field should exist on collection
439
+ if (!Object.prototype.hasOwnProperty.call(fields, field)) {
440
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `Field '${field}' in protectedFields:${entity} does not exist`);
441
+ }
442
+ }
204
443
  }
205
- return;
444
+ // proceed with next operationKey
445
+ continue;
206
446
  }
207
447
 
208
- // -disable-next
209
- Object.keys(perms[operation]).forEach(key => {
210
- verifyPermissionKey(key);
211
- // -disable-next
212
- const perm = perms[operation][key];
213
- if (perm !== true) {
214
- // -disable-next
215
- throw new Parse.Error(Parse.Error.INVALID_JSON, `'${perm}' is not a valid value for class level permissions ${operation}:${key}:${perm}`);
448
+ // validate other fields
449
+ // Entity can be:
450
+ // "*" - Public,
451
+ // "requiresAuthentication" - authenticated users,
452
+ // "objectId" - _User id,
453
+ // "role:rolename",
454
+ // "pointerFields" - array of field names containing pointers to users
455
+ for (const entity in operation) {
456
+ // throws on unexpected key
457
+ validatePermissionKey(entity, userIdRegExp);
458
+
459
+ // entity can be either:
460
+ // "pointerFields": string[]
461
+ if (entity === 'pointerFields') {
462
+ const pointerFields = operation[entity];
463
+ if (Array.isArray(pointerFields)) {
464
+ for (const pointerField of pointerFields) {
465
+ validatePointerPermission(pointerField, fields, operation);
466
+ }
467
+ } else {
468
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`);
469
+ }
470
+ // proceed with next entity key
471
+ continue;
216
472
  }
217
- });
218
- });
473
+ const permit = operation[entity];
474
+ if (operationKey === 'ACL') {
475
+ if (Object.prototype.toString.call(permit) !== '[object Object]') {
476
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${permit}' is not a valid value for class level permissions acl`);
477
+ }
478
+ const invalidKeys = Object.keys(permit).filter(key => !['read', 'write'].includes(key));
479
+ const invalidValues = Object.values(permit).filter(key => typeof key !== 'boolean');
480
+ if (invalidKeys.length) {
481
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${invalidKeys.join(',')}' is not a valid key for class level permissions acl`);
482
+ }
483
+ if (invalidValues.length) {
484
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${invalidValues.join(',')}' is not a valid value for class level permissions acl`);
485
+ }
486
+ } else if (permit !== true) {
487
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${permit}' is not a valid value for class level permissions acl ${operationKey}:${entity}`);
488
+ }
489
+ }
490
+ }
491
+ }
492
+ function validateCLPjson(operation, operationKey) {
493
+ if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {
494
+ if (!Array.isArray(operation)) {
495
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an array`);
496
+ }
497
+ } else {
498
+ if (typeof operation === 'object' && operation !== null) {
499
+ // ok to proceed
500
+ return;
501
+ } else {
502
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an object`);
503
+ }
504
+ }
505
+ }
506
+ function validatePointerPermission(fieldName, fields, operation) {
507
+ // Uses collection schema to ensure the field is of type:
508
+ // - Pointer<_User> (pointers)
509
+ // - Array
510
+ //
511
+ // It's not possible to enforce type on Array's items in schema
512
+ // so we accept any Array field, and later when applying permissions
513
+ // only items that are pointers to _User are considered.
514
+ if (!(fields[fieldName] && (fields[fieldName].type == 'Pointer' && fields[fieldName].targetClass == '_User' || fields[fieldName].type == 'Array'))) {
515
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`);
516
+ }
219
517
  }
220
518
  const joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;
221
519
  const classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;
@@ -227,18 +525,24 @@ function classNameIsValid(className) {
227
525
  // Be a join table OR
228
526
  joinClassRegex.test(className) ||
229
527
  // Include only alpha-numeric and underscores, and not start with an underscore or number
230
- fieldNameIsValid(className)
528
+ fieldNameIsValid(className, className)
231
529
  );
232
530
  }
233
531
 
234
532
  // Valid fields must be alpha-numeric, and not start with an underscore or number
235
- function fieldNameIsValid(fieldName) {
236
- return classAndFieldRegex.test(fieldName);
533
+ // must not be a reserved key
534
+ function fieldNameIsValid(fieldName, className) {
535
+ if (className && className !== '_Hooks') {
536
+ if (fieldName === 'className') {
537
+ return false;
538
+ }
539
+ }
540
+ return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);
237
541
  }
238
542
 
239
543
  // Checks that it's not trying to clobber one of the default fields of the class.
240
544
  function fieldNameIsValidForClass(fieldName, className) {
241
- if (!fieldNameIsValid(fieldName)) {
545
+ if (!fieldNameIsValid(fieldName, className)) {
242
546
  return false;
243
547
  }
244
548
  if (defaultColumns._Default[fieldName]) {
@@ -249,15 +553,16 @@ function fieldNameIsValidForClass(fieldName, className) {
249
553
  }
250
554
  return true;
251
555
  }
252
-
253
556
  function invalidClassNameMessage(className) {
254
557
  return 'Invalid classname: ' + className + ', classnames can only have alphanumeric characters and _, and must start with an alpha character ';
255
558
  }
256
-
257
- const invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, "invalid JSON");
559
+ const invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');
258
560
  const validNonRelationOrPointerTypes = ['Number', 'String', 'Boolean', 'Date', 'Object', 'Array', 'GeoPoint', 'File', 'Bytes', 'Polygon'];
259
561
  // Returns an error suitable for throwing if the type is invalid
260
- const fieldTypeIsInvalid = ({ type, targetClass }) => {
562
+ const fieldTypeIsInvalid = ({
563
+ type,
564
+ targetClass
565
+ }) => {
261
566
  if (['Pointer', 'Relation'].indexOf(type) >= 0) {
262
567
  if (!targetClass) {
263
568
  return new Parse.Error(135, `type ${type} needs a class name`);
@@ -277,46 +582,108 @@ const fieldTypeIsInvalid = ({ type, targetClass }) => {
277
582
  }
278
583
  return undefined;
279
584
  };
280
-
281
585
  const convertSchemaToAdapterSchema = schema => {
282
586
  schema = injectDefaultSchema(schema);
283
587
  delete schema.fields.ACL;
284
- schema.fields._rperm = { type: 'Array' };
285
- schema.fields._wperm = { type: 'Array' };
286
-
588
+ schema.fields._rperm = {
589
+ type: 'Array'
590
+ };
591
+ schema.fields._wperm = {
592
+ type: 'Array'
593
+ };
287
594
  if (schema.className === '_User') {
288
595
  delete schema.fields.password;
289
- schema.fields._hashed_password = { type: 'String' };
596
+ schema.fields._hashed_password = {
597
+ type: 'String'
598
+ };
290
599
  }
291
-
292
600
  return schema;
293
601
  };
294
-
295
- const convertAdapterSchemaToParseSchema = (_ref) => {
296
- let schema = _objectWithoutProperties(_ref, []);
297
-
602
+ exports.convertSchemaToAdapterSchema = convertSchemaToAdapterSchema;
603
+ const convertAdapterSchemaToParseSchema = ({
604
+ ...schema
605
+ }) => {
298
606
  delete schema.fields._rperm;
299
607
  delete schema.fields._wperm;
300
-
301
- schema.fields.ACL = { type: 'ACL' };
302
-
608
+ schema.fields.ACL = {
609
+ type: 'ACL'
610
+ };
303
611
  if (schema.className === '_User') {
304
612
  delete schema.fields.authData; //Auth data is implicit
305
613
  delete schema.fields._hashed_password;
306
- schema.fields.password = { type: 'String' };
614
+ schema.fields.password = {
615
+ type: 'String'
616
+ };
307
617
  }
308
-
309
618
  if (schema.indexes && Object.keys(schema.indexes).length === 0) {
310
619
  delete schema.indexes;
311
620
  }
312
-
313
621
  return schema;
314
622
  };
623
+ class SchemaData {
624
+ constructor(allSchemas = [], protectedFields = {}) {
625
+ this.__data = {};
626
+ this.__protectedFields = protectedFields;
627
+ allSchemas.forEach(schema => {
628
+ if (volatileClasses.includes(schema.className)) {
629
+ return;
630
+ }
631
+ Object.defineProperty(this, schema.className, {
632
+ get: () => {
633
+ if (!this.__data[schema.className]) {
634
+ const data = {};
635
+ data.fields = injectDefaultSchema(schema).fields;
636
+ data.classLevelPermissions = (0, _deepcopy.default)(schema.classLevelPermissions);
637
+ data.indexes = schema.indexes;
638
+ const classProtectedFields = this.__protectedFields[schema.className];
639
+ if (classProtectedFields) {
640
+ for (const key in classProtectedFields) {
641
+ const unq = new Set([...(data.classLevelPermissions.protectedFields[key] || []), ...classProtectedFields[key]]);
642
+ data.classLevelPermissions.protectedFields[key] = Array.from(unq);
643
+ }
644
+ }
645
+ this.__data[schema.className] = data;
646
+ }
647
+ return this.__data[schema.className];
648
+ }
649
+ });
650
+ });
315
651
 
316
- const injectDefaultSchema = ({ className, fields, classLevelPermissions, indexes }) => {
652
+ // Inject the in-memory classes
653
+ volatileClasses.forEach(className => {
654
+ Object.defineProperty(this, className, {
655
+ get: () => {
656
+ if (!this.__data[className]) {
657
+ const schema = injectDefaultSchema({
658
+ className,
659
+ fields: {},
660
+ classLevelPermissions: {}
661
+ });
662
+ const data = {};
663
+ data.fields = schema.fields;
664
+ data.classLevelPermissions = schema.classLevelPermissions;
665
+ data.indexes = schema.indexes;
666
+ this.__data[className] = data;
667
+ }
668
+ return this.__data[className];
669
+ }
670
+ });
671
+ });
672
+ }
673
+ }
674
+ const injectDefaultSchema = ({
675
+ className,
676
+ fields,
677
+ classLevelPermissions,
678
+ indexes
679
+ }) => {
317
680
  const defaultSchema = {
318
681
  className,
319
- fields: _extends({}, defaultColumns._Default, defaultColumns[className] || {}, fields),
682
+ fields: {
683
+ ...defaultColumns._Default,
684
+ ...(defaultColumns[className] || {}),
685
+ ...fields
686
+ },
320
687
  classLevelPermissions
321
688
  };
322
689
  if (indexes && Object.keys(indexes).length !== 0) {
@@ -324,39 +691,59 @@ const injectDefaultSchema = ({ className, fields, classLevelPermissions, indexes
324
691
  }
325
692
  return defaultSchema;
326
693
  };
327
-
328
- const _HooksSchema = { className: "_Hooks", fields: defaultColumns._Hooks };
329
- const _GlobalConfigSchema = { className: "_GlobalConfig", fields: defaultColumns._GlobalConfig };
694
+ const _HooksSchema = {
695
+ className: '_Hooks',
696
+ fields: defaultColumns._Hooks
697
+ };
698
+ const _GlobalConfigSchema = {
699
+ className: '_GlobalConfig',
700
+ fields: defaultColumns._GlobalConfig
701
+ };
702
+ const _GraphQLConfigSchema = {
703
+ className: '_GraphQLConfig',
704
+ fields: defaultColumns._GraphQLConfig
705
+ };
330
706
  const _PushStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
331
- className: "_PushStatus",
707
+ className: '_PushStatus',
332
708
  fields: {},
333
709
  classLevelPermissions: {}
334
710
  }));
335
711
  const _JobStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
336
- className: "_JobStatus",
712
+ className: '_JobStatus',
337
713
  fields: {},
338
714
  classLevelPermissions: {}
339
715
  }));
340
716
  const _JobScheduleSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
341
- className: "_JobSchedule",
717
+ className: '_JobSchedule',
342
718
  fields: {},
343
719
  classLevelPermissions: {}
344
720
  }));
345
721
  const _AudienceSchema = convertSchemaToAdapterSchema(injectDefaultSchema({
346
- className: "_Audience",
722
+ className: '_Audience',
347
723
  fields: defaultColumns._Audience,
348
724
  classLevelPermissions: {}
349
725
  }));
350
- const VolatileClassesSchemas = [_HooksSchema, _JobStatusSchema, _JobScheduleSchema, _PushStatusSchema, _GlobalConfigSchema, _AudienceSchema];
351
-
726
+ const _IdempotencySchema = convertSchemaToAdapterSchema(injectDefaultSchema({
727
+ className: '_Idempotency',
728
+ fields: defaultColumns._Idempotency,
729
+ classLevelPermissions: {}
730
+ }));
731
+ const VolatileClassesSchemas = exports.VolatileClassesSchemas = [_HooksSchema, _JobStatusSchema, _JobScheduleSchema, _PushStatusSchema, _GlobalConfigSchema, _GraphQLConfigSchema, _AudienceSchema, _IdempotencySchema];
352
732
  const dbTypeMatchesObjectType = (dbType, objectType) => {
353
- if (dbType.type !== objectType.type) return false;
354
- if (dbType.targetClass !== objectType.targetClass) return false;
355
- if (dbType === objectType.type) return true;
356
- if (dbType.type === objectType.type) return true;
733
+ if (dbType.type !== objectType.type) {
734
+ return false;
735
+ }
736
+ if (dbType.targetClass !== objectType.targetClass) {
737
+ return false;
738
+ }
739
+ if (dbType === objectType.type) {
740
+ return true;
741
+ }
742
+ if (dbType.type === objectType.type) {
743
+ return true;
744
+ }
357
745
  return false;
358
746
  };
359
-
360
747
  const typeToString = type => {
361
748
  if (typeof type === 'string') {
362
749
  return type;
@@ -366,108 +753,108 @@ const typeToString = type => {
366
753
  }
367
754
  return `${type.type}`;
368
755
  };
756
+ const ttl = {
757
+ date: Date.now(),
758
+ duration: undefined
759
+ };
369
760
 
370
761
  // Stores the entire schema of the app in a weird hybrid format somewhere between
371
762
  // the mongo format and the Parse format. Soon, this will all be Parse format.
372
763
  class SchemaController {
373
-
374
- constructor(databaseAdapter, schemaCache) {
764
+ constructor(databaseAdapter) {
375
765
  this._dbAdapter = databaseAdapter;
376
- this._cache = schemaCache;
377
- // this.data[className][fieldName] tells you the type of that field, in mongo format
378
- this.data = {};
379
- // this.perms[className][operation] tells you the acl-style permissions
380
- this.perms = {};
381
- // this.indexes[className][operation] tells you the indexes
382
- this.indexes = {};
766
+ const config = _Config.default.get(Parse.applicationId);
767
+ this.schemaData = new SchemaData(_SchemaCache.default.all(), this.protectedFields);
768
+ this.protectedFields = config.protectedFields;
769
+ const customIds = config.allowCustomObjectId;
770
+ const customIdRegEx = /^.{1,}$/u; // 1+ chars
771
+ const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;
772
+ this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;
773
+ this._dbAdapter.watch(() => {
774
+ this.reloadData({
775
+ clearCache: true
776
+ });
777
+ });
383
778
  }
384
-
385
- reloadData(options = { clearCache: false }) {
386
- let promise = Promise.resolve();
387
- if (options.clearCache) {
388
- promise = promise.then(() => {
389
- return this._cache.clear();
779
+ async reloadDataIfNeeded() {
780
+ if (this._dbAdapter.enableSchemaHooks) {
781
+ return;
782
+ }
783
+ const {
784
+ date,
785
+ duration
786
+ } = ttl || {};
787
+ if (!duration) {
788
+ return;
789
+ }
790
+ const now = Date.now();
791
+ if (now - date > duration) {
792
+ ttl.date = now;
793
+ await this.reloadData({
794
+ clearCache: true
390
795
  });
391
796
  }
797
+ }
798
+ reloadData(options = {
799
+ clearCache: false
800
+ }) {
392
801
  if (this.reloadDataPromise && !options.clearCache) {
393
802
  return this.reloadDataPromise;
394
803
  }
395
- this.reloadDataPromise = promise.then(() => {
396
- return this.getAllClasses(options).then(allSchemas => {
397
- const data = {};
398
- const perms = {};
399
- const indexes = {};
400
- allSchemas.forEach(schema => {
401
- data[schema.className] = injectDefaultSchema(schema).fields;
402
- perms[schema.className] = schema.classLevelPermissions;
403
- indexes[schema.className] = schema.indexes;
404
- });
405
-
406
- // Inject the in-memory classes
407
- volatileClasses.forEach(className => {
408
- const schema = injectDefaultSchema({ className, fields: {}, classLevelPermissions: {} });
409
- data[className] = schema.fields;
410
- perms[className] = schema.classLevelPermissions;
411
- indexes[className] = schema.indexes;
412
- });
413
- this.data = data;
414
- this.perms = perms;
415
- this.indexes = indexes;
416
- delete this.reloadDataPromise;
417
- }, err => {
418
- this.data = {};
419
- this.perms = {};
420
- this.indexes = {};
421
- delete this.reloadDataPromise;
422
- throw err;
423
- });
804
+ this.reloadDataPromise = this.getAllClasses(options).then(allSchemas => {
805
+ this.schemaData = new SchemaData(allSchemas, this.protectedFields);
806
+ delete this.reloadDataPromise;
807
+ }, err => {
808
+ this.schemaData = new SchemaData();
809
+ delete this.reloadDataPromise;
810
+ throw err;
424
811
  }).then(() => {});
425
812
  return this.reloadDataPromise;
426
813
  }
427
-
428
- getAllClasses(options = { clearCache: false }) {
429
- let promise = Promise.resolve();
814
+ async getAllClasses(options = {
815
+ clearCache: false
816
+ }) {
430
817
  if (options.clearCache) {
431
- promise = this._cache.clear();
818
+ return this.setAllClasses();
432
819
  }
433
- return promise.then(() => {
434
- return this._cache.getAllClasses();
435
- }).then(allClasses => {
436
- if (allClasses && allClasses.length && !options.clearCache) {
437
- return Promise.resolve(allClasses);
438
- }
439
- return this._dbAdapter.getAllClasses().then(allSchemas => allSchemas.map(injectDefaultSchema)).then(allSchemas => {
440
- return this._cache.setAllClasses(allSchemas).then(() => {
441
- return allSchemas;
442
- });
443
- });
820
+ await this.reloadDataIfNeeded();
821
+ const cached = _SchemaCache.default.all();
822
+ if (cached && cached.length) {
823
+ return Promise.resolve(cached);
824
+ }
825
+ return this.setAllClasses();
826
+ }
827
+ setAllClasses() {
828
+ return this._dbAdapter.getAllClasses().then(allSchemas => allSchemas.map(injectDefaultSchema)).then(allSchemas => {
829
+ _SchemaCache.default.put(allSchemas);
830
+ return allSchemas;
444
831
  });
445
832
  }
446
-
447
- getOneSchema(className, allowVolatileClasses = false, options = { clearCache: false }) {
448
- let promise = Promise.resolve();
833
+ getOneSchema(className, allowVolatileClasses = false, options = {
834
+ clearCache: false
835
+ }) {
449
836
  if (options.clearCache) {
450
- promise = this._cache.clear();
451
- }
452
- return promise.then(() => {
453
- if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {
454
- return Promise.resolve({
455
- className,
456
- fields: this.data[className],
457
- classLevelPermissions: this.perms[className],
458
- indexes: this.indexes[className]
459
- });
460
- }
461
- return this._cache.getOneSchema(className).then(cached => {
462
- if (cached && !options.clearCache) {
463
- return Promise.resolve(cached);
464
- }
465
- return this._dbAdapter.getClass(className).then(injectDefaultSchema).then(result => {
466
- return this._cache.setOneSchema(className, result).then(() => {
467
- return result;
468
- });
469
- });
837
+ _SchemaCache.default.clear();
838
+ }
839
+ if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {
840
+ const data = this.schemaData[className];
841
+ return Promise.resolve({
842
+ className,
843
+ fields: data.fields,
844
+ classLevelPermissions: data.classLevelPermissions,
845
+ indexes: data.indexes
470
846
  });
847
+ }
848
+ const cached = _SchemaCache.default.get(className);
849
+ if (cached && !options.clearCache) {
850
+ return Promise.resolve(cached);
851
+ }
852
+ return this.setAllClasses().then(allSchemas => {
853
+ const oneSchema = allSchemas.find(schema => schema.className === className);
854
+ if (!oneSchema) {
855
+ return Promise.reject(undefined);
856
+ }
857
+ return oneSchema;
471
858
  });
472
859
  }
473
860
 
@@ -478,38 +865,49 @@ class SchemaController {
478
865
  // on success, and rejects with an error on fail. Ensure you
479
866
  // have authorization (master key, or client class creation
480
867
  // enabled) before calling this function.
481
- addClassIfNotExists(className, fields = {}, classLevelPermissions, indexes = {}) {
868
+ async addClassIfNotExists(className, fields = {}, classLevelPermissions, indexes = {}) {
482
869
  var validationError = this.validateNewClass(className, fields, classLevelPermissions);
483
870
  if (validationError) {
871
+ if (validationError instanceof Parse.Error) {
872
+ return Promise.reject(validationError);
873
+ } else if (validationError.code && validationError.error) {
874
+ return Promise.reject(new Parse.Error(validationError.code, validationError.error));
875
+ }
484
876
  return Promise.reject(validationError);
485
877
  }
486
-
487
- return this._dbAdapter.createClass(className, convertSchemaToAdapterSchema({ fields, classLevelPermissions, indexes, className })).then(convertAdapterSchemaToParseSchema).then(res => {
488
- return this._cache.clear().then(() => {
489
- return Promise.resolve(res);
878
+ try {
879
+ const adapterSchema = await this._dbAdapter.createClass(className, convertSchemaToAdapterSchema({
880
+ fields,
881
+ classLevelPermissions,
882
+ indexes,
883
+ className
884
+ }));
885
+ // TODO: Remove by updating schema cache directly
886
+ await this.reloadData({
887
+ clearCache: true
490
888
  });
491
- }).catch(error => {
889
+ const parseSchema = convertAdapterSchemaToParseSchema(adapterSchema);
890
+ return parseSchema;
891
+ } catch (error) {
492
892
  if (error && error.code === Parse.Error.DUPLICATE_VALUE) {
493
893
  throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);
494
894
  } else {
495
895
  throw error;
496
896
  }
497
- });
897
+ }
498
898
  }
499
-
500
899
  updateClass(className, submittedFields, classLevelPermissions, indexes, database) {
501
900
  return this.getOneSchema(className).then(schema => {
502
901
  const existingFields = schema.fields;
503
902
  Object.keys(submittedFields).forEach(name => {
504
903
  const field = submittedFields[name];
505
- if (existingFields[name] && field.__op !== 'Delete') {
904
+ if (existingFields[name] && existingFields[name].type !== field.type && field.__op !== 'Delete') {
506
905
  throw new Parse.Error(255, `Field ${name} exists, cannot update.`);
507
906
  }
508
907
  if (!existingFields[name] && field.__op === 'Delete') {
509
908
  throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);
510
909
  }
511
910
  });
512
-
513
911
  delete existingFields._rperm;
514
912
  delete existingFields._wperm;
515
913
  const newSchema = buildMergedSchemaObject(existingFields, submittedFields);
@@ -531,29 +929,38 @@ class SchemaController {
531
929
  insertedFields.push(fieldName);
532
930
  }
533
931
  });
534
-
535
932
  let deletePromise = Promise.resolve();
536
933
  if (deletedFields.length > 0) {
537
934
  deletePromise = this.deleteFields(deletedFields, className, database);
538
935
  }
936
+ let enforceFields = [];
539
937
  return deletePromise // Delete Everything
540
- .then(() => this.reloadData({ clearCache: true })) // Reload our Schema, so we have all the new values
938
+ .then(() => this.reloadData({
939
+ clearCache: true
940
+ })) // Reload our Schema, so we have all the new values
541
941
  .then(() => {
542
942
  const promises = insertedFields.map(fieldName => {
543
943
  const type = submittedFields[fieldName];
544
944
  return this.enforceFieldExists(className, fieldName, type);
545
945
  });
546
946
  return Promise.all(promises);
547
- }).then(() => this.setPermissions(className, classLevelPermissions, newSchema)).then(() => this._dbAdapter.setIndexesWithSchemaFormat(className, indexes, schema.indexes, fullNewSchema)).then(() => this.reloadData({ clearCache: true }))
947
+ }).then(results => {
948
+ enforceFields = results.filter(result => !!result);
949
+ return this.setPermissions(className, classLevelPermissions, newSchema);
950
+ }).then(() => this._dbAdapter.setIndexesWithSchemaFormat(className, indexes, schema.indexes, fullNewSchema)).then(() => this.reloadData({
951
+ clearCache: true
952
+ }))
548
953
  //TODO: Move this logic into the database adapter
549
954
  .then(() => {
955
+ this.ensureFields(enforceFields);
956
+ const schema = this.schemaData[className];
550
957
  const reloadedSchema = {
551
958
  className: className,
552
- fields: this.data[className],
553
- classLevelPermissions: this.perms[className]
959
+ fields: schema.fields,
960
+ classLevelPermissions: schema.classLevelPermissions
554
961
  };
555
- if (this.indexes[className] && Object.keys(this.indexes[className]).length !== 0) {
556
- reloadedSchema.indexes = this.indexes[className];
962
+ if (schema.indexes && Object.keys(schema.indexes).length !== 0) {
963
+ reloadedSchema.indexes = schema.indexes;
557
964
  }
558
965
  return reloadedSchema;
559
966
  });
@@ -569,33 +976,35 @@ class SchemaController {
569
976
  // Returns a promise that resolves successfully to the new schema
570
977
  // object or fails with a reason.
571
978
  enforceClassExists(className) {
572
- if (this.data[className]) {
979
+ if (this.schemaData[className]) {
573
980
  return Promise.resolve(this);
574
981
  }
575
982
  // We don't have this class. Update the schema
576
- return this.addClassIfNotExists(className)
577
- // The schema update succeeded. Reload the schema
578
- .then(() => this.reloadData({ clearCache: true })).catch(() => {
579
- // The schema update failed. This can be okay - it might
580
- // have failed because there's a race condition and a different
581
- // client is making the exact same schema update that we want.
582
- // So just reload the schema.
583
- return this.reloadData({ clearCache: true });
584
- }).then(() => {
585
- // Ensure that the schema now validates
586
- if (this.data[className]) {
587
- return this;
588
- } else {
589
- throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);
590
- }
591
- }).catch(() => {
592
- // The schema still doesn't validate. Give up
593
- throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');
594
- });
983
+ return (
984
+ // The schema update succeeded. Reload the schema
985
+ this.addClassIfNotExists(className).catch(() => {
986
+ // The schema update failed. This can be okay - it might
987
+ // have failed because there's a race condition and a different
988
+ // client is making the exact same schema update that we want.
989
+ // So just reload the schema.
990
+ return this.reloadData({
991
+ clearCache: true
992
+ });
993
+ }).then(() => {
994
+ // Ensure that the schema now validates
995
+ if (this.schemaData[className]) {
996
+ return this;
997
+ } else {
998
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);
999
+ }
1000
+ }).catch(() => {
1001
+ // The schema still doesn't validate. Give up
1002
+ throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');
1003
+ })
1004
+ );
595
1005
  }
596
-
597
1006
  validateNewClass(className, fields = {}, classLevelPermissions) {
598
- if (this.data[className]) {
1007
+ if (this.schemaData[className]) {
599
1008
  throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);
600
1009
  }
601
1010
  if (!classNameIsValid(className)) {
@@ -606,11 +1015,10 @@ class SchemaController {
606
1015
  }
607
1016
  return this.validateSchemaData(className, fields, classLevelPermissions, []);
608
1017
  }
609
-
610
1018
  validateSchemaData(className, fields, classLevelPermissions, existingFieldNames) {
611
1019
  for (const fieldName in fields) {
612
1020
  if (existingFieldNames.indexOf(fieldName) < 0) {
613
- if (!fieldNameIsValid(fieldName)) {
1021
+ if (!fieldNameIsValid(fieldName, className)) {
614
1022
  return {
615
1023
  code: Parse.Error.INVALID_KEY_NAME,
616
1024
  error: 'invalid field name: ' + fieldName
@@ -622,15 +1030,45 @@ class SchemaController {
622
1030
  error: 'field ' + fieldName + ' cannot be added'
623
1031
  };
624
1032
  }
625
- const error = fieldTypeIsInvalid(fields[fieldName]);
626
- if (error) return { code: error.code, error: error.message };
1033
+ const fieldType = fields[fieldName];
1034
+ const error = fieldTypeIsInvalid(fieldType);
1035
+ if (error) {
1036
+ return {
1037
+ code: error.code,
1038
+ error: error.message
1039
+ };
1040
+ }
1041
+ if (fieldType.defaultValue !== undefined) {
1042
+ let defaultValueType = getType(fieldType.defaultValue);
1043
+ if (typeof defaultValueType === 'string') {
1044
+ defaultValueType = {
1045
+ type: defaultValueType
1046
+ };
1047
+ } else if (typeof defaultValueType === 'object' && fieldType.type === 'Relation') {
1048
+ return {
1049
+ code: Parse.Error.INCORRECT_TYPE,
1050
+ error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`
1051
+ };
1052
+ }
1053
+ if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {
1054
+ return {
1055
+ code: Parse.Error.INCORRECT_TYPE,
1056
+ error: `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(fieldType)} but got ${typeToString(defaultValueType)}`
1057
+ };
1058
+ }
1059
+ } else if (fieldType.required) {
1060
+ if (typeof fieldType === 'object' && fieldType.type === 'Relation') {
1061
+ return {
1062
+ code: Parse.Error.INCORRECT_TYPE,
1063
+ error: `The 'required' option is not applicable for ${typeToString(fieldType)}`
1064
+ };
1065
+ }
1066
+ }
627
1067
  }
628
1068
  }
629
-
630
1069
  for (const fieldName in defaultColumns[className]) {
631
1070
  fields[fieldName] = defaultColumns[className][fieldName];
632
1071
  }
633
-
634
1072
  const geoPoints = Object.keys(fields).filter(key => fields[key] && fields[key].type === 'GeoPoint');
635
1073
  if (geoPoints.length > 1) {
636
1074
  return {
@@ -638,76 +1076,118 @@ class SchemaController {
638
1076
  error: 'currently, only one GeoPoint field may exist in an object. Adding ' + geoPoints[1] + ' when ' + geoPoints[0] + ' already exists.'
639
1077
  };
640
1078
  }
641
- validateCLP(classLevelPermissions, fields);
1079
+ validateCLP(classLevelPermissions, fields, this.userIdRegEx);
642
1080
  }
643
1081
 
644
1082
  // Sets the Class-level permissions for a given className, which must exist.
645
- setPermissions(className, perms, newSchema) {
1083
+ async setPermissions(className, perms, newSchema) {
646
1084
  if (typeof perms === 'undefined') {
647
1085
  return Promise.resolve();
648
1086
  }
649
- validateCLP(perms, newSchema);
650
- return this._dbAdapter.setClassLevelPermissions(className, perms);
1087
+ validateCLP(perms, newSchema, this.userIdRegEx);
1088
+ await this._dbAdapter.setClassLevelPermissions(className, perms);
1089
+ const cached = _SchemaCache.default.get(className);
1090
+ if (cached) {
1091
+ cached.classLevelPermissions = perms;
1092
+ }
651
1093
  }
652
1094
 
653
1095
  // Returns a promise that resolves successfully to the new schema
654
1096
  // object if the provided className-fieldName-type tuple is valid.
655
1097
  // The className must already be validated.
656
1098
  // If 'freeze' is true, refuse to update the schema for this field.
657
- enforceFieldExists(className, fieldName, type) {
658
- if (fieldName.indexOf(".") > 0) {
659
- // subdocument key (x.y) => ok if x is of type 'object'
660
- fieldName = fieldName.split(".")[0];
661
- type = 'Object';
1099
+ enforceFieldExists(className, fieldName, type, isValidation, maintenance) {
1100
+ if (fieldName.indexOf('.') > 0) {
1101
+ // "<array>.<index>" for Nested Arrays
1102
+ // "<embedded document>.<field>" for Nested Objects
1103
+ // JSON Arrays are treated as Nested Objects
1104
+ const [x, y] = fieldName.split('.');
1105
+ fieldName = x;
1106
+ const isArrayIndex = Array.from(y).every(c => c >= '0' && c <= '9');
1107
+ if (isArrayIndex && !['sentPerUTCOffset', 'failedPerUTCOffset'].includes(fieldName)) {
1108
+ type = 'Array';
1109
+ } else {
1110
+ type = 'Object';
1111
+ }
1112
+ }
1113
+ let fieldNameToValidate = `${fieldName}`;
1114
+ if (maintenance && fieldNameToValidate.charAt(0) === '_') {
1115
+ fieldNameToValidate = fieldNameToValidate.substring(1);
662
1116
  }
663
- if (!fieldNameIsValid(fieldName)) {
1117
+ if (!fieldNameIsValid(fieldNameToValidate, className)) {
664
1118
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);
665
1119
  }
666
1120
 
667
1121
  // If someone tries to create a new field with null/undefined as the value, return;
668
1122
  if (!type) {
669
- return Promise.resolve(this);
1123
+ return undefined;
670
1124
  }
671
-
672
- return this.reloadData().then(() => {
1125
+ const expectedType = this.getExpectedType(className, fieldName);
1126
+ if (typeof type === 'string') {
1127
+ type = {
1128
+ type
1129
+ };
1130
+ }
1131
+ if (type.defaultValue !== undefined) {
1132
+ let defaultValueType = getType(type.defaultValue);
1133
+ if (typeof defaultValueType === 'string') {
1134
+ defaultValueType = {
1135
+ type: defaultValueType
1136
+ };
1137
+ }
1138
+ if (!dbTypeMatchesObjectType(type, defaultValueType)) {
1139
+ throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(type)} but got ${typeToString(defaultValueType)}`);
1140
+ }
1141
+ }
1142
+ if (expectedType) {
1143
+ if (!dbTypeMatchesObjectType(expectedType, type)) {
1144
+ throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName}; expected ${typeToString(expectedType)} but got ${typeToString(type)}`);
1145
+ }
1146
+ // If type options do not change
1147
+ // we can safely return
1148
+ if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {
1149
+ return undefined;
1150
+ }
1151
+ // Field options are may be changed
1152
+ // ensure to have an update to date schema field
1153
+ return this._dbAdapter.updateFieldOptions(className, fieldName, type);
1154
+ }
1155
+ return this._dbAdapter.addFieldIfNotExists(className, fieldName, type).catch(error => {
1156
+ if (error.code == Parse.Error.INCORRECT_TYPE) {
1157
+ // Make sure that we throw errors when it is appropriate to do so.
1158
+ throw error;
1159
+ }
1160
+ // The update failed. This can be okay - it might have been a race
1161
+ // condition where another client updated the schema in the same
1162
+ // way that we wanted to. So, just reload the schema
1163
+ return Promise.resolve();
1164
+ }).then(() => {
1165
+ return {
1166
+ className,
1167
+ fieldName,
1168
+ type
1169
+ };
1170
+ });
1171
+ }
1172
+ ensureFields(fields) {
1173
+ for (let i = 0; i < fields.length; i += 1) {
1174
+ const {
1175
+ className,
1176
+ fieldName
1177
+ } = fields[i];
1178
+ let {
1179
+ type
1180
+ } = fields[i];
673
1181
  const expectedType = this.getExpectedType(className, fieldName);
674
1182
  if (typeof type === 'string') {
675
- type = { type };
1183
+ type = {
1184
+ type: type
1185
+ };
676
1186
  }
677
-
678
- if (expectedType) {
679
- if (!dbTypeMatchesObjectType(expectedType, type)) {
680
- throw new Parse.Error(Parse.Error.INCORRECT_TYPE, `schema mismatch for ${className}.${fieldName}; expected ${typeToString(expectedType)} but got ${typeToString(type)}`);
681
- }
682
- return this;
1187
+ if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {
1188
+ throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);
683
1189
  }
684
-
685
- return this._dbAdapter.addFieldIfNotExists(className, fieldName, type).then(() => {
686
- // The update succeeded. Reload the schema
687
- return this.reloadData({ clearCache: true });
688
- }, error => {
689
- if (error.code == Parse.Error.INCORRECT_TYPE) {
690
- // Make sure that we throw errors when it is appropriate to do so.
691
- throw error;
692
- }
693
- // The update failed. This can be okay - it might have been a race
694
- // condition where another client updated the schema in the same
695
- // way that we wanted to. So, just reload the schema
696
- return this.reloadData({ clearCache: true });
697
- }).then(() => {
698
- // Ensure that the schema now validates
699
- const expectedType = this.getExpectedType(className, fieldName);
700
- if (typeof type === 'string') {
701
- type = { type };
702
- }
703
- if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {
704
- throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);
705
- }
706
- // Remove the cached schema
707
- this._cache.clear();
708
- return this;
709
- });
710
- });
1190
+ }
711
1191
  }
712
1192
 
713
1193
  // maintain compatibility
@@ -726,9 +1206,8 @@ class SchemaController {
726
1206
  if (!classNameIsValid(className)) {
727
1207
  throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));
728
1208
  }
729
-
730
1209
  fieldNames.forEach(fieldName => {
731
- if (!fieldNameIsValid(fieldName)) {
1210
+ if (!fieldNameIsValid(fieldName, className)) {
732
1211
  throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);
733
1212
  }
734
1213
  //Don't allow deleting the default fields.
@@ -736,8 +1215,9 @@ class SchemaController {
736
1215
  throw new Parse.Error(136, `field ${fieldName} cannot be changed`);
737
1216
  }
738
1217
  });
739
-
740
- return this.getOneSchema(className, false, { clearCache: true }).catch(error => {
1218
+ return this.getOneSchema(className, false, {
1219
+ clearCache: true
1220
+ }).catch(error => {
741
1221
  if (error === undefined) {
742
1222
  throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} does not exist.`);
743
1223
  } else {
@@ -749,8 +1229,9 @@ class SchemaController {
749
1229
  throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);
750
1230
  }
751
1231
  });
752
-
753
- const schemaFields = _extends({}, schema.fields);
1232
+ const schemaFields = {
1233
+ ...schema.fields
1234
+ };
754
1235
  return database.adapter.deleteFields(className, schema, fieldNames).then(() => {
755
1236
  return Promise.all(fieldNames.map(fieldName => {
756
1237
  const field = schemaFields[fieldName];
@@ -762,31 +1243,30 @@ class SchemaController {
762
1243
  }));
763
1244
  });
764
1245
  }).then(() => {
765
- this._cache.clear();
1246
+ _SchemaCache.default.clear();
766
1247
  });
767
1248
  }
768
1249
 
769
1250
  // Validates an object provided in REST format.
770
1251
  // Returns a promise that resolves to the new schema if this object is
771
1252
  // valid.
772
- validateObject(className, object, query) {
1253
+ async validateObject(className, object, query, maintenance) {
773
1254
  let geocount = 0;
774
- let promise = this.enforceClassExists(className);
1255
+ const schema = await this.enforceClassExists(className);
1256
+ const promises = [];
775
1257
  for (const fieldName in object) {
776
- if (object[fieldName] === undefined) {
777
- continue;
778
- }
779
- const expected = getType(object[fieldName]);
780
- if (expected === 'GeoPoint') {
1258
+ if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {
781
1259
  geocount++;
782
1260
  }
783
1261
  if (geocount > 1) {
784
- // Make sure all field validation operations run before we return.
785
- // If not - we are continuing to run logic, but already provided response from the server.
786
- return promise.then(() => {
787
- return Promise.reject(new Parse.Error(Parse.Error.INCORRECT_TYPE, 'there can only be one geopoint field in a class'));
788
- });
1262
+ return Promise.reject(new Parse.Error(Parse.Error.INCORRECT_TYPE, 'there can only be one geopoint field in a class'));
789
1263
  }
1264
+ }
1265
+ for (const fieldName in object) {
1266
+ if (object[fieldName] === undefined) {
1267
+ continue;
1268
+ }
1269
+ const expected = getType(object[fieldName]);
790
1270
  if (!expected) {
791
1271
  continue;
792
1272
  }
@@ -794,23 +1274,30 @@ class SchemaController {
794
1274
  // Every object has ACL implicitly.
795
1275
  continue;
796
1276
  }
797
-
798
- promise = promise.then(schema => schema.enforceFieldExists(className, fieldName, expected));
1277
+ promises.push(schema.enforceFieldExists(className, fieldName, expected, true, maintenance));
1278
+ }
1279
+ const results = await Promise.all(promises);
1280
+ const enforceFields = results.filter(result => !!result);
1281
+ if (enforceFields.length !== 0) {
1282
+ // TODO: Remove by updating schema cache directly
1283
+ await this.reloadData({
1284
+ clearCache: true
1285
+ });
799
1286
  }
800
- promise = thenValidateRequiredColumns(promise, className, object, query);
801
- return promise;
1287
+ this.ensureFields(enforceFields);
1288
+ const promise = Promise.resolve(schema);
1289
+ return thenValidateRequiredColumns(promise, className, object, query);
802
1290
  }
803
1291
 
804
1292
  // Validates that all the properties are set for the object
805
1293
  validateRequiredColumns(className, object, query) {
806
- const columns = requiredColumns[className];
1294
+ const columns = requiredColumns.write[className];
807
1295
  if (!columns || columns.length == 0) {
808
1296
  return Promise.resolve(this);
809
1297
  }
810
-
811
1298
  const missingColumns = columns.filter(function (column) {
812
1299
  if (query && query.objectId) {
813
- if (object[column] && typeof object[column] === "object") {
1300
+ if (object[column] && typeof object[column] === 'object') {
814
1301
  // Trying to delete a required column
815
1302
  return object[column].__op == 'Delete';
816
1303
  }
@@ -819,21 +1306,21 @@ class SchemaController {
819
1306
  }
820
1307
  return !object[column];
821
1308
  });
822
-
823
1309
  if (missingColumns.length > 0) {
824
1310
  throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');
825
1311
  }
826
1312
  return Promise.resolve(this);
827
1313
  }
1314
+ testPermissionsForClassName(className, aclGroup, operation) {
1315
+ return SchemaController.testPermissions(this.getClassLevelPermissions(className), aclGroup, operation);
1316
+ }
828
1317
 
829
- // Validates the base CLP for an operation
830
- testBaseCLP(className, aclGroup, operation) {
831
- if (!this.perms[className] || !this.perms[className][operation]) {
1318
+ // Tests that the class level permission let pass the operation for a given aclGroup
1319
+ static testPermissions(classPermissions, aclGroup, operation) {
1320
+ if (!classPermissions || !classPermissions[operation]) {
832
1321
  return true;
833
1322
  }
834
- const classPerms = this.perms[className];
835
- const perms = classPerms[operation];
836
- // Handle the public scenario quickly
1323
+ const perms = classPermissions[operation];
837
1324
  if (perms['*']) {
838
1325
  return true;
839
1326
  }
@@ -847,26 +1334,23 @@ class SchemaController {
847
1334
  }
848
1335
 
849
1336
  // Validates an operation passes class-level-permissions set in the schema
850
- validatePermission(className, aclGroup, operation) {
851
-
852
- if (this.testBaseCLP(className, aclGroup, operation)) {
1337
+ static validatePermission(classPermissions, className, aclGroup, operation, action) {
1338
+ if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {
853
1339
  return Promise.resolve();
854
1340
  }
855
-
856
- if (!this.perms[className] || !this.perms[className][operation]) {
1341
+ if (!classPermissions || !classPermissions[operation]) {
857
1342
  return true;
858
1343
  }
859
- const classPerms = this.perms[className];
860
- const perms = classPerms[operation];
861
-
1344
+ const perms = classPermissions[operation];
1345
+ const config = _Config.default.get(Parse.applicationId);
862
1346
  // If only for authenticated users
863
1347
  // make sure we have an aclGroup
864
1348
  if (perms['requiresAuthentication']) {
865
1349
  // If aclGroup has * (public)
866
1350
  if (!aclGroup || aclGroup.length == 0) {
867
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.');
1351
+ throw (0, _Error.createSanitizedError)(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.', config);
868
1352
  } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {
869
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.');
1353
+ throw (0, _Error.createSanitizedError)(Parse.Error.OBJECT_NOT_FOUND, 'Permission denied, user needs to be authenticated.', config);
870
1354
  }
871
1355
  // requiresAuthentication passed, just move forward
872
1356
  // probably would be wise at some point to rename to 'authenticatedUser'
@@ -879,21 +1363,37 @@ class SchemaController {
879
1363
 
880
1364
  // Reject create when write lockdown
881
1365
  if (permissionField == 'writeUserFields' && operation == 'create') {
882
- throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`);
1366
+ throw (0, _Error.createSanitizedError)(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`, config);
883
1367
  }
884
1368
 
885
1369
  // Process the readUserFields later
886
- if (Array.isArray(classPerms[permissionField]) && classPerms[permissionField].length > 0) {
1370
+ if (Array.isArray(classPermissions[permissionField]) && classPermissions[permissionField].length > 0) {
887
1371
  return Promise.resolve();
888
1372
  }
889
- throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`);
1373
+ const pointerFields = classPermissions[operation].pointerFields;
1374
+ if (Array.isArray(pointerFields) && pointerFields.length > 0) {
1375
+ // any op except 'addField as part of create' is ok.
1376
+ if (operation !== 'addField' || action === 'update') {
1377
+ // We can allow adding field on update flow only.
1378
+ return Promise.resolve();
1379
+ }
1380
+ }
1381
+ throw (0, _Error.createSanitizedError)(Parse.Error.OPERATION_FORBIDDEN, `Permission denied for action ${operation} on class ${className}.`, config);
1382
+ }
1383
+
1384
+ // Validates an operation passes class-level-permissions set in the schema
1385
+ validatePermission(className, aclGroup, operation, action) {
1386
+ return SchemaController.validatePermission(this.getClassLevelPermissions(className), className, aclGroup, operation, action);
1387
+ }
1388
+ getClassLevelPermissions(className) {
1389
+ return this.schemaData[className] && this.schemaData[className].classLevelPermissions;
890
1390
  }
891
1391
 
892
1392
  // Returns the expected type for a className+key combination
893
1393
  // or undefined if the schema is not set
894
1394
  getExpectedType(className, fieldName) {
895
- if (this.data && this.data[className]) {
896
- const expectedType = this.data[className][fieldName];
1395
+ if (this.schemaData[className]) {
1396
+ const expectedType = this.schemaData[className].fields[fieldName];
897
1397
  return expectedType === 'map' ? 'Object' : expectedType;
898
1398
  }
899
1399
  return undefined;
@@ -901,14 +1401,18 @@ class SchemaController {
901
1401
 
902
1402
  // Checks if a given class is in the schema.
903
1403
  hasClass(className) {
904
- return this.reloadData().then(() => !!this.data[className]);
1404
+ if (this.schemaData[className]) {
1405
+ return Promise.resolve(true);
1406
+ }
1407
+ return this.reloadData().then(() => !!this.schemaData[className]);
905
1408
  }
906
1409
  }
907
1410
 
908
- exports.default = SchemaController; // Returns a promise for a new Schema.
909
-
910
- const load = (dbAdapter, schemaCache, options) => {
911
- const schema = new SchemaController(dbAdapter, schemaCache);
1411
+ // Returns a promise for a new Schema.
1412
+ exports.SchemaController = exports.default = SchemaController;
1413
+ const load = (dbAdapter, options) => {
1414
+ const schema = new SchemaController(dbAdapter);
1415
+ ttl.duration = dbAdapter.schemaCacheTtl;
912
1416
  return schema.reloadData(options).then(() => schema);
913
1417
  };
914
1418
 
@@ -917,6 +1421,7 @@ const load = (dbAdapter, schemaCache, options) => {
917
1421
  // does not include the default fields, as it is intended to be passed
918
1422
  // to mongoSchemaFromFieldsAndClassName. No validation is done here, it
919
1423
  // is done in mongoSchemaFromFieldsAndClassName.
1424
+ exports.load = load;
920
1425
  function buildMergedSchemaObject(existingFields, putRequest) {
921
1426
  const newSchema = {};
922
1427
  // -disable-next
@@ -1030,7 +1535,7 @@ function getObjectType(obj) {
1030
1535
  }
1031
1536
  break;
1032
1537
  }
1033
- throw new Parse.Error(Parse.Error.INCORRECT_TYPE, "This is not a valid " + obj.__type);
1538
+ throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);
1034
1539
  }
1035
1540
  if (obj['$ne']) {
1036
1541
  return getObjectType(obj['$ne']);
@@ -1059,15 +1564,4 @@ function getObjectType(obj) {
1059
1564
  }
1060
1565
  return 'Object';
1061
1566
  }
1062
-
1063
- exports.load = load;
1064
- exports.classNameIsValid = classNameIsValid;
1065
- exports.fieldNameIsValid = fieldNameIsValid;
1066
- exports.invalidClassNameMessage = invalidClassNameMessage;
1067
- exports.buildMergedSchemaObject = buildMergedSchemaObject;
1068
- exports.systemClasses = systemClasses;
1069
- exports.defaultColumns = defaultColumns;
1070
- exports.convertSchemaToAdapterSchema = convertSchemaToAdapterSchema;
1071
- exports.VolatileClassesSchemas = VolatileClassesSchemas;
1072
- exports.SchemaController = SchemaController;
1073
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/SchemaController.js"],"names":["Parse","require","defaultColumns","Object","freeze","_Default","type","_User","_Installation","_Role","targetClass","_Session","_Product","_PushStatus","_JobStatus","_JobSchedule","_Hooks","_GlobalConfig","_Audience","requiredColumns","systemClasses","volatileClasses","userIdRegex","roleRegex","publicRegex","requireAuthenticationRegex","permissionKeyRegex","verifyPermissionKey","key","result","reduce","isGood","regEx","match","Error","INVALID_JSON","CLPValidKeys","validateCLP","perms","fields","keys","forEach","operation","indexOf","Array","isArray","perm","joinClassRegex","classAndFieldRegex","classNameIsValid","className","test","fieldNameIsValid","fieldName","fieldNameIsValidForClass","invalidClassNameMessage","invalidJsonError","validNonRelationOrPointerTypes","fieldTypeIsInvalid","INVALID_CLASS_NAME","undefined","INCORRECT_TYPE","convertSchemaToAdapterSchema","schema","injectDefaultSchema","ACL","_rperm","_wperm","password","_hashed_password","convertAdapterSchemaToParseSchema","authData","indexes","length","classLevelPermissions","defaultSchema","_HooksSchema","_GlobalConfigSchema","_PushStatusSchema","_JobStatusSchema","_JobScheduleSchema","_AudienceSchema","VolatileClassesSchemas","dbTypeMatchesObjectType","dbType","objectType","typeToString","SchemaController","constructor","databaseAdapter","schemaCache","_dbAdapter","_cache","data","reloadData","options","clearCache","promise","Promise","resolve","then","clear","reloadDataPromise","getAllClasses","allSchemas","err","allClasses","map","setAllClasses","getOneSchema","allowVolatileClasses","cached","getClass","setOneSchema","addClassIfNotExists","validationError","validateNewClass","reject","createClass","res","catch","error","code","DUPLICATE_VALUE","updateClass","submittedFields","database","existingFields","name","field","__op","newSchema","buildMergedSchemaObject","defaultFields","fullNewSchema","assign","validateSchemaData","deletedFields","insertedFields","push","deletePromise","deleteFields","promises","enforceFieldExists","all","setPermissions","setIndexesWithSchemaFormat","reloadedSchema","enforceClassExists","existingFieldNames","INVALID_KEY_NAME","message","geoPoints","filter","setClassLevelPermissions","split","expectedType","getExpectedType","addFieldIfNotExists","deleteField","fieldNames","schemaFields","adapter","deleteClass","validateObject","object","query","geocount","expected","getType","thenValidateRequiredColumns","validateRequiredColumns","columns","missingColumns","column","objectId","testBaseCLP","aclGroup","classPerms","some","acl","validatePermission","OBJECT_NOT_FOUND","permissionField","OPERATION_FORBIDDEN","hasClass","load","dbAdapter","putRequest","sysSchemaField","_id","oldField","fieldIsDeleted","newField","schemaPromise","obj","getObjectType","__type","iso","latitude","longitude","base64","coordinates","objects","ops"],"mappings":"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","file":"SchemaController.js","sourcesContent":["// @flow\n// This class handles schema validation, persistence, and modification.\n//\n// Each individual Schema object should be immutable. The helpers to\n// do things with the Schema just return a new schema when the schema\n// is changed.\n//\n// The canonical place to store this Schema is in the database itself,\n// in a _SCHEMA collection. This is not the right way to do it for an\n// open source framework, but it's backward compatible, so we're\n// keeping it this way for now.\n//\n// In API-handling code, you should only use the Schema class via the\n// DatabaseController. This will let us replace the schema logic for\n// different databases.\n// TODO: hide all schema logic inside the database adapter.\n// @flow-disable-next\nconst Parse = require('parse/node').Parse;\nimport { StorageAdapter }     from '../Adapters/Storage/StorageAdapter';\nimport DatabaseController     from './DatabaseController';\nimport type {\n  Schema,\n  SchemaFields,\n  ClassLevelPermissions,\n  SchemaField,\n  LoadSchemaOptions,\n} from './types';\n\nconst defaultColumns: {[string]: SchemaFields} = Object.freeze({\n  // Contain the default columns for every parse object type (except _Join collection)\n  _Default: {\n    \"objectId\":  {type:'String'},\n    \"createdAt\": {type:'Date'},\n    \"updatedAt\": {type:'Date'},\n    \"ACL\":       {type:'ACL'},\n  },\n  // The additional default columns for the _User collection (in addition to DefaultCols)\n  _User: {\n    \"username\":      {type:'String'},\n    \"password\":      {type:'String'},\n    \"email\":         {type:'String'},\n    \"emailVerified\": {type:'Boolean'},\n    \"authData\":      {type:'Object'}\n  },\n  // The additional default columns for the _Installation collection (in addition to DefaultCols)\n  _Installation: {\n    \"installationId\":   {type:'String'},\n    \"deviceToken\":      {type:'String'},\n    \"channels\":         {type:'Array'},\n    \"deviceType\":       {type:'String'},\n    \"pushType\":         {type:'String'},\n    \"GCMSenderId\":      {type:'String'},\n    \"timeZone\":         {type:'String'},\n    \"localeIdentifier\": {type:'String'},\n    \"badge\":            {type:'Number'},\n    \"appVersion\":       {type:'String'},\n    \"appName\":          {type:'String'},\n    \"appIdentifier\":    {type:'String'},\n    \"parseVersion\":     {type:'String'},\n  },\n  // The additional default columns for the _Role collection (in addition to DefaultCols)\n  _Role: {\n    \"name\":  {type:'String'},\n    \"users\": {type:'Relation', targetClass:'_User'},\n    \"roles\": {type:'Relation', targetClass:'_Role'}\n  },\n  // The additional default columns for the _Session collection (in addition to DefaultCols)\n  _Session: {\n    \"restricted\":     {type:'Boolean'},\n    \"user\":           {type:'Pointer', targetClass:'_User'},\n    \"installationId\": {type:'String'},\n    \"sessionToken\":   {type:'String'},\n    \"expiresAt\":      {type:'Date'},\n    \"createdWith\":    {type:'Object'}\n  },\n  _Product: {\n    \"productIdentifier\":  {type:'String'},\n    \"download\":           {type:'File'},\n    \"downloadName\":       {type:'String'},\n    \"icon\":               {type:'File'},\n    \"order\":              {type:'Number'},\n    \"title\":              {type:'String'},\n    \"subtitle\":           {type:'String'},\n  },\n  _PushStatus: {\n    \"pushTime\":            {type:'String'},\n    \"source\":              {type:'String'}, // rest or webui\n    \"query\":               {type:'String'}, // the stringified JSON query\n    \"payload\":             {type:'String'}, // the stringified JSON payload,\n    \"title\":               {type:'String'},\n    \"expiry\":              {type:'Number'},\n    \"expiration_interval\": {type:'Number'},\n    \"status\":              {type:'String'},\n    \"numSent\":             {type:'Number'},\n    \"numFailed\":           {type:'Number'},\n    \"pushHash\":            {type:'String'},\n    \"errorMessage\":        {type:'Object'},\n    \"sentPerType\":         {type:'Object'},\n    \"failedPerType\":       {type:'Object'},\n    \"sentPerUTCOffset\":    {type:'Object'},\n    \"failedPerUTCOffset\":  {type:'Object'},\n    \"count\":               {type:'Number'} // tracks # of batches queued and pending\n  },\n  _JobStatus: {\n    \"jobName\":    {type: 'String'},\n    \"source\":     {type: 'String'},\n    \"status\":     {type: 'String'},\n    \"message\":    {type: 'String'},\n    \"params\":     {type: 'Object'}, // params received when calling the job\n    \"finishedAt\": {type: 'Date'}\n  },\n  _JobSchedule: {\n    \"jobName\":      {type:'String'},\n    \"description\":  {type:'String'},\n    \"params\":       {type:'String'},\n    \"startAfter\":   {type:'String'},\n    \"daysOfWeek\":   {type:'Array'},\n    \"timeOfDay\":    {type:'String'},\n    \"lastRun\":      {type:'Number'},\n    \"repeatMinutes\":{type:'Number'}\n  },\n  _Hooks: {\n    \"functionName\": {type:'String'},\n    \"className\":    {type:'String'},\n    \"triggerName\":  {type:'String'},\n    \"url\":          {type:'String'}\n  },\n  _GlobalConfig: {\n    \"objectId\": {type: 'String'},\n    \"params\":   {type: 'Object'}\n  },\n  _Audience: {\n    \"objectId\":  {type:'String'},\n    \"name\":      {type:'String'},\n    \"query\":     {type:'String'}, //storing query as JSON string to prevent \"Nested keys should not contain the '$' or '.' characters\" error\n    \"lastUsed\":  {type:'Date'},\n    \"timesUsed\": {type:'Number'}\n  }\n});\n\nconst requiredColumns = Object.freeze({\n  _Product: [\"productIdentifier\", \"icon\", \"order\", \"title\", \"subtitle\"],\n  _Role: [\"name\", \"ACL\"]\n});\n\nconst systemClasses = Object.freeze(['_User', '_Installation', '_Role', '_Session', '_Product', '_PushStatus', '_JobStatus', '_JobSchedule', '_Audience']);\n\nconst volatileClasses = Object.freeze(['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_JobSchedule', '_Audience']);\n\n// 10 alpha numberic chars + uppercase\nconst userIdRegex = /^[a-zA-Z0-9]{10}$/;\n// Anything that start with role\nconst roleRegex = /^role:.*/;\n// * permission\nconst publicRegex = /^\\*$/\n\nconst requireAuthenticationRegex = /^requiresAuthentication$/\n\nconst permissionKeyRegex = Object.freeze([userIdRegex, roleRegex, publicRegex, requireAuthenticationRegex]);\n\nfunction verifyPermissionKey(key) {\n  const result = permissionKeyRegex.reduce((isGood, regEx) => {\n    isGood = isGood || key.match(regEx) != null;\n    return isGood;\n  }, false);\n  if (!result) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid key for class level permissions`);\n  }\n}\n\nconst CLPValidKeys = Object.freeze(['find', 'count', 'get', 'create', 'update', 'delete', 'addField', 'readUserFields', 'writeUserFields']);\nfunction validateCLP(perms: ClassLevelPermissions, fields: SchemaFields) {\n  if (!perms) {\n    return;\n  }\n  Object.keys(perms).forEach((operation) => {\n    if (CLPValidKeys.indexOf(operation) == -1) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, `${operation} is not a valid operation for class level permissions`);\n    }\n    if (!perms[operation]) {\n      return;\n    }\n\n    if (operation === 'readUserFields' || operation === 'writeUserFields') {\n      if (!Array.isArray(perms[operation])) {\n        // @flow-disable-next\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `'${perms[operation]}' is not a valid value for class level permissions ${operation}`);\n      } else {\n        perms[operation].forEach((key) => {\n          if (!fields[key] || fields[key].type != 'Pointer' || fields[key].targetClass != '_User') {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, `'${key}' is not a valid column for class level pointer permissions ${operation}`);\n          }\n        });\n      }\n      return;\n    }\n\n    // @flow-disable-next\n    Object.keys(perms[operation]).forEach((key) => {\n      verifyPermissionKey(key);\n      // @flow-disable-next\n      const perm = perms[operation][key];\n      if (perm !== true) {\n        // @flow-disable-next\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `'${perm}' is not a valid value for class level permissions ${operation}:${key}:${perm}`);\n      }\n    });\n  });\n}\nconst joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;\nconst classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;\nfunction classNameIsValid(className: string): boolean {\n  // Valid classes must:\n  return (\n    // Be one of _User, _Installation, _Role, _Session OR\n    systemClasses.indexOf(className) > -1 ||\n    // Be a join table OR\n    joinClassRegex.test(className) ||\n    // Include only alpha-numeric and underscores, and not start with an underscore or number\n    fieldNameIsValid(className)\n  );\n}\n\n// Valid fields must be alpha-numeric, and not start with an underscore or number\nfunction fieldNameIsValid(fieldName: string): boolean {\n  return classAndFieldRegex.test(fieldName);\n}\n\n// Checks that it's not trying to clobber one of the default fields of the class.\nfunction fieldNameIsValidForClass(fieldName: string, className: string): boolean {\n  if (!fieldNameIsValid(fieldName)) {\n    return false;\n  }\n  if (defaultColumns._Default[fieldName]) {\n    return false;\n  }\n  if (defaultColumns[className] && defaultColumns[className][fieldName]) {\n    return false;\n  }\n  return true;\n}\n\nfunction invalidClassNameMessage(className: string): string {\n  return 'Invalid classname: ' + className + ', classnames can only have alphanumeric characters and _, and must start with an alpha character ';\n}\n\nconst invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, \"invalid JSON\");\nconst validNonRelationOrPointerTypes = [\n  'Number',\n  'String',\n  'Boolean',\n  'Date',\n  'Object',\n  'Array',\n  'GeoPoint',\n  'File',\n  'Bytes',\n  'Polygon'\n];\n// Returns an error suitable for throwing if the type is invalid\nconst fieldTypeIsInvalid = ({ type, targetClass }) => {\n  if (['Pointer', 'Relation'].indexOf(type) >= 0) {\n    if (!targetClass) {\n      return new Parse.Error(135, `type ${type} needs a class name`);\n    } else if (typeof targetClass !== 'string') {\n      return invalidJsonError;\n    } else if (!classNameIsValid(targetClass)) {\n      return new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(targetClass));\n    } else {\n      return undefined;\n    }\n  }\n  if (typeof type !== 'string') {\n    return invalidJsonError;\n  }\n  if (validNonRelationOrPointerTypes.indexOf(type) < 0) {\n    return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);\n  }\n  return undefined;\n}\n\nconst convertSchemaToAdapterSchema = (schema: any) => {\n  schema = injectDefaultSchema(schema);\n  delete schema.fields.ACL;\n  schema.fields._rperm = { type: 'Array' };\n  schema.fields._wperm = { type: 'Array' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.password;\n    schema.fields._hashed_password = { type: 'String' };\n  }\n\n  return schema;\n}\n\nconst convertAdapterSchemaToParseSchema = ({...schema}) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  schema.fields.ACL = { type: 'ACL' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.authData; //Auth data is implicit\n    delete schema.fields._hashed_password;\n    schema.fields.password = { type: 'String' };\n  }\n\n  if (schema.indexes && Object.keys(schema.indexes).length === 0) {\n    delete schema.indexes;\n  }\n\n  return schema;\n}\n\nconst injectDefaultSchema = ({className, fields, classLevelPermissions, indexes}: Schema) => {\n  const defaultSchema: Schema = {\n    className,\n    fields: {\n      ...defaultColumns._Default,\n      ...(defaultColumns[className] || {}),\n      ...fields,\n    },\n    classLevelPermissions,\n  };\n  if (indexes && Object.keys(indexes).length !== 0) {\n    defaultSchema.indexes = indexes;\n  }\n  return defaultSchema;\n};\n\nconst _HooksSchema =  {className: \"_Hooks\", fields: defaultColumns._Hooks};\nconst _GlobalConfigSchema = { className: \"_GlobalConfig\", fields: defaultColumns._GlobalConfig }\nconst _PushStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({\n  className: \"_PushStatus\",\n  fields: {},\n  classLevelPermissions: {}\n}));\nconst _JobStatusSchema = convertSchemaToAdapterSchema(injectDefaultSchema({\n  className: \"_JobStatus\",\n  fields: {},\n  classLevelPermissions: {}\n}));\nconst _JobScheduleSchema = convertSchemaToAdapterSchema(injectDefaultSchema({\n  className: \"_JobSchedule\",\n  fields: {},\n  classLevelPermissions: {}\n}));\nconst _AudienceSchema = convertSchemaToAdapterSchema(injectDefaultSchema({\n  className: \"_Audience\",\n  fields: defaultColumns._Audience,\n  classLevelPermissions: {}\n}));\nconst VolatileClassesSchemas = [_HooksSchema, _JobStatusSchema, _JobScheduleSchema, _PushStatusSchema, _GlobalConfigSchema, _AudienceSchema];\n\nconst dbTypeMatchesObjectType = (dbType: SchemaField | string, objectType: SchemaField) => {\n  if (dbType.type !== objectType.type) return false;\n  if (dbType.targetClass !== objectType.targetClass) return false;\n  if (dbType === objectType.type) return true;\n  if (dbType.type === objectType.type) return true;\n  return false;\n}\n\nconst typeToString = (type: SchemaField | string): string => {\n  if (typeof type === 'string') {\n    return type;\n  }\n  if (type.targetClass) {\n    return `${type.type}<${type.targetClass}>`;\n  }\n  return `${type.type}`;\n}\n\n// Stores the entire schema of the app in a weird hybrid format somewhere between\n// the mongo format and the Parse format. Soon, this will all be Parse format.\nexport default class SchemaController {\n  _dbAdapter: StorageAdapter;\n  data: any;\n  perms: any;\n  indexes: any;\n  _cache: any;\n  reloadDataPromise: Promise<any>;\n\n  constructor(databaseAdapter: StorageAdapter, schemaCache: any) {\n    this._dbAdapter = databaseAdapter;\n    this._cache = schemaCache;\n    // this.data[className][fieldName] tells you the type of that field, in mongo format\n    this.data = {};\n    // this.perms[className][operation] tells you the acl-style permissions\n    this.perms = {};\n    // this.indexes[className][operation] tells you the indexes\n    this.indexes = {};\n  }\n\n  reloadData(options: LoadSchemaOptions = {clearCache: false}): Promise<any> {\n    let promise = Promise.resolve();\n    if (options.clearCache) {\n      promise = promise.then(() => {\n        return this._cache.clear();\n      });\n    }\n    if (this.reloadDataPromise && !options.clearCache) {\n      return this.reloadDataPromise;\n    }\n    this.reloadDataPromise = promise.then(() => {\n      return this.getAllClasses(options).then((allSchemas) => {\n        const data = {};\n        const perms = {};\n        const indexes = {};\n        allSchemas.forEach(schema => {\n          data[schema.className] = injectDefaultSchema(schema).fields;\n          perms[schema.className] = schema.classLevelPermissions;\n          indexes[schema.className] = schema.indexes;\n        });\n\n        // Inject the in-memory classes\n        volatileClasses.forEach(className => {\n          const schema = injectDefaultSchema({ className, fields: {}, classLevelPermissions: {} });\n          data[className] = schema.fields;\n          perms[className] = schema.classLevelPermissions;\n          indexes[className] = schema.indexes;\n        });\n        this.data = data;\n        this.perms = perms;\n        this.indexes = indexes;\n        delete this.reloadDataPromise;\n      }, (err) => {\n        this.data = {};\n        this.perms = {};\n        this.indexes = {};\n        delete this.reloadDataPromise;\n        throw err;\n      })\n    }).then(() => {});\n    return this.reloadDataPromise;\n  }\n\n  getAllClasses(options: LoadSchemaOptions = {clearCache: false}): Promise<Array<Schema>> {\n    let promise = Promise.resolve();\n    if (options.clearCache) {\n      promise = this._cache.clear();\n    }\n    return promise.then(() => {\n      return this._cache.getAllClasses()\n    }).then((allClasses) => {\n      if (allClasses && allClasses.length && !options.clearCache) {\n        return Promise.resolve(allClasses);\n      }\n      return this._dbAdapter.getAllClasses()\n        .then(allSchemas => allSchemas.map(injectDefaultSchema))\n        .then(allSchemas => {\n          return this._cache.setAllClasses(allSchemas).then(() => {\n            return allSchemas;\n          });\n        })\n    });\n  }\n\n  getOneSchema(className: string, allowVolatileClasses: boolean = false, options: LoadSchemaOptions = {clearCache: false}): Promise<Schema> {\n    let promise = Promise.resolve();\n    if (options.clearCache) {\n      promise = this._cache.clear();\n    }\n    return promise.then(() => {\n      if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {\n        return Promise.resolve({\n          className,\n          fields: this.data[className],\n          classLevelPermissions: this.perms[className],\n          indexes: this.indexes[className]\n        });\n      }\n      return this._cache.getOneSchema(className).then((cached) => {\n        if (cached && !options.clearCache) {\n          return Promise.resolve(cached);\n        }\n        return this._dbAdapter.getClass(className)\n          .then(injectDefaultSchema)\n          .then((result) => {\n            return this._cache.setOneSchema(className, result).then(() => {\n              return result;\n            })\n          });\n      });\n    });\n  }\n\n  // Create a new class that includes the three default fields.\n  // ACL is an implicit column that does not get an entry in the\n  // _SCHEMAS database. Returns a promise that resolves with the\n  // created schema, in mongo format.\n  // on success, and rejects with an error on fail. Ensure you\n  // have authorization (master key, or client class creation\n  // enabled) before calling this function.\n  addClassIfNotExists(className: string, fields: SchemaFields = {}, classLevelPermissions: any, indexes: any = {}): Promise<void> {\n    var validationError = this.validateNewClass(className, fields, classLevelPermissions);\n    if (validationError) {\n      return Promise.reject(validationError);\n    }\n\n    return this._dbAdapter.createClass(className, convertSchemaToAdapterSchema({ fields, classLevelPermissions, indexes, className }))\n      .then(convertAdapterSchemaToParseSchema)\n      .then((res) => {\n        return this._cache.clear().then(() => {\n          return Promise.resolve(res);\n        });\n      })\n      .catch(error => {\n        if (error && error.code === Parse.Error.DUPLICATE_VALUE) {\n          throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n        } else {\n          throw error;\n        }\n      });\n  }\n\n  updateClass(className: string, submittedFields: SchemaFields, classLevelPermissions: any, indexes: any, database: DatabaseController) {\n    return this.getOneSchema(className)\n      .then(schema => {\n        const existingFields = schema.fields;\n        Object.keys(submittedFields).forEach(name => {\n          const field = submittedFields[name];\n          if (existingFields[name] && field.__op !== 'Delete') {\n            throw new Parse.Error(255, `Field ${name} exists, cannot update.`);\n          }\n          if (!existingFields[name] && field.__op === 'Delete') {\n            throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);\n          }\n        });\n\n        delete existingFields._rperm;\n        delete existingFields._wperm;\n        const newSchema = buildMergedSchemaObject(existingFields, submittedFields);\n        const defaultFields = defaultColumns[className] || defaultColumns._Default;\n        const fullNewSchema = Object.assign({}, newSchema, defaultFields);\n        const validationError = this.validateSchemaData(className, newSchema, classLevelPermissions, Object.keys(existingFields));\n        if (validationError) {\n          throw new Parse.Error(validationError.code, validationError.error);\n        }\n\n        // Finally we have checked to make sure the request is valid and we can start deleting fields.\n        // Do all deletions first, then a single save to _SCHEMA collection to handle all additions.\n        const deletedFields: string[] = [];\n        const insertedFields = [];\n        Object.keys(submittedFields).forEach(fieldName => {\n          if (submittedFields[fieldName].__op === 'Delete') {\n            deletedFields.push(fieldName);\n          } else {\n            insertedFields.push(fieldName);\n          }\n        });\n\n        let deletePromise = Promise.resolve();\n        if (deletedFields.length > 0) {\n          deletePromise = this.deleteFields(deletedFields, className, database);\n        }\n        return deletePromise // Delete Everything\n          .then(() => this.reloadData({ clearCache: true })) // Reload our Schema, so we have all the new values\n          .then(() => {\n            const promises = insertedFields.map(fieldName => {\n              const type = submittedFields[fieldName];\n              return this.enforceFieldExists(className, fieldName, type);\n            });\n            return Promise.all(promises);\n          })\n          .then(() => this.setPermissions(className, classLevelPermissions, newSchema))\n          .then(() => this._dbAdapter.setIndexesWithSchemaFormat(className, indexes, schema.indexes, fullNewSchema))\n          .then(() => this.reloadData({ clearCache: true }))\n        //TODO: Move this logic into the database adapter\n          .then(() => {\n            const reloadedSchema: Schema = {\n              className: className,\n              fields: this.data[className],\n              classLevelPermissions: this.perms[className],\n            };\n            if (this.indexes[className] && Object.keys(this.indexes[className]).length !== 0) {\n              reloadedSchema.indexes = this.indexes[className];\n            }\n            return reloadedSchema;\n          });\n      })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} does not exist.`);\n        } else {\n          throw error;\n        }\n      })\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object or fails with a reason.\n  enforceClassExists(className: string): Promise<SchemaController> {\n    if (this.data[className]) {\n      return Promise.resolve(this);\n    }\n    // We don't have this class. Update the schema\n    return this.addClassIfNotExists(className)\n    // The schema update succeeded. Reload the schema\n      .then(() => this.reloadData({ clearCache: true }))\n      .catch(() => {\n      // The schema update failed. This can be okay - it might\n      // have failed because there's a race condition and a different\n      // client is making the exact same schema update that we want.\n      // So just reload the schema.\n        return this.reloadData({ clearCache: true });\n      })\n      .then(() => {\n      // Ensure that the schema now validates\n        if (this.data[className]) {\n          return this;\n        } else {\n          throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);\n        }\n      })\n      .catch(() => {\n      // The schema still doesn't validate. Give up\n        throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');\n      });\n  }\n\n  validateNewClass(className: string, fields: SchemaFields = {}, classLevelPermissions: any): any {\n    if (this.data[className]) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n    }\n    if (!classNameIsValid(className)) {\n      return {\n        code: Parse.Error.INVALID_CLASS_NAME,\n        error: invalidClassNameMessage(className),\n      };\n    }\n    return this.validateSchemaData(className, fields, classLevelPermissions, []);\n  }\n\n  validateSchemaData(className: string, fields: SchemaFields, classLevelPermissions: ClassLevelPermissions, existingFieldNames: Array<string>) {\n    for (const fieldName in fields) {\n      if (existingFieldNames.indexOf(fieldName) < 0) {\n        if (!fieldNameIsValid(fieldName)) {\n          return {\n            code: Parse.Error.INVALID_KEY_NAME,\n            error: 'invalid field name: ' + fieldName,\n          };\n        }\n        if (!fieldNameIsValidForClass(fieldName, className)) {\n          return {\n            code: 136,\n            error: 'field ' + fieldName + ' cannot be added',\n          };\n        }\n        const error = fieldTypeIsInvalid(fields[fieldName]);\n        if (error) return { code: error.code, error: error.message };\n      }\n    }\n\n    for (const fieldName in defaultColumns[className]) {\n      fields[fieldName] = defaultColumns[className][fieldName];\n    }\n\n    const geoPoints = Object.keys(fields).filter(key => fields[key] && fields[key].type === 'GeoPoint');\n    if (geoPoints.length > 1) {\n      return {\n        code: Parse.Error.INCORRECT_TYPE,\n        error: 'currently, only one GeoPoint field may exist in an object. Adding ' + geoPoints[1] + ' when ' + geoPoints[0] + ' already exists.',\n      };\n    }\n    validateCLP(classLevelPermissions, fields);\n  }\n\n  // Sets the Class-level permissions for a given className, which must exist.\n  setPermissions(className: string, perms: any, newSchema: SchemaFields) {\n    if (typeof perms === 'undefined') {\n      return Promise.resolve();\n    }\n    validateCLP(perms, newSchema);\n    return this._dbAdapter.setClassLevelPermissions(className, perms);\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object if the provided className-fieldName-type tuple is valid.\n  // The className must already be validated.\n  // If 'freeze' is true, refuse to update the schema for this field.\n  enforceFieldExists(className: string, fieldName: string, type: string | SchemaField) {\n    if (fieldName.indexOf(\".\") > 0) {\n      // subdocument key (x.y) => ok if x is of type 'object'\n      fieldName = fieldName.split(\".\")[ 0 ];\n      type = 'Object';\n    }\n    if (!fieldNameIsValid(fieldName)) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);\n    }\n\n    // If someone tries to create a new field with null/undefined as the value, return;\n    if (!type) {\n      return Promise.resolve(this);\n    }\n\n    return this.reloadData().then(() => {\n      const expectedType = this.getExpectedType(className, fieldName);\n      if (typeof type === 'string') {\n        type = { type };\n      }\n\n      if (expectedType) {\n        if (!dbTypeMatchesObjectType(expectedType, type)) {\n          throw new Parse.Error(\n            Parse.Error.INCORRECT_TYPE,\n            `schema mismatch for ${className}.${fieldName}; expected ${typeToString(expectedType)} but got ${typeToString(type)}`\n          );\n        }\n        return this;\n      }\n\n      return this._dbAdapter.addFieldIfNotExists(className, fieldName, type).then(() => {\n        // The update succeeded. Reload the schema\n        return this.reloadData({ clearCache: true });\n      }, (error) => {\n        if (error.code == Parse.Error.INCORRECT_TYPE) {\n          // Make sure that we throw errors when it is appropriate to do so.\n          throw error;\n        }\n        // The update failed. This can be okay - it might have been a race\n        // condition where another client updated the schema in the same\n        // way that we wanted to. So, just reload the schema\n        return this.reloadData({ clearCache: true });\n      }).then(() => {\n        // Ensure that the schema now validates\n        const expectedType = this.getExpectedType(className, fieldName);\n        if (typeof type === 'string') {\n          type = { type };\n        }\n        if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {\n          throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);\n        }\n        // Remove the cached schema\n        this._cache.clear();\n        return this;\n      });\n    });\n  }\n\n  // maintain compatibility\n  deleteField(fieldName: string, className: string, database: DatabaseController) {\n    return this.deleteFields([fieldName], className, database);\n  }\n\n  // Delete fields, and remove that data from all objects. This is intended\n  // to remove unused fields, if other writers are writing objects that include\n  // this field, the field may reappear. Returns a Promise that resolves with\n  // no object on success, or rejects with { code, error } on failure.\n  // Passing the database and prefix is necessary in order to drop relation collections\n  // and remove fields from objects. Ideally the database would belong to\n  // a database adapter and this function would close over it or access it via member.\n  deleteFields(fieldNames: Array<string>, className: string, database: DatabaseController) {\n    if (!classNameIsValid(className)) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));\n    }\n\n    fieldNames.forEach(fieldName => {\n      if (!fieldNameIsValid(fieldName)) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);\n      }\n      //Don't allow deleting the default fields.\n      if (!fieldNameIsValidForClass(fieldName, className)) {\n        throw new Parse.Error(136, `field ${fieldName} cannot be changed`);\n      }\n    });\n\n    return this.getOneSchema(className, false, {clearCache: true})\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} does not exist.`);\n        } else {\n          throw error;\n        }\n      })\n      .then(schema => {\n        fieldNames.forEach(fieldName => {\n          if (!schema.fields[fieldName]) {\n            throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);\n          }\n        });\n\n        const schemaFields = { ...schema.fields };\n        return database.adapter.deleteFields(className, schema, fieldNames)\n          .then(() => {\n            return Promise.all(fieldNames.map(fieldName => {\n              const field = schemaFields[fieldName];\n              if (field && field.type === 'Relation') {\n              //For relations, drop the _Join table\n                return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);\n              }\n              return Promise.resolve();\n            }));\n          });\n      }).then(() => {\n        this._cache.clear();\n      });\n  }\n\n  // Validates an object provided in REST format.\n  // Returns a promise that resolves to the new schema if this object is\n  // valid.\n  validateObject(className: string, object: any, query: any) {\n    let geocount = 0;\n    let promise = this.enforceClassExists(className);\n    for (const fieldName in object) {\n      if (object[fieldName] === undefined) {\n        continue;\n      }\n      const expected = getType(object[fieldName]);\n      if (expected === 'GeoPoint') {\n        geocount++;\n      }\n      if (geocount > 1) {\n        // Make sure all field validation operations run before we return.\n        // If not - we are continuing to run logic, but already provided response from the server.\n        return promise.then(() => {\n          return Promise.reject(new Parse.Error(Parse.Error.INCORRECT_TYPE,\n            'there can only be one geopoint field in a class'));\n        });\n      }\n      if (!expected) {\n        continue;\n      }\n      if (fieldName === 'ACL') {\n        // Every object has ACL implicitly.\n        continue;\n      }\n\n      promise = promise.then(schema => schema.enforceFieldExists(className, fieldName, expected));\n    }\n    promise = thenValidateRequiredColumns(promise, className, object, query);\n    return promise;\n  }\n\n  // Validates that all the properties are set for the object\n  validateRequiredColumns(className: string, object: any, query: any) {\n    const columns = requiredColumns[className];\n    if (!columns || columns.length == 0) {\n      return Promise.resolve(this);\n    }\n\n    const missingColumns = columns.filter(function(column){\n      if (query && query.objectId) {\n        if (object[column] && typeof object[column] === \"object\") {\n          // Trying to delete a required column\n          return object[column].__op == 'Delete';\n        }\n        // Not trying to do anything there\n        return false;\n      }\n      return !object[column]\n    });\n\n    if (missingColumns.length > 0) {\n      throw new Parse.Error(\n        Parse.Error.INCORRECT_TYPE,\n        missingColumns[0] + ' is required.');\n    }\n    return Promise.resolve(this);\n  }\n\n  // Validates the base CLP for an operation\n  testBaseCLP(className: string, aclGroup: string[], operation: string) {\n    if (!this.perms[className] || !this.perms[className][operation]) {\n      return true;\n    }\n    const classPerms = this.perms[className];\n    const perms = classPerms[operation];\n    // Handle the public scenario quickly\n    if (perms['*']) {\n      return true;\n    }\n    // Check permissions against the aclGroup provided (array of userId/roles)\n    if (aclGroup.some(acl => { return perms[acl] === true })) {\n      return true;\n    }\n    return false;\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  validatePermission(className: string, aclGroup: string[], operation: string) {\n\n    if (this.testBaseCLP(className, aclGroup, operation)) {\n      return Promise.resolve();\n    }\n\n    if (!this.perms[className] || !this.perms[className][operation]) {\n      return true;\n    }\n    const classPerms = this.perms[className];\n    const perms = classPerms[operation];\n\n    // If only for authenticated users\n    // make sure we have an aclGroup\n    if (perms['requiresAuthentication']) {\n      // If aclGroup has * (public)\n      if (!aclGroup || aclGroup.length == 0) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.');\n      } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.');\n      }\n      // requiresAuthentication passed, just move forward\n      // probably would be wise at some point to rename to 'authenticatedUser'\n      return Promise.resolve();\n    }\n\n    // No matching CLP, let's check the Pointer permissions\n    // And handle those later\n    const permissionField = ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    // Reject create when write lockdown\n    if (permissionField == 'writeUserFields' && operation == 'create') {\n      throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN,\n        `Permission denied for action ${operation} on class ${className}.`);\n    }\n\n    // Process the readUserFields later\n    if (Array.isArray(classPerms[permissionField]) && classPerms[permissionField].length > 0) {\n      return Promise.resolve();\n    }\n    throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN,\n      `Permission denied for action ${operation} on class ${className}.`);\n  }\n\n  // Returns the expected type for a className+key combination\n  // or undefined if the schema is not set\n  getExpectedType(className: string, fieldName: string): ?(SchemaField | string) {\n    if (this.data && this.data[className]) {\n      const expectedType = this.data[className][fieldName]\n      return expectedType === 'map' ? 'Object' : expectedType;\n    }\n    return undefined;\n  }\n\n  // Checks if a given class is in the schema.\n  hasClass(className: string) {\n    return this.reloadData().then(() => !!(this.data[className]));\n  }\n}\n\n// Returns a promise for a new Schema.\nconst load = (dbAdapter: StorageAdapter, schemaCache: any, options: any): Promise<SchemaController> => {\n  const schema = new SchemaController(dbAdapter, schemaCache);\n  return schema.reloadData(options).then(() => schema);\n}\n\n// Builds a new schema (in schema API response format) out of an\n// existing mongo schema + a schemas API put request. This response\n// does not include the default fields, as it is intended to be passed\n// to mongoSchemaFromFieldsAndClassName. No validation is done here, it\n// is done in mongoSchemaFromFieldsAndClassName.\nfunction buildMergedSchemaObject(existingFields: SchemaFields, putRequest: any): SchemaFields {\n  const newSchema = {};\n  // @flow-disable-next\n  const sysSchemaField = Object.keys(defaultColumns).indexOf(existingFields._id) === -1 ? [] : Object.keys(defaultColumns[existingFields._id]);\n  for (const oldField in existingFields) {\n    if (oldField !== '_id' && oldField !== 'ACL' &&  oldField !== 'updatedAt' && oldField !== 'createdAt' && oldField !== 'objectId') {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {\n        continue;\n      }\n      const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete'\n      if (!fieldIsDeleted) {\n        newSchema[oldField] = existingFields[oldField];\n      }\n    }\n  }\n  for (const newField in putRequest) {\n    if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {\n        continue;\n      }\n      newSchema[newField] = putRequest[newField];\n    }\n  }\n  return newSchema;\n}\n\n// Given a schema promise, construct another schema promise that\n// validates this field once the schema loads.\nfunction thenValidateRequiredColumns(schemaPromise, className, object, query) {\n  return schemaPromise.then((schema) => {\n    return schema.validateRequiredColumns(className, object, query);\n  });\n}\n\n// Gets the type from a REST API formatted object, where 'type' is\n// extended past javascript types to include the rest of the Parse\n// type system.\n// The output should be a valid schema value.\n// TODO: ensure that this is compatible with the format used in Open DB\nfunction getType(obj: any): ?(SchemaField | string) {\n  const type = typeof obj;\n  switch(type) {\n  case 'boolean':\n    return 'Boolean';\n  case 'string':\n    return 'String';\n  case 'number':\n    return 'Number';\n  case 'map':\n  case 'object':\n    if (!obj) {\n      return undefined;\n    }\n    return getObjectType(obj);\n  case 'function':\n  case 'symbol':\n  case 'undefined':\n  default:\n    throw 'bad obj: ' + obj;\n  }\n}\n\n// This gets the type for non-JSON types like pointers and files, but\n// also gets the appropriate type for $ operators.\n// Returns null if the type is unknown.\nfunction getObjectType(obj): ?(SchemaField | string) {\n  if (obj instanceof Array) {\n    return 'Array';\n  }\n  if (obj.__type){\n    switch(obj.__type) {\n    case 'Pointer' :\n      if(obj.className) {\n        return {\n          type: 'Pointer',\n          targetClass: obj.className\n        }\n      }\n      break;\n    case 'Relation' :\n      if(obj.className) {\n        return {\n          type: 'Relation',\n          targetClass: obj.className\n        }\n      }\n      break;\n    case 'File' :\n      if(obj.name) {\n        return 'File';\n      }\n      break;\n    case 'Date' :\n      if(obj.iso) {\n        return 'Date';\n      }\n      break;\n    case 'GeoPoint' :\n      if(obj.latitude != null && obj.longitude != null) {\n        return 'GeoPoint';\n      }\n      break;\n    case 'Bytes' :\n      if(obj.base64) {\n        return 'Bytes';\n      }\n      break;\n    case 'Polygon' :\n      if(obj.coordinates) {\n        return 'Polygon';\n      }\n      break;\n    }\n    throw new Parse.Error(Parse.Error.INCORRECT_TYPE, \"This is not a valid \" + obj.__type);\n  }\n  if (obj['$ne']) {\n    return getObjectType(obj['$ne']);\n  }\n  if (obj.__op) {\n    switch(obj.__op) {\n    case 'Increment':\n      return 'Number';\n    case 'Delete':\n      return null;\n    case 'Add':\n    case 'AddUnique':\n    case 'Remove':\n      return 'Array';\n    case 'AddRelation':\n    case 'RemoveRelation':\n      return {\n        type: 'Relation',\n        targetClass: obj.objects[0].className\n      }\n    case 'Batch':\n      return getObjectType(obj.ops[0]);\n    default:\n      throw 'unexpected op: ' + obj.__op;\n    }\n  }\n  return 'Object';\n}\n\nexport {\n  load,\n  classNameIsValid,\n  fieldNameIsValid,\n  invalidClassNameMessage,\n  buildMergedSchemaObject,\n  systemClasses,\n  defaultColumns,\n  convertSchemaToAdapterSchema,\n  VolatileClassesSchemas,\n  SchemaController,\n};\n"]}
1567
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_StorageAdapter","require","_SchemaCache","_interopRequireDefault","_DatabaseController","_Config","_Error","_deepcopy","e","__esModule","default","Parse","defaultColumns","exports","Object","freeze","_Default","objectId","type","createdAt","updatedAt","ACL","_User","username","password","email","emailVerified","authData","_Installation","installationId","deviceToken","channels","deviceType","pushType","GCMSenderId","timeZone","localeIdentifier","badge","appVersion","appName","appIdentifier","parseVersion","_Role","name","users","targetClass","roles","_Session","user","sessionToken","expiresAt","createdWith","_Product","productIdentifier","download","downloadName","icon","order","title","subtitle","_PushStatus","pushTime","source","query","payload","expiry","expiration_interval","status","numSent","numFailed","pushHash","errorMessage","sentPerType","failedPerType","sentPerUTCOffset","failedPerUTCOffset","count","_JobStatus","jobName","message","params","finishedAt","_JobSchedule","description","startAfter","daysOfWeek","timeOfDay","lastRun","repeatMinutes","_Hooks","functionName","className","triggerName","url","_GlobalConfig","masterKeyOnly","_GraphQLConfig","config","_Audience","lastUsed","timesUsed","_Idempotency","reqId","expire","requiredColumns","read","write","invalidColumns","systemClasses","volatileClasses","roleRegex","protectedFieldsPointerRegex","publicRegex","authenticatedRegex","requiresAuthenticationRegex","clpPointerRegex","protectedFieldsRegex","clpFieldsRegex","validatePermissionKey","key","userIdRegExp","matchesSome","regEx","match","valid","Error","INVALID_JSON","validateProtectedFieldsKey","CLPValidKeys","validateCLP","perms","fields","operationKey","indexOf","operation","validateCLPjson","fieldName","validatePointerPermission","entity","protectedFields","Array","isArray","field","prototype","hasOwnProperty","call","pointerFields","pointerField","permit","toString","invalidKeys","keys","filter","includes","invalidValues","values","length","join","joinClassRegex","classAndFieldRegex","classNameIsValid","test","fieldNameIsValid","fieldNameIsValidForClass","invalidClassNameMessage","invalidJsonError","validNonRelationOrPointerTypes","fieldTypeIsInvalid","INVALID_CLASS_NAME","undefined","INCORRECT_TYPE","convertSchemaToAdapterSchema","schema","injectDefaultSchema","_rperm","_wperm","_hashed_password","convertAdapterSchemaToParseSchema","indexes","SchemaData","constructor","allSchemas","__data","__protectedFields","forEach","defineProperty","get","data","classLevelPermissions","deepcopy","classProtectedFields","unq","Set","from","defaultSchema","_HooksSchema","_GlobalConfigSchema","_GraphQLConfigSchema","_PushStatusSchema","_JobStatusSchema","_JobScheduleSchema","_AudienceSchema","_IdempotencySchema","VolatileClassesSchemas","dbTypeMatchesObjectType","dbType","objectType","typeToString","ttl","date","Date","now","duration","SchemaController","databaseAdapter","_dbAdapter","Config","applicationId","schemaData","SchemaCache","all","customIds","allowCustomObjectId","customIdRegEx","autoIdRegEx","userIdRegEx","watch","reloadData","clearCache","reloadDataIfNeeded","enableSchemaHooks","options","reloadDataPromise","getAllClasses","then","err","setAllClasses","cached","Promise","resolve","map","put","getOneSchema","allowVolatileClasses","clear","oneSchema","find","reject","addClassIfNotExists","validationError","validateNewClass","code","error","adapterSchema","createClass","parseSchema","DUPLICATE_VALUE","updateClass","submittedFields","database","existingFields","__op","newSchema","buildMergedSchemaObject","defaultFields","fullNewSchema","assign","validateSchemaData","deletedFields","insertedFields","push","deletePromise","deleteFields","enforceFields","promises","enforceFieldExists","results","result","setPermissions","setIndexesWithSchemaFormat","ensureFields","reloadedSchema","catch","enforceClassExists","existingFieldNames","INVALID_KEY_NAME","fieldType","defaultValue","defaultValueType","getType","required","geoPoints","setClassLevelPermissions","isValidation","maintenance","x","y","split","isArrayIndex","every","c","fieldNameToValidate","charAt","substring","expectedType","getExpectedType","JSON","stringify","updateFieldOptions","addFieldIfNotExists","i","deleteField","fieldNames","schemaFields","adapter","deleteClass","validateObject","object","geocount","expected","promise","thenValidateRequiredColumns","validateRequiredColumns","columns","missingColumns","column","testPermissionsForClassName","aclGroup","testPermissions","getClassLevelPermissions","classPermissions","some","acl","validatePermission","action","createSanitizedError","OBJECT_NOT_FOUND","permissionField","OPERATION_FORBIDDEN","hasClass","load","dbAdapter","schemaCacheTtl","putRequest","sysSchemaField","_id","oldField","fieldIsDeleted","newField","schemaPromise","obj","getObjectType","__type","iso","latitude","longitude","base64","coordinates","objects","ops"],"sources":["../../src/Controllers/SchemaController.js"],"sourcesContent":["// @flow\n// This class handles schema validation, persistence, and modification.\n//\n// Each individual Schema object should be immutable. The helpers to\n// do things with the Schema just return a new schema when the schema\n// is changed.\n//\n// The canonical place to store this Schema is in the database itself,\n// in a _SCHEMA collection. This is not the right way to do it for an\n// open source framework, but it's backward compatible, so we're\n// keeping it this way for now.\n//\n// In API-handling code, you should only use the Schema class via the\n// DatabaseController. This will let us replace the schema logic for\n// different databases.\n// TODO: hide all schema logic inside the database adapter.\n// @flow-disable-next\nconst Parse = require('parse/node').Parse;\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport SchemaCache from '../Adapters/Cache/SchemaCache';\nimport DatabaseController from './DatabaseController';\nimport Config from '../Config';\nimport { createSanitizedError } from '../Error';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport type {\n  Schema,\n  SchemaFields,\n  ClassLevelPermissions,\n  SchemaField,\n  LoadSchemaOptions,\n} from './types';\n\nconst defaultColumns: { [string]: SchemaFields } = Object.freeze({\n  // Contain the default columns for every parse object type (except _Join collection)\n  _Default: {\n    objectId: { type: 'String' },\n    createdAt: { type: 'Date' },\n    updatedAt: { type: 'Date' },\n    ACL: { type: 'ACL' },\n  },\n  // The additional default columns for the _User collection (in addition to DefaultCols)\n  _User: {\n    username: { type: 'String' },\n    password: { type: 'String' },\n    email: { type: 'String' },\n    emailVerified: { type: 'Boolean' },\n    authData: { type: 'Object' },\n  },\n  // The additional default columns for the _Installation collection (in addition to DefaultCols)\n  _Installation: {\n    installationId: { type: 'String' },\n    deviceToken: { type: 'String' },\n    channels: { type: 'Array' },\n    deviceType: { type: 'String' },\n    pushType: { type: 'String' },\n    GCMSenderId: { type: 'String' },\n    timeZone: { type: 'String' },\n    localeIdentifier: { type: 'String' },\n    badge: { type: 'Number' },\n    appVersion: { type: 'String' },\n    appName: { type: 'String' },\n    appIdentifier: { type: 'String' },\n    parseVersion: { type: 'String' },\n  },\n  // The additional default columns for the _Role collection (in addition to DefaultCols)\n  _Role: {\n    name: { type: 'String' },\n    users: { type: 'Relation', targetClass: '_User' },\n    roles: { type: 'Relation', targetClass: '_Role' },\n  },\n  // The additional default columns for the _Session collection (in addition to DefaultCols)\n  _Session: {\n    user: { type: 'Pointer', targetClass: '_User' },\n    installationId: { type: 'String' },\n    sessionToken: { type: 'String' },\n    expiresAt: { type: 'Date' },\n    createdWith: { type: 'Object' },\n  },\n  _Product: {\n    productIdentifier: { type: 'String' },\n    download: { type: 'File' },\n    downloadName: { type: 'String' },\n    icon: { type: 'File' },\n    order: { type: 'Number' },\n    title: { type: 'String' },\n    subtitle: { type: 'String' },\n  },\n  _PushStatus: {\n    pushTime: { type: 'String' },\n    source: { type: 'String' }, // rest or webui\n    query: { type: 'String' }, // the stringified JSON query\n    payload: { type: 'String' }, // the stringified JSON payload,\n    title: { type: 'String' },\n    expiry: { type: 'Number' },\n    expiration_interval: { type: 'Number' },\n    status: { type: 'String' },\n    numSent: { type: 'Number' },\n    numFailed: { type: 'Number' },\n    pushHash: { type: 'String' },\n    errorMessage: { type: 'Object' },\n    sentPerType: { type: 'Object' },\n    failedPerType: { type: 'Object' },\n    sentPerUTCOffset: { type: 'Object' },\n    failedPerUTCOffset: { type: 'Object' },\n    count: { type: 'Number' }, // tracks # of batches queued and pending\n  },\n  _JobStatus: {\n    jobName: { type: 'String' },\n    source: { type: 'String' },\n    status: { type: 'String' },\n    message: { type: 'String' },\n    params: { type: 'Object' }, // params received when calling the job\n    finishedAt: { type: 'Date' },\n  },\n  _JobSchedule: {\n    jobName: { type: 'String' },\n    description: { type: 'String' },\n    params: { type: 'String' },\n    startAfter: { type: 'String' },\n    daysOfWeek: { type: 'Array' },\n    timeOfDay: { type: 'String' },\n    lastRun: { type: 'Number' },\n    repeatMinutes: { type: 'Number' },\n  },\n  _Hooks: {\n    functionName: { type: 'String' },\n    className: { type: 'String' },\n    triggerName: { type: 'String' },\n    url: { type: 'String' },\n  },\n  _GlobalConfig: {\n    objectId: { type: 'String' },\n    params: { type: 'Object' },\n    masterKeyOnly: { type: 'Object' },\n  },\n  _GraphQLConfig: {\n    objectId: { type: 'String' },\n    config: { type: 'Object' },\n  },\n  _Audience: {\n    objectId: { type: 'String' },\n    name: { type: 'String' },\n    query: { type: 'String' }, //storing query as JSON string to prevent \"Nested keys should not contain the '$' or '.' characters\" error\n    lastUsed: { type: 'Date' },\n    timesUsed: { type: 'Number' },\n  },\n  _Idempotency: {\n    reqId: { type: 'String' },\n    expire: { type: 'Date' },\n  },\n});\n\n// fields required for read or write operations on their respective classes.\nconst requiredColumns = Object.freeze({\n  read: {\n    _User: ['username'],\n  },\n  write: {\n    _Product: ['productIdentifier', 'icon', 'order', 'title', 'subtitle'],\n    _Role: ['name', 'ACL'],\n  },\n});\n\nconst invalidColumns = ['length'];\n\nconst systemClasses = Object.freeze([\n  '_User',\n  '_Installation',\n  '_Role',\n  '_Session',\n  '_Product',\n  '_PushStatus',\n  '_JobStatus',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\nconst volatileClasses = Object.freeze([\n  '_JobStatus',\n  '_PushStatus',\n  '_Hooks',\n  '_GlobalConfig',\n  '_GraphQLConfig',\n  '_JobSchedule',\n  '_Audience',\n  '_Idempotency',\n]);\n\n// Anything that start with role\nconst roleRegex = /^role:.*/;\n// Anything that starts with userField (allowed for protected fields only)\nconst protectedFieldsPointerRegex = /^userField:.*/;\n// * permission\nconst publicRegex = /^\\*$/;\n\nconst authenticatedRegex = /^authenticated$/;\n\nconst requiresAuthenticationRegex = /^requiresAuthentication$/;\n\nconst clpPointerRegex = /^pointerFields$/;\n\n// regex for validating entities in protectedFields object\nconst protectedFieldsRegex = Object.freeze([\n  protectedFieldsPointerRegex,\n  publicRegex,\n  authenticatedRegex,\n  roleRegex,\n]);\n\n// clp regex\nconst clpFieldsRegex = Object.freeze([\n  clpPointerRegex,\n  publicRegex,\n  requiresAuthenticationRegex,\n  roleRegex,\n]);\n\nfunction validatePermissionKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of clpFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId depends on startup options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nfunction validateProtectedFieldsKey(key, userIdRegExp) {\n  let matchesSome = false;\n  for (const regEx of protectedFieldsRegex) {\n    if (key.match(regEx) !== null) {\n      matchesSome = true;\n      break;\n    }\n  }\n\n  // userId regex depends on launch options so it's dynamic\n  const valid = matchesSome || key.match(userIdRegExp) !== null;\n  if (!valid) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${key}' is not a valid key for class level permissions`\n    );\n  }\n}\n\nconst CLPValidKeys = Object.freeze([\n  'ACL',\n  'find',\n  'count',\n  'get',\n  'create',\n  'update',\n  'delete',\n  'addField',\n  'readUserFields',\n  'writeUserFields',\n  'protectedFields',\n]);\n\n// validation before setting class-level permissions on collection\nfunction validateCLP(perms: ClassLevelPermissions, fields: SchemaFields, userIdRegExp: RegExp) {\n  if (!perms) {\n    return;\n  }\n  for (const operationKey in perms) {\n    if (CLPValidKeys.indexOf(operationKey) == -1) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `${operationKey} is not a valid operation for class level permissions`\n      );\n    }\n\n    const operation = perms[operationKey];\n    // proceed with next operationKey\n\n    // throws when root fields are of wrong type\n    validateCLPjson(operation, operationKey);\n\n    if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n      // validate grouped pointer permissions\n      // must be an array with field names\n      for (const fieldName of operation) {\n        validatePointerPermission(fieldName, fields, operationKey);\n      }\n      // readUserFields and writerUserFields do not have nesdted fields\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate protected fields\n    if (operationKey === 'protectedFields') {\n      for (const entity in operation) {\n        // throws on unexpected key\n        validateProtectedFieldsKey(entity, userIdRegExp);\n\n        const protectedFields = operation[entity];\n\n        if (!Array.isArray(protectedFields)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${protectedFields}' is not a valid value for protectedFields[${entity}] - expected an array.`\n          );\n        }\n\n        // if the field is in form of array\n        for (const field of protectedFields) {\n          // do not alloow to protect default fields\n          if (defaultColumns._Default[field]) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Default field '${field}' can not be protected`\n            );\n          }\n          // field should exist on collection\n          if (!Object.prototype.hasOwnProperty.call(fields, field)) {\n            throw new Parse.Error(\n              Parse.Error.INVALID_JSON,\n              `Field '${field}' in protectedFields:${entity} does not exist`\n            );\n          }\n        }\n      }\n      // proceed with next operationKey\n      continue;\n    }\n\n    // validate other fields\n    // Entity can be:\n    // \"*\" - Public,\n    // \"requiresAuthentication\" - authenticated users,\n    // \"objectId\" - _User id,\n    // \"role:rolename\",\n    // \"pointerFields\" - array of field names containing pointers to users\n    for (const entity in operation) {\n      // throws on unexpected key\n      validatePermissionKey(entity, userIdRegExp);\n\n      // entity can be either:\n      // \"pointerFields\": string[]\n      if (entity === 'pointerFields') {\n        const pointerFields = operation[entity];\n\n        if (Array.isArray(pointerFields)) {\n          for (const pointerField of pointerFields) {\n            validatePointerPermission(pointerField, fields, operation);\n          }\n        } else {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${pointerFields}' is not a valid value for ${operationKey}[${entity}] - expected an array.`\n          );\n        }\n        // proceed with next entity key\n        continue;\n      }\n\n      const permit = operation[entity];\n\n      if (operationKey === 'ACL') {\n        if (Object.prototype.toString.call(permit) !== '[object Object]') {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${permit}' is not a valid value for class level permissions acl`\n          );\n        }\n        const invalidKeys = Object.keys(permit).filter(key => !['read', 'write'].includes(key));\n        const invalidValues = Object.values(permit).filter(key => typeof key !== 'boolean');\n        if (invalidKeys.length) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${invalidKeys.join(',')}' is not a valid key for class level permissions acl`\n          );\n        }\n\n        if (invalidValues.length) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_JSON,\n            `'${invalidValues.join(',')}' is not a valid value for class level permissions acl`\n          );\n        }\n      } else if (permit !== true) {\n        throw new Parse.Error(\n          Parse.Error.INVALID_JSON,\n          `'${permit}' is not a valid value for class level permissions acl ${operationKey}:${entity}`\n        );\n      }\n    }\n  }\n}\n\nfunction validateCLPjson(operation: any, operationKey: string) {\n  if (operationKey === 'readUserFields' || operationKey === 'writeUserFields') {\n    if (!Array.isArray(operation)) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an array`\n      );\n    }\n  } else {\n    if (typeof operation === 'object' && operation !== null) {\n      // ok to proceed\n      return;\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_JSON,\n        `'${operation}' is not a valid value for class level permissions ${operationKey} - must be an object`\n      );\n    }\n  }\n}\n\nfunction validatePointerPermission(fieldName: string, fields: Object, operation: string) {\n  // Uses collection schema to ensure the field is of type:\n  // - Pointer<_User> (pointers)\n  // - Array\n  //\n  //    It's not possible to enforce type on Array's items in schema\n  //  so we accept any Array field, and later when applying permissions\n  //  only items that are pointers to _User are considered.\n  if (\n    !(\n      fields[fieldName] &&\n      ((fields[fieldName].type == 'Pointer' && fields[fieldName].targetClass == '_User') ||\n        fields[fieldName].type == 'Array')\n    )\n  ) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `'${fieldName}' is not a valid column for class level pointer permissions ${operation}`\n    );\n  }\n}\n\nconst joinClassRegex = /^_Join:[A-Za-z0-9_]+:[A-Za-z0-9_]+/;\nconst classAndFieldRegex = /^[A-Za-z][A-Za-z0-9_]*$/;\nfunction classNameIsValid(className: string): boolean {\n  // Valid classes must:\n  return (\n    // Be one of _User, _Installation, _Role, _Session OR\n    systemClasses.indexOf(className) > -1 ||\n    // Be a join table OR\n    joinClassRegex.test(className) ||\n    // Include only alpha-numeric and underscores, and not start with an underscore or number\n    fieldNameIsValid(className, className)\n  );\n}\n\n// Valid fields must be alpha-numeric, and not start with an underscore or number\n// must not be a reserved key\nfunction fieldNameIsValid(fieldName: string, className: string): boolean {\n  if (className && className !== '_Hooks') {\n    if (fieldName === 'className') {\n      return false;\n    }\n  }\n  return classAndFieldRegex.test(fieldName) && !invalidColumns.includes(fieldName);\n}\n\n// Checks that it's not trying to clobber one of the default fields of the class.\nfunction fieldNameIsValidForClass(fieldName: string, className: string): boolean {\n  if (!fieldNameIsValid(fieldName, className)) {\n    return false;\n  }\n  if (defaultColumns._Default[fieldName]) {\n    return false;\n  }\n  if (defaultColumns[className] && defaultColumns[className][fieldName]) {\n    return false;\n  }\n  return true;\n}\n\nfunction invalidClassNameMessage(className: string): string {\n  return (\n    'Invalid classname: ' +\n    className +\n    ', classnames can only have alphanumeric characters and _, and must start with an alpha character '\n  );\n}\n\nconst invalidJsonError = new Parse.Error(Parse.Error.INVALID_JSON, 'invalid JSON');\nconst validNonRelationOrPointerTypes = [\n  'Number',\n  'String',\n  'Boolean',\n  'Date',\n  'Object',\n  'Array',\n  'GeoPoint',\n  'File',\n  'Bytes',\n  'Polygon',\n];\n// Returns an error suitable for throwing if the type is invalid\nconst fieldTypeIsInvalid = ({ type, targetClass }) => {\n  if (['Pointer', 'Relation'].indexOf(type) >= 0) {\n    if (!targetClass) {\n      return new Parse.Error(135, `type ${type} needs a class name`);\n    } else if (typeof targetClass !== 'string') {\n      return invalidJsonError;\n    } else if (!classNameIsValid(targetClass)) {\n      return new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(targetClass));\n    } else {\n      return undefined;\n    }\n  }\n  if (typeof type !== 'string') {\n    return invalidJsonError;\n  }\n  if (validNonRelationOrPointerTypes.indexOf(type) < 0) {\n    return new Parse.Error(Parse.Error.INCORRECT_TYPE, `invalid field type: ${type}`);\n  }\n  return undefined;\n};\n\nconst convertSchemaToAdapterSchema = (schema: any) => {\n  schema = injectDefaultSchema(schema);\n  delete schema.fields.ACL;\n  schema.fields._rperm = { type: 'Array' };\n  schema.fields._wperm = { type: 'Array' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.password;\n    schema.fields._hashed_password = { type: 'String' };\n  }\n\n  return schema;\n};\n\nconst convertAdapterSchemaToParseSchema = ({ ...schema }) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  schema.fields.ACL = { type: 'ACL' };\n\n  if (schema.className === '_User') {\n    delete schema.fields.authData; //Auth data is implicit\n    delete schema.fields._hashed_password;\n    schema.fields.password = { type: 'String' };\n  }\n\n  if (schema.indexes && Object.keys(schema.indexes).length === 0) {\n    delete schema.indexes;\n  }\n\n  return schema;\n};\n\nclass SchemaData {\n  __data: any;\n  __protectedFields: any;\n  constructor(allSchemas = [], protectedFields = {}) {\n    this.__data = {};\n    this.__protectedFields = protectedFields;\n    allSchemas.forEach(schema => {\n      if (volatileClasses.includes(schema.className)) {\n        return;\n      }\n      Object.defineProperty(this, schema.className, {\n        get: () => {\n          if (!this.__data[schema.className]) {\n            const data = {};\n            data.fields = injectDefaultSchema(schema).fields;\n            data.classLevelPermissions = deepcopy(schema.classLevelPermissions);\n            data.indexes = schema.indexes;\n\n            const classProtectedFields = this.__protectedFields[schema.className];\n            if (classProtectedFields) {\n              for (const key in classProtectedFields) {\n                const unq = new Set([\n                  ...(data.classLevelPermissions.protectedFields[key] || []),\n                  ...classProtectedFields[key],\n                ]);\n                data.classLevelPermissions.protectedFields[key] = Array.from(unq);\n              }\n            }\n\n            this.__data[schema.className] = data;\n          }\n          return this.__data[schema.className];\n        },\n      });\n    });\n\n    // Inject the in-memory classes\n    volatileClasses.forEach(className => {\n      Object.defineProperty(this, className, {\n        get: () => {\n          if (!this.__data[className]) {\n            const schema = injectDefaultSchema({\n              className,\n              fields: {},\n              classLevelPermissions: {},\n            });\n            const data = {};\n            data.fields = schema.fields;\n            data.classLevelPermissions = schema.classLevelPermissions;\n            data.indexes = schema.indexes;\n            this.__data[className] = data;\n          }\n          return this.__data[className];\n        },\n      });\n    });\n  }\n}\n\nconst injectDefaultSchema = ({ className, fields, classLevelPermissions, indexes }: Schema) => {\n  const defaultSchema: Schema = {\n    className,\n    fields: {\n      ...defaultColumns._Default,\n      ...(defaultColumns[className] || {}),\n      ...fields,\n    },\n    classLevelPermissions,\n  };\n  if (indexes && Object.keys(indexes).length !== 0) {\n    defaultSchema.indexes = indexes;\n  }\n  return defaultSchema;\n};\n\nconst _HooksSchema = { className: '_Hooks', fields: defaultColumns._Hooks };\nconst _GlobalConfigSchema = {\n  className: '_GlobalConfig',\n  fields: defaultColumns._GlobalConfig,\n};\nconst _GraphQLConfigSchema = {\n  className: '_GraphQLConfig',\n  fields: defaultColumns._GraphQLConfig,\n};\nconst _PushStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_PushStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobStatusSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobStatus',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _JobScheduleSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_JobSchedule',\n    fields: {},\n    classLevelPermissions: {},\n  })\n);\nconst _AudienceSchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Audience',\n    fields: defaultColumns._Audience,\n    classLevelPermissions: {},\n  })\n);\nconst _IdempotencySchema = convertSchemaToAdapterSchema(\n  injectDefaultSchema({\n    className: '_Idempotency',\n    fields: defaultColumns._Idempotency,\n    classLevelPermissions: {},\n  })\n);\nconst VolatileClassesSchemas = [\n  _HooksSchema,\n  _JobStatusSchema,\n  _JobScheduleSchema,\n  _PushStatusSchema,\n  _GlobalConfigSchema,\n  _GraphQLConfigSchema,\n  _AudienceSchema,\n  _IdempotencySchema,\n];\n\nconst dbTypeMatchesObjectType = (dbType: SchemaField | string, objectType: SchemaField) => {\n  if (dbType.type !== objectType.type) { return false; }\n  if (dbType.targetClass !== objectType.targetClass) { return false; }\n  if (dbType === objectType.type) { return true; }\n  if (dbType.type === objectType.type) { return true; }\n  return false;\n};\n\nconst typeToString = (type: SchemaField | string): string => {\n  if (typeof type === 'string') {\n    return type;\n  }\n  if (type.targetClass) {\n    return `${type.type}<${type.targetClass}>`;\n  }\n  return `${type.type}`;\n};\nconst ttl = {\n  date: Date.now(),\n  duration: undefined,\n};\n\n// Stores the entire schema of the app in a weird hybrid format somewhere between\n// the mongo format and the Parse format. Soon, this will all be Parse format.\nexport default class SchemaController {\n  _dbAdapter: StorageAdapter;\n  schemaData: { [string]: Schema };\n  reloadDataPromise: ?Promise<any>;\n  protectedFields: any;\n  userIdRegEx: RegExp;\n\n  constructor(databaseAdapter: StorageAdapter) {\n    this._dbAdapter = databaseAdapter;\n    const config = Config.get(Parse.applicationId);\n    this.schemaData = new SchemaData(SchemaCache.all(), this.protectedFields);\n    this.protectedFields = config.protectedFields;\n\n    const customIds = config.allowCustomObjectId;\n\n    const customIdRegEx = /^.{1,}$/u; // 1+ chars\n    const autoIdRegEx = /^[a-zA-Z0-9]{1,}$/;\n\n    this.userIdRegEx = customIds ? customIdRegEx : autoIdRegEx;\n\n    this._dbAdapter.watch(() => {\n      this.reloadData({ clearCache: true });\n    });\n  }\n\n  async reloadDataIfNeeded() {\n    if (this._dbAdapter.enableSchemaHooks) {\n      return;\n    }\n    const { date, duration } = ttl || {};\n    if (!duration) {\n      return;\n    }\n    const now = Date.now();\n    if (now - date > duration) {\n      ttl.date = now;\n      await this.reloadData({ clearCache: true });\n    }\n  }\n\n  reloadData(options: LoadSchemaOptions = { clearCache: false }): Promise<any> {\n    if (this.reloadDataPromise && !options.clearCache) {\n      return this.reloadDataPromise;\n    }\n    this.reloadDataPromise = this.getAllClasses(options)\n      .then(\n        allSchemas => {\n          this.schemaData = new SchemaData(allSchemas, this.protectedFields);\n          delete this.reloadDataPromise;\n        },\n        err => {\n          this.schemaData = new SchemaData();\n          delete this.reloadDataPromise;\n          throw err;\n        }\n      )\n      .then(() => {});\n    return this.reloadDataPromise;\n  }\n\n  async getAllClasses(options: LoadSchemaOptions = { clearCache: false }): Promise<Array<Schema>> {\n    if (options.clearCache) {\n      return this.setAllClasses();\n    }\n    await this.reloadDataIfNeeded();\n    const cached = SchemaCache.all();\n    if (cached && cached.length) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses();\n  }\n\n  setAllClasses(): Promise<Array<Schema>> {\n    return this._dbAdapter\n      .getAllClasses()\n      .then(allSchemas => allSchemas.map(injectDefaultSchema))\n      .then(allSchemas => {\n        SchemaCache.put(allSchemas);\n        return allSchemas;\n      });\n  }\n\n  getOneSchema(\n    className: string,\n    allowVolatileClasses: boolean = false,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<Schema> {\n    if (options.clearCache) {\n      SchemaCache.clear();\n    }\n    if (allowVolatileClasses && volatileClasses.indexOf(className) > -1) {\n      const data = this.schemaData[className];\n      return Promise.resolve({\n        className,\n        fields: data.fields,\n        classLevelPermissions: data.classLevelPermissions,\n        indexes: data.indexes,\n      });\n    }\n    const cached = SchemaCache.get(className);\n    if (cached && !options.clearCache) {\n      return Promise.resolve(cached);\n    }\n    return this.setAllClasses().then(allSchemas => {\n      const oneSchema = allSchemas.find(schema => schema.className === className);\n      if (!oneSchema) {\n        return Promise.reject(undefined);\n      }\n      return oneSchema;\n    });\n  }\n\n  // Create a new class that includes the three default fields.\n  // ACL is an implicit column that does not get an entry in the\n  // _SCHEMAS database. Returns a promise that resolves with the\n  // created schema, in mongo format.\n  // on success, and rejects with an error on fail. Ensure you\n  // have authorization (master key, or client class creation\n  // enabled) before calling this function.\n  async addClassIfNotExists(\n    className: string,\n    fields: SchemaFields = {},\n    classLevelPermissions: any,\n    indexes: any = {}\n  ): Promise<void | Schema> {\n    var validationError = this.validateNewClass(className, fields, classLevelPermissions);\n    if (validationError) {\n      if (validationError instanceof Parse.Error) {\n        return Promise.reject(validationError);\n      } else if (validationError.code && validationError.error) {\n        return Promise.reject(new Parse.Error(validationError.code, validationError.error));\n      }\n      return Promise.reject(validationError);\n    }\n    try {\n      const adapterSchema = await this._dbAdapter.createClass(\n        className,\n        convertSchemaToAdapterSchema({\n          fields,\n          classLevelPermissions,\n          indexes,\n          className,\n        })\n      );\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n      const parseSchema = convertAdapterSchemaToParseSchema(adapterSchema);\n      return parseSchema;\n    } catch (error) {\n      if (error && error.code === Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n      } else {\n        throw error;\n      }\n    }\n  }\n\n  updateClass(\n    className: string,\n    submittedFields: SchemaFields,\n    classLevelPermissions: any,\n    indexes: any,\n    database: DatabaseController\n  ) {\n    return this.getOneSchema(className)\n      .then(schema => {\n        const existingFields = schema.fields;\n        Object.keys(submittedFields).forEach(name => {\n          const field = submittedFields[name];\n          if (\n            existingFields[name] &&\n            existingFields[name].type !== field.type &&\n            field.__op !== 'Delete'\n          ) {\n            throw new Parse.Error(255, `Field ${name} exists, cannot update.`);\n          }\n          if (!existingFields[name] && field.__op === 'Delete') {\n            throw new Parse.Error(255, `Field ${name} does not exist, cannot delete.`);\n          }\n        });\n\n        delete existingFields._rperm;\n        delete existingFields._wperm;\n        const newSchema = buildMergedSchemaObject(existingFields, submittedFields);\n        const defaultFields = defaultColumns[className] || defaultColumns._Default;\n        const fullNewSchema = Object.assign({}, newSchema, defaultFields);\n        const validationError = this.validateSchemaData(\n          className,\n          newSchema,\n          classLevelPermissions,\n          Object.keys(existingFields)\n        );\n        if (validationError) {\n          throw new Parse.Error(validationError.code, validationError.error);\n        }\n\n        // Finally we have checked to make sure the request is valid and we can start deleting fields.\n        // Do all deletions first, then a single save to _SCHEMA collection to handle all additions.\n        const deletedFields: string[] = [];\n        const insertedFields = [];\n        Object.keys(submittedFields).forEach(fieldName => {\n          if (submittedFields[fieldName].__op === 'Delete') {\n            deletedFields.push(fieldName);\n          } else {\n            insertedFields.push(fieldName);\n          }\n        });\n\n        let deletePromise = Promise.resolve();\n        if (deletedFields.length > 0) {\n          deletePromise = this.deleteFields(deletedFields, className, database);\n        }\n        let enforceFields = [];\n        return (\n          deletePromise // Delete Everything\n            .then(() => this.reloadData({ clearCache: true })) // Reload our Schema, so we have all the new values\n            .then(() => {\n              const promises = insertedFields.map(fieldName => {\n                const type = submittedFields[fieldName];\n                return this.enforceFieldExists(className, fieldName, type);\n              });\n              return Promise.all(promises);\n            })\n            .then(results => {\n              enforceFields = results.filter(result => !!result);\n              return this.setPermissions(className, classLevelPermissions, newSchema);\n            })\n            .then(() =>\n              this._dbAdapter.setIndexesWithSchemaFormat(\n                className,\n                indexes,\n                schema.indexes,\n                fullNewSchema\n              )\n            )\n            .then(() => this.reloadData({ clearCache: true }))\n            //TODO: Move this logic into the database adapter\n            .then(() => {\n              this.ensureFields(enforceFields);\n              const schema = this.schemaData[className];\n              const reloadedSchema: Schema = {\n                className: className,\n                fields: schema.fields,\n                classLevelPermissions: schema.classLevelPermissions,\n              };\n              if (schema.indexes && Object.keys(schema.indexes).length !== 0) {\n                reloadedSchema.indexes = schema.indexes;\n              }\n              return reloadedSchema;\n            })\n        );\n      })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      });\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object or fails with a reason.\n  enforceClassExists(className: string): Promise<SchemaController> {\n    if (this.schemaData[className]) {\n      return Promise.resolve(this);\n    }\n    // We don't have this class. Update the schema\n    return (\n      // The schema update succeeded. Reload the schema\n      this.addClassIfNotExists(className)\n        .catch(() => {\n          // The schema update failed. This can be okay - it might\n          // have failed because there's a race condition and a different\n          // client is making the exact same schema update that we want.\n          // So just reload the schema.\n          return this.reloadData({ clearCache: true });\n        })\n        .then(() => {\n          // Ensure that the schema now validates\n          if (this.schemaData[className]) {\n            return this;\n          } else {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, `Failed to add ${className}`);\n          }\n        })\n        .catch(() => {\n          // The schema still doesn't validate. Give up\n          throw new Parse.Error(Parse.Error.INVALID_JSON, 'schema class name does not revalidate');\n        })\n    );\n  }\n\n  validateNewClass(className: string, fields: SchemaFields = {}, classLevelPermissions: any): any {\n    if (this.schemaData[className]) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, `Class ${className} already exists.`);\n    }\n    if (!classNameIsValid(className)) {\n      return {\n        code: Parse.Error.INVALID_CLASS_NAME,\n        error: invalidClassNameMessage(className),\n      };\n    }\n    return this.validateSchemaData(className, fields, classLevelPermissions, []);\n  }\n\n  validateSchemaData(\n    className: string,\n    fields: SchemaFields,\n    classLevelPermissions: ClassLevelPermissions,\n    existingFieldNames: Array<string>\n  ) {\n    for (const fieldName in fields) {\n      if (existingFieldNames.indexOf(fieldName) < 0) {\n        if (!fieldNameIsValid(fieldName, className)) {\n          return {\n            code: Parse.Error.INVALID_KEY_NAME,\n            error: 'invalid field name: ' + fieldName,\n          };\n        }\n        if (!fieldNameIsValidForClass(fieldName, className)) {\n          return {\n            code: 136,\n            error: 'field ' + fieldName + ' cannot be added',\n          };\n        }\n        const fieldType = fields[fieldName];\n        const error = fieldTypeIsInvalid(fieldType);\n        if (error) { return { code: error.code, error: error.message }; }\n        if (fieldType.defaultValue !== undefined) {\n          let defaultValueType = getType(fieldType.defaultValue);\n          if (typeof defaultValueType === 'string') {\n            defaultValueType = { type: defaultValueType };\n          } else if (typeof defaultValueType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'default value' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n          if (!dbTypeMatchesObjectType(fieldType, defaultValueType)) {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n                fieldType\n              )} but got ${typeToString(defaultValueType)}`,\n            };\n          }\n        } else if (fieldType.required) {\n          if (typeof fieldType === 'object' && fieldType.type === 'Relation') {\n            return {\n              code: Parse.Error.INCORRECT_TYPE,\n              error: `The 'required' option is not applicable for ${typeToString(fieldType)}`,\n            };\n          }\n        }\n      }\n    }\n\n    for (const fieldName in defaultColumns[className]) {\n      fields[fieldName] = defaultColumns[className][fieldName];\n    }\n\n    const geoPoints = Object.keys(fields).filter(\n      key => fields[key] && fields[key].type === 'GeoPoint'\n    );\n    if (geoPoints.length > 1) {\n      return {\n        code: Parse.Error.INCORRECT_TYPE,\n        error:\n          'currently, only one GeoPoint field may exist in an object. Adding ' +\n          geoPoints[1] +\n          ' when ' +\n          geoPoints[0] +\n          ' already exists.',\n      };\n    }\n    validateCLP(classLevelPermissions, fields, this.userIdRegEx);\n  }\n\n  // Sets the Class-level permissions for a given className, which must exist.\n  async setPermissions(className: string, perms: any, newSchema: SchemaFields) {\n    if (typeof perms === 'undefined') {\n      return Promise.resolve();\n    }\n    validateCLP(perms, newSchema, this.userIdRegEx);\n    await this._dbAdapter.setClassLevelPermissions(className, perms);\n    const cached = SchemaCache.get(className);\n    if (cached) {\n      cached.classLevelPermissions = perms;\n    }\n  }\n\n  // Returns a promise that resolves successfully to the new schema\n  // object if the provided className-fieldName-type tuple is valid.\n  // The className must already be validated.\n  // If 'freeze' is true, refuse to update the schema for this field.\n  enforceFieldExists(\n    className: string,\n    fieldName: string,\n    type: string | SchemaField,\n    isValidation?: boolean,\n    maintenance?: boolean\n  ) {\n    if (fieldName.indexOf('.') > 0) {\n      // \"<array>.<index>\" for Nested Arrays\n      // \"<embedded document>.<field>\" for Nested Objects\n      // JSON Arrays are treated as Nested Objects\n      const [x, y] = fieldName.split('.');\n      fieldName = x;\n      const isArrayIndex = Array.from(y).every(c => c >= '0' && c <= '9');\n      if (isArrayIndex && !['sentPerUTCOffset', 'failedPerUTCOffset'].includes(fieldName)) {\n        type = 'Array';\n      } else {\n        type = 'Object';\n      }\n    }\n    let fieldNameToValidate = `${fieldName}`;\n    if (maintenance && fieldNameToValidate.charAt(0) === '_') {\n      fieldNameToValidate = fieldNameToValidate.substring(1);\n    }\n    if (!fieldNameIsValid(fieldNameToValidate, className)) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid field name: ${fieldName}.`);\n    }\n\n    // If someone tries to create a new field with null/undefined as the value, return;\n    if (!type) {\n      return undefined;\n    }\n\n    const expectedType = this.getExpectedType(className, fieldName);\n    if (typeof type === 'string') {\n      type = ({ type }: SchemaField);\n    }\n\n    if (type.defaultValue !== undefined) {\n      let defaultValueType = getType(type.defaultValue);\n      if (typeof defaultValueType === 'string') {\n        defaultValueType = { type: defaultValueType };\n      }\n      if (!dbTypeMatchesObjectType(type, defaultValueType)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName} default value; expected ${typeToString(\n            type\n          )} but got ${typeToString(defaultValueType)}`\n        );\n      }\n    }\n\n    if (expectedType) {\n      if (!dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(\n          Parse.Error.INCORRECT_TYPE,\n          `schema mismatch for ${className}.${fieldName}; expected ${typeToString(\n            expectedType\n          )} but got ${typeToString(type)}`\n        );\n      }\n      // If type options do not change\n      // we can safely return\n      if (isValidation || JSON.stringify(expectedType) === JSON.stringify(type)) {\n        return undefined;\n      }\n      // Field options are may be changed\n      // ensure to have an update to date schema field\n      return this._dbAdapter.updateFieldOptions(className, fieldName, type);\n    }\n\n    return this._dbAdapter\n      .addFieldIfNotExists(className, fieldName, type)\n      .catch(error => {\n        if (error.code == Parse.Error.INCORRECT_TYPE) {\n          // Make sure that we throw errors when it is appropriate to do so.\n          throw error;\n        }\n        // The update failed. This can be okay - it might have been a race\n        // condition where another client updated the schema in the same\n        // way that we wanted to. So, just reload the schema\n        return Promise.resolve();\n      })\n      .then(() => {\n        return {\n          className,\n          fieldName,\n          type,\n        };\n      });\n  }\n\n  ensureFields(fields: any) {\n    for (let i = 0; i < fields.length; i += 1) {\n      const { className, fieldName } = fields[i];\n      let { type } = fields[i];\n      const expectedType = this.getExpectedType(className, fieldName);\n      if (typeof type === 'string') {\n        type = { type: type };\n      }\n      if (!expectedType || !dbTypeMatchesObjectType(expectedType, type)) {\n        throw new Parse.Error(Parse.Error.INVALID_JSON, `Could not add field ${fieldName}`);\n      }\n    }\n  }\n\n  // maintain compatibility\n  deleteField(fieldName: string, className: string, database: DatabaseController) {\n    return this.deleteFields([fieldName], className, database);\n  }\n\n  // Delete fields, and remove that data from all objects. This is intended\n  // to remove unused fields, if other writers are writing objects that include\n  // this field, the field may reappear. Returns a Promise that resolves with\n  // no object on success, or rejects with { code, error } on failure.\n  // Passing the database and prefix is necessary in order to drop relation collections\n  // and remove fields from objects. Ideally the database would belong to\n  // a database adapter and this function would close over it or access it via member.\n  deleteFields(fieldNames: Array<string>, className: string, database: DatabaseController) {\n    if (!classNameIsValid(className)) {\n      throw new Parse.Error(Parse.Error.INVALID_CLASS_NAME, invalidClassNameMessage(className));\n    }\n\n    fieldNames.forEach(fieldName => {\n      if (!fieldNameIsValid(fieldName, className)) {\n        throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `invalid field name: ${fieldName}`);\n      }\n      //Don't allow deleting the default fields.\n      if (!fieldNameIsValidForClass(fieldName, className)) {\n        throw new Parse.Error(136, `field ${fieldName} cannot be changed`);\n      }\n    });\n\n    return this.getOneSchema(className, false, { clearCache: true })\n      .catch(error => {\n        if (error === undefined) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_CLASS_NAME,\n            `Class ${className} does not exist.`\n          );\n        } else {\n          throw error;\n        }\n      })\n      .then(schema => {\n        fieldNames.forEach(fieldName => {\n          if (!schema.fields[fieldName]) {\n            throw new Parse.Error(255, `Field ${fieldName} does not exist, cannot delete.`);\n          }\n        });\n\n        const schemaFields = { ...schema.fields };\n        return database.adapter.deleteFields(className, schema, fieldNames).then(() => {\n          return Promise.all(\n            fieldNames.map(fieldName => {\n              const field = schemaFields[fieldName];\n              if (field && field.type === 'Relation') {\n                //For relations, drop the _Join table\n                return database.adapter.deleteClass(`_Join:${fieldName}:${className}`);\n              }\n              return Promise.resolve();\n            })\n          );\n        });\n      })\n      .then(() => {\n        SchemaCache.clear();\n      });\n  }\n\n  // Validates an object provided in REST format.\n  // Returns a promise that resolves to the new schema if this object is\n  // valid.\n  async validateObject(className: string, object: any, query: any, maintenance: boolean) {\n    let geocount = 0;\n    const schema = await this.enforceClassExists(className);\n    const promises = [];\n\n    for (const fieldName in object) {\n      if (object[fieldName] && getType(object[fieldName]) === 'GeoPoint') {\n        geocount++;\n      }\n      if (geocount > 1) {\n        return Promise.reject(\n          new Parse.Error(\n            Parse.Error.INCORRECT_TYPE,\n            'there can only be one geopoint field in a class'\n          )\n        );\n      }\n    }\n    for (const fieldName in object) {\n      if (object[fieldName] === undefined) {\n        continue;\n      }\n      const expected = getType(object[fieldName]);\n      if (!expected) {\n        continue;\n      }\n      if (fieldName === 'ACL') {\n        // Every object has ACL implicitly.\n        continue;\n      }\n      promises.push(schema.enforceFieldExists(className, fieldName, expected, true, maintenance));\n    }\n    const results = await Promise.all(promises);\n    const enforceFields = results.filter(result => !!result);\n\n    if (enforceFields.length !== 0) {\n      // TODO: Remove by updating schema cache directly\n      await this.reloadData({ clearCache: true });\n    }\n    this.ensureFields(enforceFields);\n\n    const promise = Promise.resolve(schema);\n    return thenValidateRequiredColumns(promise, className, object, query);\n  }\n\n  // Validates that all the properties are set for the object\n  validateRequiredColumns(className: string, object: any, query: any) {\n    const columns = requiredColumns.write[className];\n    if (!columns || columns.length == 0) {\n      return Promise.resolve(this);\n    }\n\n    const missingColumns = columns.filter(function (column) {\n      if (query && query.objectId) {\n        if (object[column] && typeof object[column] === 'object') {\n          // Trying to delete a required column\n          return object[column].__op == 'Delete';\n        }\n        // Not trying to do anything there\n        return false;\n      }\n      return !object[column];\n    });\n\n    if (missingColumns.length > 0) {\n      throw new Parse.Error(Parse.Error.INCORRECT_TYPE, missingColumns[0] + ' is required.');\n    }\n    return Promise.resolve(this);\n  }\n\n  testPermissionsForClassName(className: string, aclGroup: string[], operation: string) {\n    return SchemaController.testPermissions(\n      this.getClassLevelPermissions(className),\n      aclGroup,\n      operation\n    );\n  }\n\n  // Tests that the class level permission let pass the operation for a given aclGroup\n  static testPermissions(classPermissions: ?any, aclGroup: string[], operation: string): boolean {\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    if (perms['*']) {\n      return true;\n    }\n    // Check permissions against the aclGroup provided (array of userId/roles)\n    if (\n      aclGroup.some(acl => {\n        return perms[acl] === true;\n      })\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  static validatePermission(\n    classPermissions: ?any,\n    className: string,\n    aclGroup: string[],\n    operation: string,\n    action?: string\n  ) {\n    if (SchemaController.testPermissions(classPermissions, aclGroup, operation)) {\n      return Promise.resolve();\n    }\n\n    if (!classPermissions || !classPermissions[operation]) {\n      return true;\n    }\n    const perms = classPermissions[operation];\n    const config = Config.get(Parse.applicationId)\n    // If only for authenticated users\n    // make sure we have an aclGroup\n    if (perms['requiresAuthentication']) {\n      // If aclGroup has * (public)\n      if (!aclGroup || aclGroup.length == 0) {\n        throw createSanitizedError(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.',\n          config\n        );\n      } else if (aclGroup.indexOf('*') > -1 && aclGroup.length == 1) {\n        throw createSanitizedError(\n          Parse.Error.OBJECT_NOT_FOUND,\n          'Permission denied, user needs to be authenticated.',\n          config\n        );\n      }\n      // requiresAuthentication passed, just move forward\n      // probably would be wise at some point to rename to 'authenticatedUser'\n      return Promise.resolve();\n    }\n\n    // No matching CLP, let's check the Pointer permissions\n    // And handle those later\n    const permissionField =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    // Reject create when write lockdown\n    if (permissionField == 'writeUserFields' && operation == 'create') {\n      throw createSanitizedError(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `Permission denied for action ${operation} on class ${className}.`,\n        config\n      );\n    }\n\n    // Process the readUserFields later\n    if (\n      Array.isArray(classPermissions[permissionField]) &&\n      classPermissions[permissionField].length > 0\n    ) {\n      return Promise.resolve();\n    }\n\n    const pointerFields = classPermissions[operation].pointerFields;\n    if (Array.isArray(pointerFields) && pointerFields.length > 0) {\n      // any op except 'addField as part of create' is ok.\n      if (operation !== 'addField' || action === 'update') {\n        // We can allow adding field on update flow only.\n        return Promise.resolve();\n      }\n    }\n\n    throw createSanitizedError(\n      Parse.Error.OPERATION_FORBIDDEN,\n      `Permission denied for action ${operation} on class ${className}.`,\n      config\n    );\n  }\n\n  // Validates an operation passes class-level-permissions set in the schema\n  validatePermission(className: string, aclGroup: string[], operation: string, action?: string) {\n    return SchemaController.validatePermission(\n      this.getClassLevelPermissions(className),\n      className,\n      aclGroup,\n      operation,\n      action\n    );\n  }\n\n  getClassLevelPermissions(className: string): any {\n    return this.schemaData[className] && this.schemaData[className].classLevelPermissions;\n  }\n\n  // Returns the expected type for a className+key combination\n  // or undefined if the schema is not set\n  getExpectedType(className: string, fieldName: string): ?(SchemaField | string) {\n    if (this.schemaData[className]) {\n      const expectedType = this.schemaData[className].fields[fieldName];\n      return expectedType === 'map' ? 'Object' : expectedType;\n    }\n    return undefined;\n  }\n\n  // Checks if a given class is in the schema.\n  hasClass(className: string) {\n    if (this.schemaData[className]) {\n      return Promise.resolve(true);\n    }\n    return this.reloadData().then(() => !!this.schemaData[className]);\n  }\n}\n\n// Returns a promise for a new Schema.\nconst load = (dbAdapter: StorageAdapter, options: any): Promise<SchemaController> => {\n  const schema = new SchemaController(dbAdapter);\n  ttl.duration = dbAdapter.schemaCacheTtl;\n  return schema.reloadData(options).then(() => schema);\n};\n\n// Builds a new schema (in schema API response format) out of an\n// existing mongo schema + a schemas API put request. This response\n// does not include the default fields, as it is intended to be passed\n// to mongoSchemaFromFieldsAndClassName. No validation is done here, it\n// is done in mongoSchemaFromFieldsAndClassName.\nfunction buildMergedSchemaObject(existingFields: SchemaFields, putRequest: any): SchemaFields {\n  const newSchema = {};\n  // @flow-disable-next\n  const sysSchemaField =\n    Object.keys(defaultColumns).indexOf(existingFields._id) === -1\n      ? []\n      : Object.keys(defaultColumns[existingFields._id]);\n  for (const oldField in existingFields) {\n    if (\n      oldField !== '_id' &&\n      oldField !== 'ACL' &&\n      oldField !== 'updatedAt' &&\n      oldField !== 'createdAt' &&\n      oldField !== 'objectId'\n    ) {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(oldField) !== -1) {\n        continue;\n      }\n      const fieldIsDeleted = putRequest[oldField] && putRequest[oldField].__op === 'Delete';\n      if (!fieldIsDeleted) {\n        newSchema[oldField] = existingFields[oldField];\n      }\n    }\n  }\n  for (const newField in putRequest) {\n    if (newField !== 'objectId' && putRequest[newField].__op !== 'Delete') {\n      if (sysSchemaField.length > 0 && sysSchemaField.indexOf(newField) !== -1) {\n        continue;\n      }\n      newSchema[newField] = putRequest[newField];\n    }\n  }\n  return newSchema;\n}\n\n// Given a schema promise, construct another schema promise that\n// validates this field once the schema loads.\nfunction thenValidateRequiredColumns(schemaPromise, className, object, query) {\n  return schemaPromise.then(schema => {\n    return schema.validateRequiredColumns(className, object, query);\n  });\n}\n\n// Gets the type from a REST API formatted object, where 'type' is\n// extended past javascript types to include the rest of the Parse\n// type system.\n// The output should be a valid schema value.\n// TODO: ensure that this is compatible with the format used in Open DB\nfunction getType(obj: any): ?(SchemaField | string) {\n  const type = typeof obj;\n  switch (type) {\n    case 'boolean':\n      return 'Boolean';\n    case 'string':\n      return 'String';\n    case 'number':\n      return 'Number';\n    case 'map':\n    case 'object':\n      if (!obj) {\n        return undefined;\n      }\n      return getObjectType(obj);\n    case 'function':\n    case 'symbol':\n    case 'undefined':\n    default:\n      throw 'bad obj: ' + obj;\n  }\n}\n\n// This gets the type for non-JSON types like pointers and files, but\n// also gets the appropriate type for $ operators.\n// Returns null if the type is unknown.\nfunction getObjectType(obj): ?(SchemaField | string) {\n  if (obj instanceof Array) {\n    return 'Array';\n  }\n  if (obj.__type) {\n    switch (obj.__type) {\n      case 'Pointer':\n        if (obj.className) {\n          return {\n            type: 'Pointer',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'Relation':\n        if (obj.className) {\n          return {\n            type: 'Relation',\n            targetClass: obj.className,\n          };\n        }\n        break;\n      case 'File':\n        if (obj.name) {\n          return 'File';\n        }\n        break;\n      case 'Date':\n        if (obj.iso) {\n          return 'Date';\n        }\n        break;\n      case 'GeoPoint':\n        if (obj.latitude != null && obj.longitude != null) {\n          return 'GeoPoint';\n        }\n        break;\n      case 'Bytes':\n        if (obj.base64) {\n          return 'Bytes';\n        }\n        break;\n      case 'Polygon':\n        if (obj.coordinates) {\n          return 'Polygon';\n        }\n        break;\n    }\n    throw new Parse.Error(Parse.Error.INCORRECT_TYPE, 'This is not a valid ' + obj.__type);\n  }\n  if (obj['$ne']) {\n    return getObjectType(obj['$ne']);\n  }\n  if (obj.__op) {\n    switch (obj.__op) {\n      case 'Increment':\n        return 'Number';\n      case 'Delete':\n        return null;\n      case 'Add':\n      case 'AddUnique':\n      case 'Remove':\n        return 'Array';\n      case 'AddRelation':\n      case 'RemoveRelation':\n        return {\n          type: 'Relation',\n          targetClass: obj.objects[0].className,\n        };\n      case 'Batch':\n        return getObjectType(obj.ops[0]);\n      default:\n        throw 'unexpected op: ' + obj.__op;\n    }\n  }\n  return 'Object';\n}\n\nexport {\n  load,\n  classNameIsValid,\n  fieldNameIsValid,\n  invalidClassNameMessage,\n  buildMergedSchemaObject,\n  systemClasses,\n  defaultColumns,\n  convertSchemaToAdapterSchema,\n  VolatileClassesSchemas,\n  SchemaController,\n  requiredColumns,\n};\n"],"mappings":"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","ignoreList":[]}