parse-server 5.3.0-alpha.2 → 5.3.0-alpha.20
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +50 -44
- package/lib/Adapters/Auth/gcenter.js +104 -30
- package/lib/Adapters/Cache/LRUCache.js +2 -2
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +29 -20
- package/lib/Auth.js +11 -1
- package/lib/Controllers/DatabaseController.js +4 -3
- package/lib/Controllers/LiveQueryController.js +9 -1
- package/lib/Controllers/SchemaController.js +13 -6
- package/lib/Controllers/index.js +3 -2
- package/lib/Deprecator/Deprecations.js +4 -1
- package/lib/GraphQL/ParseGraphQLSchema.js +39 -58
- package/lib/GraphQL/ParseGraphQLServer.js +37 -26
- package/lib/GraphQL/helpers/objectsQueries.js +5 -11
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +14 -11
- package/lib/GraphQL/loaders/defaultRelaySchema.js +2 -2
- package/lib/GraphQL/loaders/filesMutations.js +9 -22
- package/lib/GraphQL/loaders/schemaDirectives.js +51 -42
- package/lib/GraphQL/loaders/schemaTypes.js +14 -14
- package/lib/GraphQL/parseGraphQLUtils.js +5 -3
- package/lib/GraphQL/transformers/mutation.js +2 -2
- package/lib/GraphQL/transformers/query.js +3 -3
- package/lib/LiveQuery/ParseCloudCodePublisher.js +11 -1
- package/lib/LiveQuery/ParseLiveQueryServer.js +78 -17
- package/lib/LiveQuery/ParseWebSocketServer.js +8 -2
- package/lib/LiveQuery/SessionTokenCache.js +2 -2
- package/lib/Options/Definitions.js +55 -38
- package/lib/Options/docs.js +19 -2
- package/lib/Options/index.js +1 -1
- package/lib/RestWrite.js +33 -3
- package/lib/Routers/FilesRouter.js +23 -14
- package/lib/SchemaMigrations/Migrations.js +1 -1
- package/lib/cloud-code/Parse.Cloud.js +36 -16
- package/lib/triggers.js +7 -18
- package/package.json +34 -34
package/README.md
CHANGED
|
@@ -7,8 +7,8 @@
|
|
|
7
7
|
[](https://codecov.io/github/parse-community/parse-server?branch=alpha)
|
|
8
8
|
[](https://github.com/parse-community/parse-dashboard/releases)
|
|
9
9
|
|
|
10
|
-
[](https://nodejs.org)
|
|
11
|
-
[](https://www.mongodb.com)
|
|
10
|
+
[](https://nodejs.org)
|
|
11
|
+
[](https://www.mongodb.com)
|
|
12
12
|
[](https://www.postgresql.org)
|
|
13
13
|
|
|
14
14
|
[](https://www.npmjs.com/package/parse-server)
|
|
@@ -20,6 +20,7 @@
|
|
|
20
20
|
[![License][license-svg]][license-link]
|
|
21
21
|
[](https://community.parseplatform.org/c/parse-server)
|
|
22
22
|
[](https://twitter.com/intent/follow?screen_name=ParsePlatform)
|
|
23
|
+
[](https://chat.parseplatform.org)
|
|
23
24
|
|
|
24
25
|
---
|
|
25
26
|
|
|
@@ -41,6 +42,8 @@ A big *thank you* 🙏 to our [sponsors](#sponsors) and [backers](#backers) who
|
|
|
41
42
|
|
|
42
43
|
---
|
|
43
44
|
|
|
45
|
+
- [Flavors & Branches](#flavors--branches)
|
|
46
|
+
- [Long Term Support](#long-term-support)
|
|
44
47
|
- [Getting Started](#getting-started)
|
|
45
48
|
- [Running Parse Server](#running-parse-server)
|
|
46
49
|
- [Compatibility](#compatibility)
|
|
@@ -88,13 +91,25 @@ A big *thank you* 🙏 to our [sponsors](#sponsors) and [backers](#backers) who
|
|
|
88
91
|
- [Using automatically generated operations](#using-automatically-generated-operations)
|
|
89
92
|
- [Customizing your GraphQL Schema](#customizing-your-graphql-schema)
|
|
90
93
|
- [Learning more](#learning-more)
|
|
91
|
-
- [Upgrading to 3.0
|
|
92
|
-
- [Want to ride the bleeding edge?](#want-to-ride-the-bleeding-edge)
|
|
94
|
+
- [Upgrading to Parse Server 3.0](#upgrading-to-parse-server-30)
|
|
93
95
|
- [Contributing](#contributing)
|
|
94
96
|
- [Contributors](#contributors)
|
|
95
97
|
- [Sponsors](#sponsors)
|
|
96
98
|
- [Backers](#backers)
|
|
97
99
|
|
|
100
|
+
# Flavors & Branches
|
|
101
|
+
|
|
102
|
+
Parse Server is available in different flavors on different branches:
|
|
103
|
+
|
|
104
|
+
- The main branches are [release][log_release], [beta][log_beta] and [alpha][log_alpha]. See the [changelog overview](CHANGELOG.md) for details.
|
|
105
|
+
- The long-term-support (LTS) branches are named `release-<version>.x.x`, for example `release-4.x.x`. LTS branches do not have pre-release branches.
|
|
106
|
+
|
|
107
|
+
## Long Term Support
|
|
108
|
+
|
|
109
|
+
Long-Term-Support (LTS) is provided for the previous Parse Server major version. For example, Parse Server 4.x will receive security updates until Parse Server 5.x is superseded by Parse Server 6.x and becomes the new LTS version. While the current major version is published on branch `release`, a LTS version is published on branch `release-#.x.x`, for example `release-4.x.x` for the Parse Server 4.x LTS branch.
|
|
110
|
+
|
|
111
|
+
⚠️ LTS versions are provided to help you transition as soon as possible to the current major version. While we aim to fix security vulnerabilities in the LTS version, our main focus is on developing the current major version and preparing the next major release. Therefore we may leave certain vulnerabilities up to the community to fix. Search for [pull requests with the specific LTS base branch](https://github.com/parse-community/parse-server/pulls?q=is%3Aopen+is%3Apr+base%3Arelease-4.x.x) to see the current open vulnerabilities for that LTS branch.
|
|
112
|
+
|
|
98
113
|
# Getting Started
|
|
99
114
|
|
|
100
115
|
The fastest and easiest way to get started is to run MongoDB and Parse Server locally.
|
|
@@ -110,27 +125,32 @@ Before you start make sure you have installed:
|
|
|
110
125
|
### Compatibility
|
|
111
126
|
|
|
112
127
|
#### Node.js
|
|
128
|
+
|
|
113
129
|
Parse Server is continuously tested with the most recent releases of Node.js to ensure compatibility. We follow the [Node.js Long Term Support plan](https://github.com/nodejs/Release) and only test against versions that are officially supported and have not reached their end-of-life date.
|
|
114
130
|
|
|
115
|
-
| Version | Latest Version | End-of-Life | Compatible
|
|
116
|
-
|
|
117
|
-
| Node.js 12 | 12.22.
|
|
118
|
-
| Node.js 14 | 14.
|
|
119
|
-
| Node.js 16 | 16.
|
|
120
|
-
| Node.js 17 | 17.
|
|
131
|
+
| Version | Latest Version | End-of-Life | Compatible |
|
|
132
|
+
|------------|----------------|-------------|------------|
|
|
133
|
+
| Node.js 12 | 12.22.11 | April 2022 | ✅ Yes |
|
|
134
|
+
| Node.js 14 | 14.19.1 | April 2023 | ✅ Yes |
|
|
135
|
+
| Node.js 16 | 16.14.2 | April 2024 | ✅ Yes |
|
|
136
|
+
| Node.js 17 | 17.9.0 | June 2022 | ✅ Yes |
|
|
137
|
+
| Node.js 18 | 18.1.0 | April 2025 | ✅ Yes |
|
|
121
138
|
|
|
122
139
|
#### MongoDB
|
|
140
|
+
|
|
123
141
|
Parse Server is continuously tested with the most recent releases of MongoDB to ensure compatibility. We follow the [MongoDB support schedule](https://www.mongodb.com/support-policy) and only test against versions that are officially supported and have not reached their end-of-life date.
|
|
124
142
|
|
|
125
|
-
| Version | Latest Version | End-of-Life
|
|
126
|
-
|
|
127
|
-
| MongoDB 4.0 | 4.0.
|
|
128
|
-
| MongoDB 4.2 | 4.2.
|
|
129
|
-
| MongoDB 4.4 | 4.4.
|
|
130
|
-
| MongoDB 5.0 | 5.0.
|
|
131
|
-
| MongoDB 5.1 | 5.1.
|
|
143
|
+
| Version | Latest Version | End-of-Life | Compatible |
|
|
144
|
+
|-------------|----------------|-------------|------------|
|
|
145
|
+
| MongoDB 4.0 | 4.0.28 | April 2022 | ✅ Yes |
|
|
146
|
+
| MongoDB 4.2 | 4.2.19 | TBD | ✅ Yes |
|
|
147
|
+
| MongoDB 4.4 | 4.4.13 | TBD | ✅ Yes |
|
|
148
|
+
| MongoDB 5.0 | 5.0.6 | TBD | ✅ Yes |
|
|
149
|
+
| MongoDB 5.1 | 5.1.1 | TBD | ✅ Yes |
|
|
150
|
+
| MongoDB 5.2 | 5.2.1 | TBD | ✅ Yes |
|
|
132
151
|
|
|
133
152
|
#### PostgreSQL
|
|
153
|
+
|
|
134
154
|
Parse Server is continuously tested with the most recent releases of PostgreSQL and PostGIS to ensure compatibility, using [PostGIS docker images](https://registry.hub.docker.com/r/postgis/postgis/tags?page=1&ordering=last_updated). We follow the [PostgreSQL support schedule](https://www.postgresql.org/support/versioning) and [PostGIS support schedule](https://www.postgis.net/eol_policy/) and only test against versions that are officially supported and have not reached their end-of-life date. Due to the extensive PostgreSQL support duration of 5 years, Parse Server drops support if a version is older than 3.5 years and a newer version has been available for at least 2.5 years.
|
|
135
155
|
|
|
136
156
|
| Version | PostGIS Version | End-of-Life | Parse Server Support End | Compatible |
|
|
@@ -141,6 +161,7 @@ Parse Server is continuously tested with the most recent releases of PostgreSQL
|
|
|
141
161
|
| Postgres 14 | 3.2 | November 2026 | April 2025 | ✅ Yes |
|
|
142
162
|
|
|
143
163
|
### Locally
|
|
164
|
+
|
|
144
165
|
```bash
|
|
145
166
|
$ npm install -g parse-server mongodb-runner
|
|
146
167
|
$ mongodb-runner start
|
|
@@ -233,7 +254,6 @@ $ curl -X GET \
|
|
|
233
254
|
}
|
|
234
255
|
]
|
|
235
256
|
}
|
|
236
|
-
|
|
237
257
|
```
|
|
238
258
|
|
|
239
259
|
To learn more about using saving and querying objects on Parse Server, check out the [Parse documentation](http://docs.parseplatform.org).
|
|
@@ -383,6 +403,7 @@ const server = ParseServer({
|
|
|
383
403
|
```
|
|
384
404
|
|
|
385
405
|
## Custom Routes
|
|
406
|
+
|
|
386
407
|
**Caution, this is an experimental feature that may not be appropriate for production.**
|
|
387
408
|
|
|
388
409
|
Custom routes allow to build user flows with webpages, similar to the existing password reset and email verification features. Custom routes are defined with the `pages` option in the Parse Server configuration:
|
|
@@ -415,6 +436,7 @@ The above route can be invoked by sending a `GET` request to:
|
|
|
415
436
|
The `handler` receives the `request` and returns a `custom_page.html` webpage from the `pages.pagesPath` directory as response. The advantage of building a custom route this way is that it automatically makes use of Parse Server's built-in capabilities, such as [page localization](#pages) and [dynamic placeholders](#dynamic-placeholders).
|
|
416
437
|
|
|
417
438
|
### Reserved Paths
|
|
439
|
+
|
|
418
440
|
The following paths are already used by Parse Server's built-in features and are therefore not available for custom routes. Custom routes with an identical combination of `path` and `method` are ignored.
|
|
419
441
|
|
|
420
442
|
| Path | HTTP Method | Feature |
|
|
@@ -510,6 +532,7 @@ Identical requests are identified by their request header `X-Parse-Request-Id`.
|
|
|
510
532
|
Deduplication is only done for object creation and update (`POST` and `PUT` requests). Deduplication is not done for object finding and deletion (`GET` and `DELETE` requests), as these operations are already idempotent by definition.
|
|
511
533
|
|
|
512
534
|
### Configuration example <!-- omit in toc -->
|
|
535
|
+
|
|
513
536
|
```
|
|
514
537
|
let api = new ParseServer({
|
|
515
538
|
idempotencyOptions: {
|
|
@@ -518,6 +541,7 @@ let api = new ParseServer({
|
|
|
518
541
|
}
|
|
519
542
|
}
|
|
520
543
|
```
|
|
544
|
+
|
|
521
545
|
### Parameters <!-- omit in toc -->
|
|
522
546
|
|
|
523
547
|
| Parameter | Optional | Type | Default value | Example values | Environment variable | Description |
|
|
@@ -550,6 +574,7 @@ Assuming the script above is named, `parse_idempotency_delete_expired_records.sh
|
|
|
550
574
|
## Localization
|
|
551
575
|
|
|
552
576
|
### Pages
|
|
577
|
+
|
|
553
578
|
**Caution, this is an experimental feature that may not be appropriate for production.**
|
|
554
579
|
|
|
555
580
|
Custom pages as well as feature pages (e.g. password reset, email verification) can be localized with the `pages` option in the Parse Server configuration:
|
|
@@ -1093,37 +1118,15 @@ You also have a very powerful tool inside your GraphQL Playground. Please look a
|
|
|
1093
1118
|
|
|
1094
1119
|
Additionally, the [GraphQL Learn Section](https://graphql.org/learn/) is a very good source to learn more about the power of the GraphQL language.
|
|
1095
1120
|
|
|
1096
|
-
# Upgrading to 3.0
|
|
1097
|
-
|
|
1098
|
-
Starting 3.0.0, parse-server uses the JS SDK version 2.0.
|
|
1099
|
-
In short, parse SDK v2.0 removes the backbone style callbacks as well as the Parse.Promise object in favor of native promises.
|
|
1100
|
-
All the Cloud Code interfaces also have been updated to reflect those changes, and all backbone style response objects are removed and replaced by Promise style resolution.
|
|
1101
|
-
|
|
1102
|
-
We have written up a [migration guide](3.0.0.md), hoping this will help you transition to the next major release.
|
|
1121
|
+
# Upgrading to Parse Server 3.0
|
|
1103
1122
|
|
|
1104
|
-
|
|
1105
|
-
|
|
1106
|
-
It is recommend to use builds deployed npm for many reasons, but if you want to use
|
|
1107
|
-
the latest not-yet-released version of parse-server, you can do so by depending
|
|
1108
|
-
directly on this branch:
|
|
1109
|
-
|
|
1110
|
-
```
|
|
1111
|
-
npm install parse-community/parse-server.git#master
|
|
1112
|
-
```
|
|
1113
|
-
|
|
1114
|
-
## Experimenting <!-- omit in toc -->
|
|
1115
|
-
|
|
1116
|
-
You can also use your own forks, and work in progress branches by specifying them:
|
|
1117
|
-
|
|
1118
|
-
```
|
|
1119
|
-
npm install github:myUsername/parse-server#my-awesome-feature
|
|
1120
|
-
```
|
|
1123
|
+
Starting Parse Server 3.0, Parse Server uses the Parse JavaScript SDK 2.0. In short, the Parse JavaScript SDK 2.0 removes the backbone style callbacks as well as the `Parse.Promise` object in favor of native promises. All the Cloud Code interfaces also have been updated to reflect those changes, and all backbone style response objects are removed and replaced by promise style resolution.
|
|
1121
1124
|
|
|
1122
|
-
|
|
1125
|
+
We have written up a [migration guide](3.0.0.md) to help you transition to the next major release.
|
|
1123
1126
|
|
|
1124
1127
|
# Contributing
|
|
1125
1128
|
|
|
1126
|
-
|
|
1129
|
+
Please see the [Contributing Guide](CONTRIBUTING.md).
|
|
1127
1130
|
|
|
1128
1131
|
# Contributors
|
|
1129
1132
|
|
|
@@ -1179,3 +1182,6 @@ As of April 5, 2017, Parse, LLC has transferred this code to the parse-community
|
|
|
1179
1182
|
[license-svg]: https://img.shields.io/badge/license-BSD-lightgrey.svg
|
|
1180
1183
|
[license-link]: LICENSE
|
|
1181
1184
|
[open-collective-link]: https://opencollective.com/parse-server
|
|
1185
|
+
[log_release]: https://github.com/parse-community/parse-server/blob/release/changelogs/CHANGELOG_release.md
|
|
1186
|
+
[log_beta]: https://github.com/parse-community/parse-server/blob/beta/changelogs/CHANGELOG_beta.md
|
|
1187
|
+
[log_alpha]: https://github.com/parse-community/parse-server/blob/alpha/changelogs/CHANGELOG_alpha.md
|
|
@@ -20,21 +20,20 @@ const crypto = require('crypto');
|
|
|
20
20
|
|
|
21
21
|
const https = require('https');
|
|
22
22
|
|
|
23
|
+
const {
|
|
24
|
+
pki
|
|
25
|
+
} = require('node-forge');
|
|
26
|
+
|
|
27
|
+
const ca = {
|
|
28
|
+
cert: null,
|
|
29
|
+
url: null
|
|
30
|
+
};
|
|
23
31
|
const cache = {}; // (publicKey -> cert) cache
|
|
24
32
|
|
|
25
33
|
function verifyPublicKeyUrl(publicKeyUrl) {
|
|
26
34
|
try {
|
|
27
|
-
const
|
|
28
|
-
|
|
29
|
-
if (parsedUrl.protocol !== 'https:') {
|
|
30
|
-
return false;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
const hostnameParts = parsedUrl.hostname.split('.');
|
|
34
|
-
const length = hostnameParts.length;
|
|
35
|
-
const domainParts = hostnameParts.slice(length - 2, length);
|
|
36
|
-
const domain = domainParts.join('.');
|
|
37
|
-
return domain === 'apple.com';
|
|
35
|
+
const regex = /^https:\/\/(?:[-_A-Za-z0-9]+\.){0,}apple\.com\/.*\.cer$/;
|
|
36
|
+
return regex.test(publicKeyUrl);
|
|
38
37
|
} catch (error) {
|
|
39
38
|
return false;
|
|
40
39
|
}
|
|
@@ -48,7 +47,7 @@ function convertX509CertToPEM(X509Cert) {
|
|
|
48
47
|
return pemPreFix + certBody + pemPostFix;
|
|
49
48
|
}
|
|
50
49
|
|
|
51
|
-
function getAppleCertificate(publicKeyUrl) {
|
|
50
|
+
async function getAppleCertificate(publicKeyUrl) {
|
|
52
51
|
if (!verifyPublicKeyUrl(publicKeyUrl)) {
|
|
53
52
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
54
53
|
}
|
|
@@ -57,28 +56,66 @@ function getAppleCertificate(publicKeyUrl) {
|
|
|
57
56
|
return cache[publicKeyUrl];
|
|
58
57
|
}
|
|
59
58
|
|
|
59
|
+
const url = new URL(publicKeyUrl);
|
|
60
|
+
const headOptions = {
|
|
61
|
+
hostname: url.hostname,
|
|
62
|
+
path: url.pathname,
|
|
63
|
+
method: 'HEAD'
|
|
64
|
+
};
|
|
65
|
+
const cert_headers = await new Promise((resolve, reject) => https.get(headOptions, res => resolve(res.headers)).on('error', reject));
|
|
66
|
+
const validContentTypes = ['application/x-x509-ca-cert', 'application/pkix-cert'];
|
|
67
|
+
|
|
68
|
+
if (!validContentTypes.includes(cert_headers['content-type']) || cert_headers['content-length'] == null || cert_headers['content-length'] > 10000) {
|
|
69
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
const {
|
|
73
|
+
certificate,
|
|
74
|
+
headers
|
|
75
|
+
} = await getCertificate(publicKeyUrl);
|
|
76
|
+
|
|
77
|
+
if (headers['cache-control']) {
|
|
78
|
+
const expire = headers['cache-control'].match(/max-age=([0-9]+)/);
|
|
79
|
+
|
|
80
|
+
if (expire) {
|
|
81
|
+
cache[publicKeyUrl] = certificate; // we'll expire the cache entry later, as per max-age
|
|
82
|
+
|
|
83
|
+
setTimeout(() => {
|
|
84
|
+
delete cache[publicKeyUrl];
|
|
85
|
+
}, parseInt(expire[1], 10) * 1000);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
return verifyPublicKeyIssuer(certificate, publicKeyUrl);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function getCertificate(url, buffer) {
|
|
60
93
|
return new Promise((resolve, reject) => {
|
|
61
|
-
https.get(
|
|
62
|
-
|
|
94
|
+
https.get(url, res => {
|
|
95
|
+
const data = [];
|
|
63
96
|
res.on('data', chunk => {
|
|
64
|
-
data
|
|
97
|
+
data.push(chunk);
|
|
65
98
|
});
|
|
66
99
|
res.on('end', () => {
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
100
|
+
if (buffer) {
|
|
101
|
+
resolve({
|
|
102
|
+
certificate: Buffer.concat(data),
|
|
103
|
+
headers: res.headers
|
|
104
|
+
});
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
71
107
|
|
|
72
|
-
|
|
73
|
-
cache[publicKeyUrl] = cert; // we'll expire the cache entry later, as per max-age
|
|
108
|
+
let cert = '';
|
|
74
109
|
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
}, parseInt(expire[1], 10) * 1000);
|
|
78
|
-
}
|
|
110
|
+
for (const chunk of data) {
|
|
111
|
+
cert += chunk.toString('base64');
|
|
79
112
|
}
|
|
80
113
|
|
|
81
|
-
|
|
114
|
+
const certificate = convertX509CertToPEM(cert);
|
|
115
|
+
resolve({
|
|
116
|
+
certificate,
|
|
117
|
+
headers: res.headers
|
|
118
|
+
});
|
|
82
119
|
});
|
|
83
120
|
}).on('error', reject);
|
|
84
121
|
});
|
|
@@ -103,6 +140,24 @@ function verifySignature(publicKey, authData) {
|
|
|
103
140
|
if (!verifier.verify(publicKey, authData.signature, 'base64')) {
|
|
104
141
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - invalid signature');
|
|
105
142
|
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
function verifyPublicKeyIssuer(cert, publicKeyUrl) {
|
|
146
|
+
const publicKeyCert = pki.certificateFromPem(cert);
|
|
147
|
+
|
|
148
|
+
if (!ca.cert) {
|
|
149
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.');
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
try {
|
|
153
|
+
if (!ca.cert.verify(publicKeyCert)) {
|
|
154
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
155
|
+
}
|
|
156
|
+
} catch (e) {
|
|
157
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
return cert;
|
|
106
161
|
} // Returns a promise that fulfills if this user id is valid.
|
|
107
162
|
|
|
108
163
|
|
|
@@ -117,12 +172,31 @@ async function validateAuthData(authData) {
|
|
|
117
172
|
} // Returns a promise that fulfills if this app id is valid.
|
|
118
173
|
|
|
119
174
|
|
|
120
|
-
function validateAppId() {
|
|
121
|
-
|
|
175
|
+
async function validateAppId(appIds, authData, options = {}) {
|
|
176
|
+
if (!options.rootCertificateUrl) {
|
|
177
|
+
options.rootCertificateUrl = 'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
if (ca.url === options.rootCertificateUrl) {
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
const {
|
|
185
|
+
certificate,
|
|
186
|
+
headers
|
|
187
|
+
} = await getCertificate(options.rootCertificateUrl, true);
|
|
188
|
+
|
|
189
|
+
if (headers['content-type'] !== 'application/x-pem-file' || headers['content-length'] == null || headers['content-length'] > 10000) {
|
|
190
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.');
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
ca.cert = pki.certificateFromPem(certificate);
|
|
194
|
+
ca.url = options.rootCertificateUrl;
|
|
122
195
|
}
|
|
123
196
|
|
|
124
197
|
module.exports = {
|
|
125
198
|
validateAppId,
|
|
126
|
-
validateAuthData
|
|
199
|
+
validateAuthData,
|
|
200
|
+
cache
|
|
127
201
|
};
|
|
128
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/Adapters/Auth/gcenter.js"],"names":["Parse","require","crypto","https","cache","verifyPublicKeyUrl","publicKeyUrl","parsedUrl","URL","protocol","hostnameParts","hostname","split","length","domainParts","slice","domain","join","error","convertX509CertToPEM","X509Cert","pemPreFix","pemPostFix","base64","certBody","match","RegExp","getAppleCertificate","Error","OBJECT_NOT_FOUND","Promise","resolve","reject","get","res","data","on","chunk","toString","cert","headers","expire","setTimeout","parseInt","convertTimestampToBigEndian","timestamp","buffer","Buffer","alloc","high","low","writeUInt32BE","verifySignature","publicKey","authData","verifier","createVerify","update","playerId","bundleId","salt","verify","signature","validateAuthData","id","validateAppId","module","exports"],"mappings":"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","sourcesContent":["/* Apple Game Center Auth\nhttps://developer.apple.com/documentation/gamekit/gklocalplayer/1515407-generateidentityverificationsign#discussion\n\nconst authData = {\n  publicKeyUrl: 'https://valid.apple.com/public/timeout.cer',\n  timestamp: 1460981421303,\n  signature: 'PoDwf39DCN464B49jJCU0d9Y0J',\n  salt: 'saltST==',\n  bundleId: 'com.valid.app'\n  id: 'playerId',\n};\n*/\n\nconst { Parse } = require('parse/node');\nconst crypto = require('crypto');\nconst https = require('https');\n\nconst cache = {}; // (publicKey -> cert) cache\n\nfunction verifyPublicKeyUrl(publicKeyUrl) {\n  try {\n    const parsedUrl = new URL(publicKeyUrl);\n    if (parsedUrl.protocol !== 'https:') {\n      return false;\n    }\n    const hostnameParts = parsedUrl.hostname.split('.');\n    const length = hostnameParts.length;\n    const domainParts = hostnameParts.slice(length - 2, length);\n    const domain = domainParts.join('.');\n    return domain === 'apple.com';\n  } catch (error) {\n    return false;\n  }\n}\n\nfunction convertX509CertToPEM(X509Cert) {\n  const pemPreFix = '-----BEGIN CERTIFICATE-----\\n';\n  const pemPostFix = '-----END CERTIFICATE-----';\n\n  const base64 = X509Cert;\n  const certBody = base64.match(new RegExp('.{0,64}', 'g')).join('\\n');\n\n  return pemPreFix + certBody + pemPostFix;\n}\n\nfunction getAppleCertificate(publicKeyUrl) {\n  if (!verifyPublicKeyUrl(publicKeyUrl)) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`\n    );\n  }\n  if (cache[publicKeyUrl]) {\n    return cache[publicKeyUrl];\n  }\n  return new Promise((resolve, reject) => {\n    https\n      .get(publicKeyUrl, res => {\n        let data = '';\n        res.on('data', chunk => {\n          data += chunk.toString('base64');\n        });\n        res.on('end', () => {\n          const cert = convertX509CertToPEM(data);\n          if (res.headers['cache-control']) {\n            var expire = res.headers['cache-control'].match(/max-age=([0-9]+)/);\n            if (expire) {\n              cache[publicKeyUrl] = cert;\n              // we'll expire the cache entry later, as per max-age\n              setTimeout(() => {\n                delete cache[publicKeyUrl];\n              }, parseInt(expire[1], 10) * 1000);\n            }\n          }\n          resolve(cert);\n        });\n      })\n      .on('error', reject);\n  });\n}\n\nfunction convertTimestampToBigEndian(timestamp) {\n  const buffer = Buffer.alloc(8);\n\n  const high = ~~(timestamp / 0xffffffff);\n  const low = timestamp % (0xffffffff + 0x1);\n\n  buffer.writeUInt32BE(parseInt(high, 10), 0);\n  buffer.writeUInt32BE(parseInt(low, 10), 4);\n\n  return buffer;\n}\n\nfunction verifySignature(publicKey, authData) {\n  const verifier = crypto.createVerify('sha256');\n  verifier.update(authData.playerId, 'utf8');\n  verifier.update(authData.bundleId, 'utf8');\n  verifier.update(convertTimestampToBigEndian(authData.timestamp));\n  verifier.update(authData.salt, 'base64');\n\n  if (!verifier.verify(publicKey, authData.signature, 'base64')) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - invalid signature');\n  }\n}\n\n// Returns a promise that fulfills if this user id is valid.\nasync function validateAuthData(authData) {\n  if (!authData.id) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - authData id missing');\n  }\n  authData.playerId = authData.id;\n  const publicKey = await getAppleCertificate(authData.publicKeyUrl);\n  return verifySignature(publicKey, authData);\n}\n\n// Returns a promise that fulfills if this app id is valid.\nfunction validateAppId() {\n  return Promise.resolve();\n}\n\nmodule.exports = {\n  validateAppId,\n  validateAuthData,\n};\n"]}
|
|
202
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/Adapters/Auth/gcenter.js"],"names":["Parse","require","crypto","https","pki","ca","cert","url","cache","verifyPublicKeyUrl","publicKeyUrl","regex","test","error","convertX509CertToPEM","X509Cert","pemPreFix","pemPostFix","base64","certBody","match","RegExp","join","getAppleCertificate","Error","OBJECT_NOT_FOUND","URL","headOptions","hostname","path","pathname","method","cert_headers","Promise","resolve","reject","get","res","headers","on","validContentTypes","includes","certificate","getCertificate","expire","setTimeout","parseInt","verifyPublicKeyIssuer","buffer","data","chunk","push","Buffer","concat","toString","convertTimestampToBigEndian","timestamp","alloc","high","low","writeUInt32BE","verifySignature","publicKey","authData","verifier","createVerify","update","playerId","bundleId","salt","verify","signature","publicKeyCert","certificateFromPem","e","validateAuthData","id","validateAppId","appIds","options","rootCertificateUrl","module","exports"],"mappings":"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","sourcesContent":["/* Apple Game Center Auth\nhttps://developer.apple.com/documentation/gamekit/gklocalplayer/1515407-generateidentityverificationsign#discussion\n\nconst authData = {\n  publicKeyUrl: 'https://valid.apple.com/public/timeout.cer',\n  timestamp: 1460981421303,\n  signature: 'PoDwf39DCN464B49jJCU0d9Y0J',\n  salt: 'saltST==',\n  bundleId: 'com.valid.app'\n  id: 'playerId',\n};\n*/\n\nconst { Parse } = require('parse/node');\nconst crypto = require('crypto');\nconst https = require('https');\nconst { pki } = require('node-forge');\nconst ca = { cert: null, url: null };\nconst cache = {}; // (publicKey -> cert) cache\n\nfunction verifyPublicKeyUrl(publicKeyUrl) {\n  try {\n    const regex = /^https:\\/\\/(?:[-_A-Za-z0-9]+\\.){0,}apple\\.com\\/.*\\.cer$/;\n    return regex.test(publicKeyUrl);\n  } catch (error) {\n    return false;\n  }\n}\n\nfunction convertX509CertToPEM(X509Cert) {\n  const pemPreFix = '-----BEGIN CERTIFICATE-----\\n';\n  const pemPostFix = '-----END CERTIFICATE-----';\n\n  const base64 = X509Cert;\n  const certBody = base64.match(new RegExp('.{0,64}', 'g')).join('\\n');\n\n  return pemPreFix + certBody + pemPostFix;\n}\n\nasync function getAppleCertificate(publicKeyUrl) {\n  if (!verifyPublicKeyUrl(publicKeyUrl)) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`\n    );\n  }\n  if (cache[publicKeyUrl]) {\n    return cache[publicKeyUrl];\n  }\n  const url = new URL(publicKeyUrl);\n  const headOptions = {\n    hostname: url.hostname,\n    path: url.pathname,\n    method: 'HEAD',\n  };\n  const cert_headers = await new Promise((resolve, reject) =>\n    https.get(headOptions, res => resolve(res.headers)).on('error', reject)\n  );\n  const validContentTypes = ['application/x-x509-ca-cert', 'application/pkix-cert'];\n  if (\n    !validContentTypes.includes(cert_headers['content-type']) ||\n    cert_headers['content-length'] == null ||\n    cert_headers['content-length'] > 10000\n  ) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`\n    );\n  }\n  const { certificate, headers } = await getCertificate(publicKeyUrl);\n  if (headers['cache-control']) {\n    const expire = headers['cache-control'].match(/max-age=([0-9]+)/);\n    if (expire) {\n      cache[publicKeyUrl] = certificate;\n      // we'll expire the cache entry later, as per max-age\n      setTimeout(() => {\n        delete cache[publicKeyUrl];\n      }, parseInt(expire[1], 10) * 1000);\n    }\n  }\n  return verifyPublicKeyIssuer(certificate, publicKeyUrl);\n}\n\nfunction getCertificate(url, buffer) {\n  return new Promise((resolve, reject) => {\n    https\n      .get(url, res => {\n        const data = [];\n        res.on('data', chunk => {\n          data.push(chunk);\n        });\n        res.on('end', () => {\n          if (buffer) {\n            resolve({ certificate: Buffer.concat(data), headers: res.headers });\n            return;\n          }\n          let cert = '';\n          for (const chunk of data) {\n            cert += chunk.toString('base64');\n          }\n          const certificate = convertX509CertToPEM(cert);\n          resolve({ certificate, headers: res.headers });\n        });\n      })\n      .on('error', reject);\n  });\n}\n\nfunction convertTimestampToBigEndian(timestamp) {\n  const buffer = Buffer.alloc(8);\n\n  const high = ~~(timestamp / 0xffffffff);\n  const low = timestamp % (0xffffffff + 0x1);\n\n  buffer.writeUInt32BE(parseInt(high, 10), 0);\n  buffer.writeUInt32BE(parseInt(low, 10), 4);\n\n  return buffer;\n}\n\nfunction verifySignature(publicKey, authData) {\n  const verifier = crypto.createVerify('sha256');\n  verifier.update(authData.playerId, 'utf8');\n  verifier.update(authData.bundleId, 'utf8');\n  verifier.update(convertTimestampToBigEndian(authData.timestamp));\n  verifier.update(authData.salt, 'base64');\n\n  if (!verifier.verify(publicKey, authData.signature, 'base64')) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - invalid signature');\n  }\n}\n\nfunction verifyPublicKeyIssuer(cert, publicKeyUrl) {\n  const publicKeyCert = pki.certificateFromPem(cert);\n  if (!ca.cert) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.'\n    );\n  }\n  try {\n    if (!ca.cert.verify(publicKeyCert)) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`\n      );\n    }\n  } catch (e) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`\n    );\n  }\n  return cert;\n}\n\n// Returns a promise that fulfills if this user id is valid.\nasync function validateAuthData(authData) {\n  if (!authData.id) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - authData id missing');\n  }\n  authData.playerId = authData.id;\n  const publicKey = await getAppleCertificate(authData.publicKeyUrl);\n  return verifySignature(publicKey, authData);\n}\n\n// Returns a promise that fulfills if this app id is valid.\nasync function validateAppId(appIds, authData, options = {}) {\n  if (!options.rootCertificateUrl) {\n    options.rootCertificateUrl =\n      'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';\n  }\n  if (ca.url === options.rootCertificateUrl) {\n    return;\n  }\n  const { certificate, headers } = await getCertificate(options.rootCertificateUrl, true);\n  if (\n    headers['content-type'] !== 'application/x-pem-file' ||\n    headers['content-length'] == null ||\n    headers['content-length'] > 10000\n  ) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.'\n    );\n  }\n  ca.cert = pki.certificateFromPem(certificate);\n  ca.url = options.rootCertificateUrl;\n}\n\nmodule.exports = {\n  validateAppId,\n  validateAuthData,\n  cache,\n};\n"]}
|
|
@@ -18,7 +18,7 @@ class LRUCache {
|
|
|
18
18
|
}) {
|
|
19
19
|
this.cache = new _lruCache.default({
|
|
20
20
|
max: maxSize,
|
|
21
|
-
|
|
21
|
+
ttl
|
|
22
22
|
});
|
|
23
23
|
}
|
|
24
24
|
|
|
@@ -43,4 +43,4 @@ class LRUCache {
|
|
|
43
43
|
exports.LRUCache = LRUCache;
|
|
44
44
|
var _default = LRUCache;
|
|
45
45
|
exports.default = _default;
|
|
46
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
46
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BZGFwdGVycy9DYWNoZS9MUlVDYWNoZS5qcyJdLCJuYW1lcyI6WyJMUlVDYWNoZSIsImNvbnN0cnVjdG9yIiwidHRsIiwiZGVmYXVsdHMiLCJjYWNoZVRUTCIsIm1heFNpemUiLCJjYWNoZU1heFNpemUiLCJjYWNoZSIsIkxSVSIsIm1heCIsImdldCIsImtleSIsInB1dCIsInZhbHVlIiwic2V0IiwiZGVsIiwiY2xlYXIiLCJyZXNldCJdLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBOztBQUNBOzs7O0FBRU8sTUFBTUEsUUFBTixDQUFlO0FBQ3BCQyxFQUFBQSxXQUFXLENBQUM7QUFBRUMsSUFBQUEsR0FBRyxHQUFHQyxrQkFBU0MsUUFBakI7QUFBMkJDLElBQUFBLE9BQU8sR0FBR0Ysa0JBQVNHO0FBQTlDLEdBQUQsRUFBK0Q7QUFDeEUsU0FBS0MsS0FBTCxHQUFhLElBQUlDLGlCQUFKLENBQVE7QUFDbkJDLE1BQUFBLEdBQUcsRUFBRUosT0FEYztBQUVuQkgsTUFBQUE7QUFGbUIsS0FBUixDQUFiO0FBSUQ7O0FBRURRLEVBQUFBLEdBQUcsQ0FBQ0MsR0FBRCxFQUFNO0FBQ1AsV0FBTyxLQUFLSixLQUFMLENBQVdHLEdBQVgsQ0FBZUMsR0FBZixLQUF1QixJQUE5QjtBQUNEOztBQUVEQyxFQUFBQSxHQUFHLENBQUNELEdBQUQsRUFBTUUsS0FBTixFQUFhWCxHQUFHLEdBQUcsS0FBS0EsR0FBeEIsRUFBNkI7QUFDOUIsU0FBS0ssS0FBTCxDQUFXTyxHQUFYLENBQWVILEdBQWYsRUFBb0JFLEtBQXBCLEVBQTJCWCxHQUEzQjtBQUNEOztBQUVEYSxFQUFBQSxHQUFHLENBQUNKLEdBQUQsRUFBTTtBQUNQLFNBQUtKLEtBQUwsQ0FBV1EsR0FBWCxDQUFlSixHQUFmO0FBQ0Q7O0FBRURLLEVBQUFBLEtBQUssR0FBRztBQUNOLFNBQUtULEtBQUwsQ0FBV1UsS0FBWDtBQUNEOztBQXRCbUI7OztlQXlCUGpCLFEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgTFJVIGZyb20gJ2xydS1jYWNoZSc7XG5pbXBvcnQgZGVmYXVsdHMgZnJvbSAnLi4vLi4vZGVmYXVsdHMnO1xuXG5leHBvcnQgY2xhc3MgTFJVQ2FjaGUge1xuICBjb25zdHJ1Y3Rvcih7IHR0bCA9IGRlZmF1bHRzLmNhY2hlVFRMLCBtYXhTaXplID0gZGVmYXVsdHMuY2FjaGVNYXhTaXplIH0pIHtcbiAgICB0aGlzLmNhY2hlID0gbmV3IExSVSh7XG4gICAgICBtYXg6IG1heFNpemUsXG4gICAgICB0dGwsXG4gICAgfSk7XG4gIH1cblxuICBnZXQoa2V5KSB7XG4gICAgcmV0dXJuIHRoaXMuY2FjaGUuZ2V0KGtleSkgfHwgbnVsbDtcbiAgfVxuXG4gIHB1dChrZXksIHZhbHVlLCB0dGwgPSB0aGlzLnR0bCkge1xuICAgIHRoaXMuY2FjaGUuc2V0KGtleSwgdmFsdWUsIHR0bCk7XG4gIH1cblxuICBkZWwoa2V5KSB7XG4gICAgdGhpcy5jYWNoZS5kZWwoa2V5KTtcbiAgfVxuXG4gIGNsZWFyKCkge1xuICAgIHRoaXMuY2FjaGUucmVzZXQoKTtcbiAgfVxufVxuXG5leHBvcnQgZGVmYXVsdCBMUlVDYWNoZTtcbiJdfQ==
|