parse-server 4.10.8 → 4.10.11
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/lib/Adapters/Auth/gcenter.js +105 -29
- package/package.json +2 -2
|
@@ -20,22 +20,23 @@ const crypto = require('crypto');
|
|
|
20
20
|
|
|
21
21
|
const https = require('https');
|
|
22
22
|
|
|
23
|
-
const
|
|
23
|
+
const {
|
|
24
|
+
pki
|
|
25
|
+
} = require('node-forge');
|
|
24
26
|
|
|
27
|
+
const ca = {
|
|
28
|
+
cert: null,
|
|
29
|
+
url: null
|
|
30
|
+
};
|
|
25
31
|
const cache = {}; // (publicKey -> cert) cache
|
|
26
32
|
|
|
27
33
|
function verifyPublicKeyUrl(publicKeyUrl) {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
34
|
+
try {
|
|
35
|
+
const regex = /^https:\/\/(?:[-_A-Za-z0-9]+\.){0,}apple\.com\/.*\.cer$/;
|
|
36
|
+
return regex.test(publicKeyUrl);
|
|
37
|
+
} catch (error) {
|
|
31
38
|
return false;
|
|
32
39
|
}
|
|
33
|
-
|
|
34
|
-
const hostnameParts = parsedUrl.hostname.split('.');
|
|
35
|
-
const length = hostnameParts.length;
|
|
36
|
-
const domainParts = hostnameParts.slice(length - 2, length);
|
|
37
|
-
const domain = domainParts.join('.');
|
|
38
|
-
return domain === 'apple.com';
|
|
39
40
|
}
|
|
40
41
|
|
|
41
42
|
function convertX509CertToPEM(X509Cert) {
|
|
@@ -46,7 +47,7 @@ function convertX509CertToPEM(X509Cert) {
|
|
|
46
47
|
return pemPreFix + certBody + pemPostFix;
|
|
47
48
|
}
|
|
48
49
|
|
|
49
|
-
function getAppleCertificate(publicKeyUrl) {
|
|
50
|
+
async function getAppleCertificate(publicKeyUrl) {
|
|
50
51
|
if (!verifyPublicKeyUrl(publicKeyUrl)) {
|
|
51
52
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
|
|
52
53
|
}
|
|
@@ -55,28 +56,66 @@ function getAppleCertificate(publicKeyUrl) {
|
|
|
55
56
|
return cache[publicKeyUrl];
|
|
56
57
|
}
|
|
57
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) {
|
|
58
93
|
return new Promise((resolve, reject) => {
|
|
59
|
-
https.get(
|
|
60
|
-
|
|
94
|
+
https.get(url, res => {
|
|
95
|
+
const data = [];
|
|
61
96
|
res.on('data', chunk => {
|
|
62
|
-
data
|
|
97
|
+
data.push(chunk);
|
|
63
98
|
});
|
|
64
99
|
res.on('end', () => {
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
100
|
+
if (buffer) {
|
|
101
|
+
resolve({
|
|
102
|
+
certificate: Buffer.concat(data),
|
|
103
|
+
headers: res.headers
|
|
104
|
+
});
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
69
107
|
|
|
70
|
-
|
|
71
|
-
cache[publicKeyUrl] = cert; // we'll expire the cache entry later, as per max-age
|
|
108
|
+
let cert = '';
|
|
72
109
|
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
}, parseInt(expire[1], 10) * 1000);
|
|
76
|
-
}
|
|
110
|
+
for (const chunk of data) {
|
|
111
|
+
cert += chunk.toString('base64');
|
|
77
112
|
}
|
|
78
113
|
|
|
79
|
-
|
|
114
|
+
const certificate = convertX509CertToPEM(cert);
|
|
115
|
+
resolve({
|
|
116
|
+
certificate,
|
|
117
|
+
headers: res.headers
|
|
118
|
+
});
|
|
80
119
|
});
|
|
81
120
|
}).on('error', reject);
|
|
82
121
|
});
|
|
@@ -101,6 +140,24 @@ function verifySignature(publicKey, authData) {
|
|
|
101
140
|
if (!verifier.verify(publicKey, authData.signature, 'base64')) {
|
|
102
141
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - invalid signature');
|
|
103
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;
|
|
104
161
|
} // Returns a promise that fulfills if this user id is valid.
|
|
105
162
|
|
|
106
163
|
|
|
@@ -115,12 +172,31 @@ async function validateAuthData(authData) {
|
|
|
115
172
|
} // Returns a promise that fulfills if this app id is valid.
|
|
116
173
|
|
|
117
174
|
|
|
118
|
-
function validateAppId() {
|
|
119
|
-
|
|
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;
|
|
120
195
|
}
|
|
121
196
|
|
|
122
197
|
module.exports = {
|
|
123
198
|
validateAppId,
|
|
124
|
-
validateAuthData
|
|
199
|
+
validateAuthData,
|
|
200
|
+
cache
|
|
125
201
|
};
|
|
126
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/Adapters/Auth/gcenter.js"],"names":["Parse","require","crypto","https","url","cache","verifyPublicKeyUrl","publicKeyUrl","parsedUrl","parse","protocol","hostnameParts","hostname","split","length","domainParts","slice","domain","join","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');\nconst url = require('url');\n\nconst cache = {}; // (publicKey -> cert) cache\n\nfunction verifyPublicKeyUrl(publicKeyUrl) {\n  const parsedUrl = url.parse(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}\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"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "parse-server",
|
|
3
|
-
"version": "4.10.
|
|
3
|
+
"version": "4.10.11",
|
|
4
4
|
"description": "An express module providing a Parse-compatible API server",
|
|
5
5
|
"main": "lib/index.js",
|
|
6
6
|
"repository": {
|
|
@@ -24,7 +24,7 @@
|
|
|
24
24
|
"@graphql-tools/stitch": "6.2.4",
|
|
25
25
|
"@graphql-tools/utils": "6.2.4",
|
|
26
26
|
"@parse/fs-files-adapter": "1.2.0",
|
|
27
|
-
"@parse/push-adapter": "
|
|
27
|
+
"@parse/push-adapter": "4.1.2",
|
|
28
28
|
"@parse/s3-files-adapter": "1.6.2",
|
|
29
29
|
"@parse/simple-mailgun-adapter": "1.1.0",
|
|
30
30
|
"apollo-server-express": "2.19.0",
|