parse-server 7.5.1 → 7.5.3

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 (39) hide show
  1. package/lib/Adapters/Auth/AuthAdapter.js +16 -9
  2. package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
  3. package/lib/Adapters/Auth/apple.js +45 -1
  4. package/lib/Adapters/Auth/facebook.js +61 -1
  5. package/lib/Adapters/Auth/gcenter.js +201 -157
  6. package/lib/Adapters/Auth/github.js +119 -31
  7. package/lib/Adapters/Auth/google.js +45 -1
  8. package/lib/Adapters/Auth/gpgames.js +120 -27
  9. package/lib/Adapters/Auth/index.js +33 -33
  10. package/lib/Adapters/Auth/instagram.js +114 -24
  11. package/lib/Adapters/Auth/janraincapture.js +45 -1
  12. package/lib/Adapters/Auth/janrainengage.js +12 -2
  13. package/lib/Adapters/Auth/keycloak.js +68 -35
  14. package/lib/Adapters/Auth/ldap.js +75 -1
  15. package/lib/Adapters/Auth/line.js +119 -32
  16. package/lib/Adapters/Auth/linkedin.js +111 -35
  17. package/lib/Adapters/Auth/meetup.js +16 -8
  18. package/lib/Adapters/Auth/mfa.js +80 -2
  19. package/lib/Adapters/Auth/microsoft.js +105 -30
  20. package/lib/Adapters/Auth/oauth2.js +100 -110
  21. package/lib/Adapters/Auth/phantauth.js +16 -8
  22. package/lib/Adapters/Auth/qq.js +107 -36
  23. package/lib/Adapters/Auth/spotify.js +108 -39
  24. package/lib/Adapters/Auth/twitter.js +187 -40
  25. package/lib/Adapters/Auth/vkontakte.js +21 -13
  26. package/lib/Adapters/Auth/wechat.js +105 -25
  27. package/lib/Adapters/Auth/weibo.js +135 -37
  28. package/lib/Auth.js +27 -17
  29. package/lib/Config.js +14 -1
  30. package/lib/Deprecator/Deprecations.js +4 -1
  31. package/lib/GraphQL/ParseGraphQLServer.js +33 -3
  32. package/lib/Options/Definitions.js +13 -1
  33. package/lib/Options/docs.js +3 -1
  34. package/lib/Options/index.js +1 -1
  35. package/lib/RestWrite.js +4 -5
  36. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +19 -1
  37. package/lib/cli/parse-server.js +1 -1
  38. package/lib/middlewares.js +1 -1
  39. package/package.json +9 -9
@@ -1,173 +1,217 @@
1
1
  "use strict";
2
2
 
3
- /* Apple Game Center Auth
4
- https://developer.apple.com/documentation/gamekit/gklocalplayer/1515407-generateidentityverificationsign#discussion
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _crypto = _interopRequireDefault(require("crypto"));
8
+ var _nodeForge = require("node-forge");
9
+ var _AuthAdapter = _interopRequireDefault(require("./AuthAdapter"));
10
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
11
+ /**
12
+ * Parse Server authentication adapter for Apple Game Center.
13
+ *
14
+ * @class AppleGameCenterAdapter
15
+ * @param {Object} options - Configuration options for the adapter.
16
+ * @param {string} options.bundleId - Your Apple Game Center bundle ID. Required for secure authentication.
17
+ * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).
18
+ *
19
+ * @param {Object} authData - The authentication data provided by the client.
20
+ * @param {string} authData.id - The user ID obtained from Apple Game Center.
21
+ * @param {string} authData.publicKeyUrl - The public key URL obtained from Apple Game Center.
22
+ * @param {string} authData.timestamp - The timestamp obtained from Apple Game Center.
23
+ * @param {string} authData.signature - The signature obtained from Apple Game Center.
24
+ * @param {string} authData.salt - The salt obtained from Apple Game Center.
25
+ * @param {string} [authData.bundleId] - **[DEPRECATED]** The bundle ID obtained from Apple Game Center (required for insecure authentication).
26
+ *
27
+ * @description
28
+ * ## Parse Server Configuration
29
+ * The following `authData` fields are required:
30
+ * `id`, `publicKeyUrl`, `timestamp`, `signature`, and `salt`. These fields are validated against the configured `bundleId` for additional security.
31
+ *
32
+ * To configure Parse Server for Apple Game Center authentication, use the following structure:
33
+ * ```json
34
+ * {
35
+ * "auth": {
36
+ * "gcenter": {
37
+ * "bundleId": "com.valid.app"
38
+ * }
39
+ * }
40
+ * ```
41
+ *
42
+ * ## Insecure Authentication (Not Recommended)
43
+ * The following `authData` fields are required for insecure authentication:
44
+ * `id`, `publicKeyUrl`, `timestamp`, `signature`, `salt`, and `bundleId` (**[DEPRECATED]**). This flow is insecure and poses potential security risks.
45
+ *
46
+ * To configure Parse Server for insecure authentication, use the following structure:
47
+ * ```json
48
+ * {
49
+ * "auth": {
50
+ * "gcenter": {
51
+ * "enableInsecureAuth": true
52
+ * }
53
+ * }
54
+ * ```
55
+ *
56
+ * ### Deprecation Notice
57
+ * The `enableInsecureAuth` option and `authData.bundleId` parameter are deprecated and may be removed in future releases. Use secure authentication with the `bundleId` configured in the `options` object instead.
58
+ *
59
+ *
60
+ * @example <caption>Secure Authentication Example</caption>
61
+ * // Example authData for secure authentication:
62
+ * const authData = {
63
+ * gcenter: {
64
+ * id: "1234567",
65
+ * publicKeyUrl: "https://valid.apple.com/public/timeout.cer",
66
+ * timestamp: 1460981421303,
67
+ * salt: "saltST==",
68
+ * signature: "PoDwf39DCN464B49jJCU0d9Y0J"
69
+ * }
70
+ * };
71
+ *
72
+ * @example <caption>Insecure Authentication Example (Not Recommended)</caption>
73
+ * // Example authData for insecure authentication:
74
+ * const authData = {
75
+ * gcenter: {
76
+ * id: "1234567",
77
+ * publicKeyUrl: "https://valid.apple.com/public/timeout.cer",
78
+ * timestamp: 1460981421303,
79
+ * salt: "saltST==",
80
+ * signature: "PoDwf39DCN464B49jJCU0d9Y0J",
81
+ * bundleId: "com.valid.app" // Deprecated.
82
+ * }
83
+ * };
84
+ *
85
+ * @see {@link https://developer.apple.com/documentation/gamekit/gklocalplayer/3516283-fetchitems Apple Game Center Documentation}
86
+ */
87
+ /* global BigInt */
5
88
 
6
- const authData = {
7
- publicKeyUrl: 'https://valid.apple.com/public/timeout.cer',
8
- timestamp: 1460981421303,
9
- signature: 'PoDwf39DCN464B49jJCU0d9Y0J',
10
- salt: 'saltST==',
11
- bundleId: 'com.valid.app'
12
- id: 'playerId',
13
- };
14
- */
15
-
16
- const {
17
- Parse
18
- } = require('parse/node');
19
- const crypto = require('crypto');
20
- const https = require('https');
21
- const {
22
- pki
23
- } = require('node-forge');
24
- const ca = {
25
- cert: null,
26
- url: null
27
- };
28
- const cache = {}; // (publicKey -> cert) cache
29
-
30
- function verifyPublicKeyUrl(publicKeyUrl) {
31
- try {
32
- const regex = /^https:\/\/(?:[-_A-Za-z0-9]+\.){0,}apple\.com\/.*\.cer$/;
33
- return regex.test(publicKeyUrl);
34
- } catch (error) {
35
- return false;
89
+ class GameCenterAuth extends _AuthAdapter.default {
90
+ constructor() {
91
+ super();
92
+ this.ca = {
93
+ cert: null,
94
+ url: null
95
+ };
96
+ this.cache = {};
97
+ this.bundleId = '';
36
98
  }
37
- }
38
- function convertX509CertToPEM(X509Cert) {
39
- const pemPreFix = '-----BEGIN CERTIFICATE-----\n';
40
- const pemPostFix = '-----END CERTIFICATE-----';
41
- const base64 = X509Cert;
42
- const certBody = base64.match(new RegExp('.{0,64}', 'g')).join('\n');
43
- return pemPreFix + certBody + pemPostFix;
44
- }
45
- async function getAppleCertificate(publicKeyUrl) {
46
- if (!verifyPublicKeyUrl(publicKeyUrl)) {
47
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
48
- }
49
- if (cache[publicKeyUrl]) {
50
- return cache[publicKeyUrl];
51
- }
52
- const url = new URL(publicKeyUrl);
53
- const headOptions = {
54
- hostname: url.hostname,
55
- path: url.pathname,
56
- method: 'HEAD'
57
- };
58
- const cert_headers = await new Promise((resolve, reject) => https.get(headOptions, res => resolve(res.headers)).on('error', reject));
59
- const validContentTypes = ['application/x-x509-ca-cert', 'application/pkix-cert'];
60
- if (!validContentTypes.includes(cert_headers['content-type']) || cert_headers['content-length'] == null || cert_headers['content-length'] > 10000) {
61
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
99
+ validateOptions(options) {
100
+ if (!options) {
101
+ throw new Error('Game center auth options are required.');
102
+ }
103
+ if (!this.loadingPromise) {
104
+ this.loadingPromise = this.loadCertificate(options);
105
+ }
106
+ this.enableInsecureAuth = options.enableInsecureAuth;
107
+ this.bundleId = options.bundleId;
108
+ if (!this.enableInsecureAuth && !this.bundleId) {
109
+ throw new Error('bundleId is required for secure auth.');
110
+ }
62
111
  }
63
- const {
64
- certificate,
65
- headers
66
- } = await getCertificate(publicKeyUrl);
67
- if (headers['cache-control']) {
68
- const expire = headers['cache-control'].match(/max-age=([0-9]+)/);
69
- if (expire) {
70
- cache[publicKeyUrl] = certificate;
71
- // we'll expire the cache entry later, as per max-age
72
- setTimeout(() => {
73
- delete cache[publicKeyUrl];
74
- }, parseInt(expire[1], 10) * 1000);
112
+ async loadCertificate(options) {
113
+ const rootCertificateUrl = options.rootCertificateUrl || 'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';
114
+ if (this.ca.url === rootCertificateUrl) {
115
+ return rootCertificateUrl;
116
+ }
117
+ const {
118
+ certificate,
119
+ headers
120
+ } = await this.fetchCertificate(rootCertificateUrl);
121
+ if (headers.get('content-type') !== 'application/x-pem-file' || !headers.get('content-length') || parseInt(headers.get('content-length'), 10) > 10000) {
122
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid rootCertificateURL.');
75
123
  }
124
+ this.ca.cert = _nodeForge.pki.certificateFromPem(certificate);
125
+ this.ca.url = rootCertificateUrl;
126
+ return rootCertificateUrl;
76
127
  }
77
- return verifyPublicKeyIssuer(certificate, publicKeyUrl);
78
- }
79
- function getCertificate(url, buffer) {
80
- return new Promise((resolve, reject) => {
81
- https.get(url, res => {
82
- const data = [];
83
- res.on('data', chunk => {
84
- data.push(chunk);
85
- });
86
- res.on('end', () => {
87
- if (buffer) {
88
- resolve({
89
- certificate: Buffer.concat(data),
90
- headers: res.headers
91
- });
92
- return;
93
- }
94
- let cert = '';
95
- for (const chunk of data) {
96
- cert += chunk.toString('base64');
97
- }
98
- const certificate = convertX509CertToPEM(cert);
99
- resolve({
100
- certificate,
101
- headers: res.headers
102
- });
103
- });
104
- }).on('error', reject);
105
- });
106
- }
107
- function convertTimestampToBigEndian(timestamp) {
108
- const buffer = Buffer.alloc(8);
109
- const high = ~~(timestamp / 0xffffffff);
110
- const low = timestamp % (0xffffffff + 0x1);
111
- buffer.writeUInt32BE(parseInt(high, 10), 0);
112
- buffer.writeUInt32BE(parseInt(low, 10), 4);
113
- return buffer;
114
- }
115
- function verifySignature(publicKey, authData) {
116
- const verifier = crypto.createVerify('sha256');
117
- verifier.update(authData.playerId, 'utf8');
118
- verifier.update(authData.bundleId, 'utf8');
119
- verifier.update(convertTimestampToBigEndian(authData.timestamp));
120
- verifier.update(authData.salt, 'base64');
121
- if (!verifier.verify(publicKey, authData.signature, 'base64')) {
122
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - invalid signature');
128
+ verifyPublicKeyUrl(publicKeyUrl) {
129
+ const regex = /^https:\/\/(?:[-_A-Za-z0-9]+\.){0,}apple\.com\/.*\.cer$/;
130
+ return regex.test(publicKeyUrl);
123
131
  }
124
- }
125
- function verifyPublicKeyIssuer(cert, publicKeyUrl) {
126
- const publicKeyCert = pki.certificateFromPem(cert);
127
- if (!ca.cert) {
128
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.');
132
+ async fetchCertificate(url) {
133
+ const response = await fetch(url);
134
+ if (!response.ok) {
135
+ throw new Error(`Failed to fetch certificate: ${url}`);
136
+ }
137
+ const contentType = response.headers.get('content-type');
138
+ const isPem = contentType === null || contentType === void 0 ? void 0 : contentType.includes('application/x-pem-file');
139
+ if (isPem) {
140
+ const certificate = await response.text();
141
+ return {
142
+ certificate,
143
+ headers: response.headers
144
+ };
145
+ }
146
+ const data = await response.arrayBuffer();
147
+ const binaryData = Buffer.from(data);
148
+ const asn1Cert = _nodeForge.asn1.fromDer(binaryData.toString('binary'));
149
+ const forgeCert = _nodeForge.pki.certificateFromAsn1(asn1Cert);
150
+ const certificate = _nodeForge.pki.certificateToPem(forgeCert);
151
+ return {
152
+ certificate,
153
+ headers: response.headers
154
+ };
129
155
  }
130
- try {
131
- if (!ca.cert.verify(publicKeyCert)) {
132
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
156
+ async getAppleCertificate(publicKeyUrl) {
157
+ if (!this.verifyPublicKeyUrl(publicKeyUrl)) {
158
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);
133
159
  }
134
- } catch (e) {
135
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
160
+ if (this.cache[publicKeyUrl]) {
161
+ return this.cache[publicKeyUrl];
162
+ }
163
+ const {
164
+ certificate,
165
+ headers
166
+ } = await this.fetchCertificate(publicKeyUrl);
167
+ const cacheControl = headers.get('cache-control');
168
+ const expire = cacheControl === null || cacheControl === void 0 ? void 0 : cacheControl.match(/max-age=([0-9]+)/);
169
+ this.verifyPublicKeyIssuer(certificate, publicKeyUrl);
170
+ if (expire) {
171
+ this.cache[publicKeyUrl] = certificate;
172
+ setTimeout(() => delete this.cache[publicKeyUrl], parseInt(expire[1], 10) * 1000);
173
+ }
174
+ return certificate;
136
175
  }
137
- return cert;
138
- }
139
-
140
- // Returns a promise that fulfills if this user id is valid.
141
- async function validateAuthData(authData) {
142
- if (!authData.id) {
143
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - authData id missing');
176
+ verifyPublicKeyIssuer(cert, publicKeyUrl) {
177
+ const publicKeyCert = _nodeForge.pki.certificateFromPem(cert);
178
+ if (!this.ca.cert) {
179
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Root certificate is invalid or missing.');
180
+ }
181
+ if (!this.ca.cert.verify(publicKeyCert)) {
182
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);
183
+ }
144
184
  }
145
- authData.playerId = authData.id;
146
- const publicKey = await getAppleCertificate(authData.publicKeyUrl);
147
- return verifySignature(publicKey, authData);
148
- }
149
-
150
- // Returns a promise that fulfills if this app id is valid.
151
- async function validateAppId(appIds, authData, options = {}) {
152
- if (!options.rootCertificateUrl) {
153
- options.rootCertificateUrl = 'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';
185
+ verifySignature(publicKey, authData) {
186
+ const bundleId = this.bundleId || this.enableInsecureAuth && authData.bundleId;
187
+ const verifier = _crypto.default.createVerify('sha256');
188
+ verifier.update(Buffer.from(authData.id, 'utf8'));
189
+ verifier.update(Buffer.from(bundleId, 'utf8'));
190
+ verifier.update(this.convertTimestampToBigEndian(authData.timestamp));
191
+ verifier.update(Buffer.from(authData.salt, 'base64'));
192
+ if (!verifier.verify(publicKey, authData.signature, 'base64')) {
193
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid signature.');
194
+ }
154
195
  }
155
- if (ca.url === options.rootCertificateUrl) {
156
- return;
196
+ async validateAuthData(authData) {
197
+ const requiredKeys = ['id', 'publicKeyUrl', 'timestamp', 'signature', 'salt'];
198
+ if (this.enableInsecureAuth) {
199
+ requiredKeys.push('bundleId');
200
+ }
201
+ for (const key of requiredKeys) {
202
+ if (!authData[key]) {
203
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `AuthData ${key} is missing.`);
204
+ }
205
+ }
206
+ await this.loadingPromise;
207
+ const publicKey = await this.getAppleCertificate(authData.publicKeyUrl);
208
+ this.verifySignature(publicKey, authData);
157
209
  }
158
- const {
159
- certificate,
160
- headers
161
- } = await getCertificate(options.rootCertificateUrl, true);
162
- if (headers['content-type'] !== 'application/x-pem-file' || headers['content-length'] == null || headers['content-length'] > 10000) {
163
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.');
210
+ convertTimestampToBigEndian(timestamp) {
211
+ const buffer = Buffer.alloc(8);
212
+ buffer.writeBigUInt64BE(BigInt(timestamp));
213
+ return buffer;
164
214
  }
165
- ca.cert = pki.certificateFromPem(certificate);
166
- ca.url = options.rootCertificateUrl;
167
215
  }
168
- module.exports = {
169
- validateAppId,
170
- validateAuthData,
171
- cache
172
- };
173
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"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"],"sources":["../../../src/Adapters/Auth/gcenter.js"],"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"],"mappings":"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","ignoreList":[]}
216
+ var _default = exports.default = new GameCenterAuth();
217
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_crypto","_interopRequireDefault","require","_nodeForge","_AuthAdapter","e","__esModule","default","GameCenterAuth","AuthAdapter","constructor","ca","cert","url","cache","bundleId","validateOptions","options","Error","loadingPromise","loadCertificate","enableInsecureAuth","rootCertificateUrl","certificate","headers","fetchCertificate","get","parseInt","Parse","OBJECT_NOT_FOUND","pki","certificateFromPem","verifyPublicKeyUrl","publicKeyUrl","regex","test","response","fetch","ok","contentType","isPem","includes","text","data","arrayBuffer","binaryData","Buffer","from","asn1Cert","asn1","fromDer","toString","forgeCert","certificateFromAsn1","certificateToPem","getAppleCertificate","cacheControl","expire","match","verifyPublicKeyIssuer","setTimeout","publicKeyCert","verify","verifySignature","publicKey","authData","verifier","crypto","createVerify","update","id","convertTimestampToBigEndian","timestamp","salt","signature","validateAuthData","requiredKeys","push","key","buffer","alloc","writeBigUInt64BE","BigInt","_default","exports"],"sources":["../../../src/Adapters/Auth/gcenter.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Apple Game Center.\n *\n * @class AppleGameCenterAdapter\n * @param {Object} options - Configuration options for the adapter.\n * @param {string} options.bundleId - Your Apple Game Center bundle ID. Required for secure authentication.\n * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {string} authData.id - The user ID obtained from Apple Game Center.\n * @param {string} authData.publicKeyUrl - The public key URL obtained from Apple Game Center.\n * @param {string} authData.timestamp - The timestamp obtained from Apple Game Center.\n * @param {string} authData.signature - The signature obtained from Apple Game Center.\n * @param {string} authData.salt - The salt obtained from Apple Game Center.\n * @param {string} [authData.bundleId] - **[DEPRECATED]** The bundle ID obtained from Apple Game Center (required for insecure authentication).\n *\n * @description\n * ## Parse Server Configuration\n * The following `authData` fields are required:\n * `id`, `publicKeyUrl`, `timestamp`, `signature`, and `salt`. These fields are validated against the configured `bundleId` for additional security.\n *\n * To configure Parse Server for Apple Game Center authentication, use the following structure:\n * ```json\n * {\n *  \"auth\": {\n *    \"gcenter\": {\n *     \"bundleId\": \"com.valid.app\"\n *  }\n * }\n * ```\n *\n * ## Insecure Authentication (Not Recommended)\n * The following `authData` fields are required for insecure authentication:\n * `id`, `publicKeyUrl`, `timestamp`, `signature`, `salt`, and `bundleId` (**[DEPRECATED]**). This flow is insecure and poses potential security risks.\n *\n * To configure Parse Server for insecure authentication, use the following structure:\n * ```json\n * {\n *   \"auth\": {\n *    \"gcenter\": {\n *      \"enableInsecureAuth\": true\n *   }\n * }\n * ```\n *\n * ### Deprecation Notice\n * The `enableInsecureAuth` option and `authData.bundleId` parameter are deprecated and may be removed in future releases. Use secure authentication with the `bundleId` configured in the `options` object instead.\n *\n *\n * @example <caption>Secure Authentication Example</caption>\n * // Example authData for secure authentication:\n * const authData = {\n *   gcenter: {\n *     id: \"1234567\",\n *     publicKeyUrl: \"https://valid.apple.com/public/timeout.cer\",\n *     timestamp: 1460981421303,\n *     salt: \"saltST==\",\n *     signature: \"PoDwf39DCN464B49jJCU0d9Y0J\"\n *   }\n * };\n *\n * @example <caption>Insecure Authentication Example (Not Recommended)</caption>\n * // Example authData for insecure authentication:\n * const authData = {\n *   gcenter: {\n *     id: \"1234567\",\n *     publicKeyUrl: \"https://valid.apple.com/public/timeout.cer\",\n *     timestamp: 1460981421303,\n *     salt: \"saltST==\",\n *     signature: \"PoDwf39DCN464B49jJCU0d9Y0J\",\n *     bundleId: \"com.valid.app\" // Deprecated.\n *   }\n * };\n *\n * @see {@link https://developer.apple.com/documentation/gamekit/gklocalplayer/3516283-fetchitems Apple Game Center Documentation}\n */\n/* global BigInt */\n\nimport crypto from 'crypto';\nimport { asn1, pki } from 'node-forge';\nimport AuthAdapter from './AuthAdapter';\nclass GameCenterAuth extends AuthAdapter {\n  constructor() {\n    super();\n    this.ca = { cert: null, url: null };\n    this.cache = {};\n    this.bundleId = '';\n  }\n\n  validateOptions(options) {\n    if (!options) {\n      throw new Error('Game center auth options are required.');\n    }\n\n    if (!this.loadingPromise) {\n      this.loadingPromise = this.loadCertificate(options);\n    }\n\n    this.enableInsecureAuth = options.enableInsecureAuth;\n    this.bundleId = options.bundleId;\n\n    if (!this.enableInsecureAuth && !this.bundleId) {\n      throw new Error('bundleId is required for secure auth.');\n    }\n  }\n\n  async loadCertificate(options) {\n    const rootCertificateUrl =\n      options.rootCertificateUrl ||\n      'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';\n\n    if (this.ca.url === rootCertificateUrl) {\n      return rootCertificateUrl;\n    }\n\n    const { certificate, headers } = await this.fetchCertificate(rootCertificateUrl);\n\n    if (\n      headers.get('content-type') !== 'application/x-pem-file' ||\n      !headers.get('content-length') ||\n      parseInt(headers.get('content-length'), 10) > 10000\n    ) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid rootCertificateURL.');\n    }\n\n    this.ca.cert = pki.certificateFromPem(certificate);\n    this.ca.url = rootCertificateUrl;\n\n    return rootCertificateUrl;\n  }\n\n  verifyPublicKeyUrl(publicKeyUrl) {\n    const regex = /^https:\\/\\/(?:[-_A-Za-z0-9]+\\.){0,}apple\\.com\\/.*\\.cer$/;\n    return regex.test(publicKeyUrl);\n  }\n\n  async fetchCertificate(url) {\n    const response = await fetch(url);\n    if (!response.ok) {\n      throw new Error(`Failed to fetch certificate: ${url}`);\n    }\n\n    const contentType = response.headers.get('content-type');\n    const isPem = contentType?.includes('application/x-pem-file');\n\n    if (isPem) {\n      const certificate = await response.text();\n      return { certificate, headers: response.headers };\n    }\n\n    const data = await response.arrayBuffer();\n    const binaryData = Buffer.from(data);\n\n    const asn1Cert = asn1.fromDer(binaryData.toString('binary'));\n    const forgeCert = pki.certificateFromAsn1(asn1Cert);\n    const certificate = pki.certificateToPem(forgeCert);\n\n    return { certificate, headers: response.headers };\n  }\n\n  async getAppleCertificate(publicKeyUrl) {\n    if (!this.verifyPublicKeyUrl(publicKeyUrl)) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);\n    }\n\n    if (this.cache[publicKeyUrl]) {\n      return this.cache[publicKeyUrl];\n    }\n\n    const { certificate, headers } = await this.fetchCertificate(publicKeyUrl);\n    const cacheControl = headers.get('cache-control');\n    const expire = cacheControl?.match(/max-age=([0-9]+)/);\n\n    this.verifyPublicKeyIssuer(certificate, publicKeyUrl);\n\n    if (expire) {\n      this.cache[publicKeyUrl] = certificate;\n      setTimeout(() => delete this.cache[publicKeyUrl], parseInt(expire[1], 10) * 1000);\n    }\n\n    return certificate;\n  }\n\n  verifyPublicKeyIssuer(cert, publicKeyUrl) {\n    const publicKeyCert = pki.certificateFromPem(cert);\n\n    if (!this.ca.cert) {\n      throw new Parse.Error(\n        Parse.Error.OBJECT_NOT_FOUND,\n        'Root certificate is invalid or missing.'\n      );\n    }\n\n    if (!this.ca.cert.verify(publicKeyCert)) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Invalid publicKeyUrl: ${publicKeyUrl}`);\n    }\n  }\n\n  verifySignature(publicKey, authData) {\n    const bundleId = this.bundleId || (this.enableInsecureAuth && authData.bundleId);\n\n    const verifier = crypto.createVerify('sha256');\n    verifier.update(Buffer.from(authData.id, 'utf8'));\n    verifier.update(Buffer.from(bundleId, 'utf8'));\n    verifier.update(this.convertTimestampToBigEndian(authData.timestamp));\n    verifier.update(Buffer.from(authData.salt, 'base64'));\n\n    if (!verifier.verify(publicKey, authData.signature, 'base64')) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid signature.');\n    }\n  }\n\n  async validateAuthData(authData) {\n\n    const requiredKeys = ['id', 'publicKeyUrl', 'timestamp', 'signature', 'salt'];\n    if (this.enableInsecureAuth) {\n      requiredKeys.push('bundleId');\n    }\n\n    for (const key of requiredKeys) {\n      if (!authData[key]) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `AuthData ${key} is missing.`);\n      }\n    }\n\n    await this.loadingPromise;\n\n    const publicKey = await this.getAppleCertificate(authData.publicKeyUrl);\n    this.verifySignature(publicKey, authData);\n  }\n\n  convertTimestampToBigEndian(timestamp) {\n    const buffer = Buffer.alloc(8);\n    buffer.writeBigUInt64BE(BigInt(timestamp));\n    return buffer;\n  }\n}\n\nexport default new GameCenterAuth();\n"],"mappings":";;;;;;AA8EA,IAAAA,OAAA,GAAAC,sBAAA,CAAAC,OAAA;AACA,IAAAC,UAAA,GAAAD,OAAA;AACA,IAAAE,YAAA,GAAAH,sBAAA,CAAAC,OAAA;AAAwC,SAAAD,uBAAAI,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAhFxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAKA,MAAMG,cAAc,SAASC,oBAAW,CAAC;EACvCC,WAAWA,CAAA,EAAG;IACZ,KAAK,CAAC,CAAC;IACP,IAAI,CAACC,EAAE,GAAG;MAAEC,IAAI,EAAE,IAAI;MAAEC,GAAG,EAAE;IAAK,CAAC;IACnC,IAAI,CAACC,KAAK,GAAG,CAAC,CAAC;IACf,IAAI,CAACC,QAAQ,GAAG,EAAE;EACpB;EAEAC,eAAeA,CAACC,OAAO,EAAE;IACvB,IAAI,CAACA,OAAO,EAAE;MACZ,MAAM,IAAIC,KAAK,CAAC,wCAAwC,CAAC;IAC3D;IAEA,IAAI,CAAC,IAAI,CAACC,cAAc,EAAE;MACxB,IAAI,CAACA,cAAc,GAAG,IAAI,CAACC,eAAe,CAACH,OAAO,CAAC;IACrD;IAEA,IAAI,CAACI,kBAAkB,GAAGJ,OAAO,CAACI,kBAAkB;IACpD,IAAI,CAACN,QAAQ,GAAGE,OAAO,CAACF,QAAQ;IAEhC,IAAI,CAAC,IAAI,CAACM,kBAAkB,IAAI,CAAC,IAAI,CAACN,QAAQ,EAAE;MAC9C,MAAM,IAAIG,KAAK,CAAC,uCAAuC,CAAC;IAC1D;EACF;EAEA,MAAME,eAAeA,CAACH,OAAO,EAAE;IAC7B,MAAMK,kBAAkB,GACtBL,OAAO,CAACK,kBAAkB,IAC1B,uFAAuF;IAEzF,IAAI,IAAI,CAACX,EAAE,CAACE,GAAG,KAAKS,kBAAkB,EAAE;MACtC,OAAOA,kBAAkB;IAC3B;IAEA,MAAM;MAAEC,WAAW;MAAEC;IAAQ,CAAC,GAAG,MAAM,IAAI,CAACC,gBAAgB,CAACH,kBAAkB,CAAC;IAEhF,IACEE,OAAO,CAACE,GAAG,CAAC,cAAc,CAAC,KAAK,wBAAwB,IACxD,CAACF,OAAO,CAACE,GAAG,CAAC,gBAAgB,CAAC,IAC9BC,QAAQ,CAACH,OAAO,CAACE,GAAG,CAAC,gBAAgB,CAAC,EAAE,EAAE,CAAC,GAAG,KAAK,EACnD;MACA,MAAM,IAAIE,KAAK,CAACV,KAAK,CAACU,KAAK,CAACV,KAAK,CAACW,gBAAgB,EAAE,6BAA6B,CAAC;IACpF;IAEA,IAAI,CAAClB,EAAE,CAACC,IAAI,GAAGkB,cAAG,CAACC,kBAAkB,CAACR,WAAW,CAAC;IAClD,IAAI,CAACZ,EAAE,CAACE,GAAG,GAAGS,kBAAkB;IAEhC,OAAOA,kBAAkB;EAC3B;EAEAU,kBAAkBA,CAACC,YAAY,EAAE;IAC/B,MAAMC,KAAK,GAAG,yDAAyD;IACvE,OAAOA,KAAK,CAACC,IAAI,CAACF,YAAY,CAAC;EACjC;EAEA,MAAMR,gBAAgBA,CAACZ,GAAG,EAAE;IAC1B,MAAMuB,QAAQ,GAAG,MAAMC,KAAK,CAACxB,GAAG,CAAC;IACjC,IAAI,CAACuB,QAAQ,CAACE,EAAE,EAAE;MAChB,MAAM,IAAIpB,KAAK,CAAC,gCAAgCL,GAAG,EAAE,CAAC;IACxD;IAEA,MAAM0B,WAAW,GAAGH,QAAQ,CAACZ,OAAO,CAACE,GAAG,CAAC,cAAc,CAAC;IACxD,MAAMc,KAAK,GAAGD,WAAW,aAAXA,WAAW,uBAAXA,WAAW,CAAEE,QAAQ,CAAC,wBAAwB,CAAC;IAE7D,IAAID,KAAK,EAAE;MACT,MAAMjB,WAAW,GAAG,MAAMa,QAAQ,CAACM,IAAI,CAAC,CAAC;MACzC,OAAO;QAAEnB,WAAW;QAAEC,OAAO,EAAEY,QAAQ,CAACZ;MAAQ,CAAC;IACnD;IAEA,MAAMmB,IAAI,GAAG,MAAMP,QAAQ,CAACQ,WAAW,CAAC,CAAC;IACzC,MAAMC,UAAU,GAAGC,MAAM,CAACC,IAAI,CAACJ,IAAI,CAAC;IAEpC,MAAMK,QAAQ,GAAGC,eAAI,CAACC,OAAO,CAACL,UAAU,CAACM,QAAQ,CAAC,QAAQ,CAAC,CAAC;IAC5D,MAAMC,SAAS,GAAGtB,cAAG,CAACuB,mBAAmB,CAACL,QAAQ,CAAC;IACnD,MAAMzB,WAAW,GAAGO,cAAG,CAACwB,gBAAgB,CAACF,SAAS,CAAC;IAEnD,OAAO;MAAE7B,WAAW;MAAEC,OAAO,EAAEY,QAAQ,CAACZ;IAAQ,CAAC;EACnD;EAEA,MAAM+B,mBAAmBA,CAACtB,YAAY,EAAE;IACtC,IAAI,CAAC,IAAI,CAACD,kBAAkB,CAACC,YAAY,CAAC,EAAE;MAC1C,MAAM,IAAIL,KAAK,CAACV,KAAK,CAACU,KAAK,CAACV,KAAK,CAACW,gBAAgB,EAAE,yBAAyBI,YAAY,EAAE,CAAC;IAC9F;IAEA,IAAI,IAAI,CAACnB,KAAK,CAACmB,YAAY,CAAC,EAAE;MAC5B,OAAO,IAAI,CAACnB,KAAK,CAACmB,YAAY,CAAC;IACjC;IAEA,MAAM;MAAEV,WAAW;MAAEC;IAAQ,CAAC,GAAG,MAAM,IAAI,CAACC,gBAAgB,CAACQ,YAAY,CAAC;IAC1E,MAAMuB,YAAY,GAAGhC,OAAO,CAACE,GAAG,CAAC,eAAe,CAAC;IACjD,MAAM+B,MAAM,GAAGD,YAAY,aAAZA,YAAY,uBAAZA,YAAY,CAAEE,KAAK,CAAC,kBAAkB,CAAC;IAEtD,IAAI,CAACC,qBAAqB,CAACpC,WAAW,EAAEU,YAAY,CAAC;IAErD,IAAIwB,MAAM,EAAE;MACV,IAAI,CAAC3C,KAAK,CAACmB,YAAY,CAAC,GAAGV,WAAW;MACtCqC,UAAU,CAAC,MAAM,OAAO,IAAI,CAAC9C,KAAK,CAACmB,YAAY,CAAC,EAAEN,QAAQ,CAAC8B,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,IAAI,CAAC;IACnF;IAEA,OAAOlC,WAAW;EACpB;EAEAoC,qBAAqBA,CAAC/C,IAAI,EAAEqB,YAAY,EAAE;IACxC,MAAM4B,aAAa,GAAG/B,cAAG,CAACC,kBAAkB,CAACnB,IAAI,CAAC;IAElD,IAAI,CAAC,IAAI,CAACD,EAAE,CAACC,IAAI,EAAE;MACjB,MAAM,IAAIgB,KAAK,CAACV,KAAK,CACnBU,KAAK,CAACV,KAAK,CAACW,gBAAgB,EAC5B,yCACF,CAAC;IACH;IAEA,IAAI,CAAC,IAAI,CAAClB,EAAE,CAACC,IAAI,CAACkD,MAAM,CAACD,aAAa,CAAC,EAAE;MACvC,MAAM,IAAIjC,KAAK,CAACV,KAAK,CAACU,KAAK,CAACV,KAAK,CAACW,gBAAgB,EAAE,yBAAyBI,YAAY,EAAE,CAAC;IAC9F;EACF;EAEA8B,eAAeA,CAACC,SAAS,EAAEC,QAAQ,EAAE;IACnC,MAAMlD,QAAQ,GAAG,IAAI,CAACA,QAAQ,IAAK,IAAI,CAACM,kBAAkB,IAAI4C,QAAQ,CAAClD,QAAS;IAEhF,MAAMmD,QAAQ,GAAGC,eAAM,CAACC,YAAY,CAAC,QAAQ,CAAC;IAC9CF,QAAQ,CAACG,MAAM,CAACvB,MAAM,CAACC,IAAI,CAACkB,QAAQ,CAACK,EAAE,EAAE,MAAM,CAAC,CAAC;IACjDJ,QAAQ,CAACG,MAAM,CAACvB,MAAM,CAACC,IAAI,CAAChC,QAAQ,EAAE,MAAM,CAAC,CAAC;IAC9CmD,QAAQ,CAACG,MAAM,CAAC,IAAI,CAACE,2BAA2B,CAACN,QAAQ,CAACO,SAAS,CAAC,CAAC;IACrEN,QAAQ,CAACG,MAAM,CAACvB,MAAM,CAACC,IAAI,CAACkB,QAAQ,CAACQ,IAAI,EAAE,QAAQ,CAAC,CAAC;IAErD,IAAI,CAACP,QAAQ,CAACJ,MAAM,CAACE,SAAS,EAAEC,QAAQ,CAACS,SAAS,EAAE,QAAQ,CAAC,EAAE;MAC7D,MAAM,IAAI9C,KAAK,CAACV,KAAK,CAACU,KAAK,CAACV,KAAK,CAACW,gBAAgB,EAAE,oBAAoB,CAAC;IAC3E;EACF;EAEA,MAAM8C,gBAAgBA,CAACV,QAAQ,EAAE;IAE/B,MAAMW,YAAY,GAAG,CAAC,IAAI,EAAE,cAAc,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,CAAC;IAC7E,IAAI,IAAI,CAACvD,kBAAkB,EAAE;MAC3BuD,YAAY,CAACC,IAAI,CAAC,UAAU,CAAC;IAC/B;IAEA,KAAK,MAAMC,GAAG,IAAIF,YAAY,EAAE;MAC9B,IAAI,CAACX,QAAQ,CAACa,GAAG,CAAC,EAAE;QAClB,MAAM,IAAIlD,KAAK,CAACV,KAAK,CAACU,KAAK,CAACV,KAAK,CAACW,gBAAgB,EAAE,YAAYiD,GAAG,cAAc,CAAC;MACpF;IACF;IAEA,MAAM,IAAI,CAAC3D,cAAc;IAEzB,MAAM6C,SAAS,GAAG,MAAM,IAAI,CAACT,mBAAmB,CAACU,QAAQ,CAAChC,YAAY,CAAC;IACvE,IAAI,CAAC8B,eAAe,CAACC,SAAS,EAAEC,QAAQ,CAAC;EAC3C;EAEAM,2BAA2BA,CAACC,SAAS,EAAE;IACrC,MAAMO,MAAM,GAAGjC,MAAM,CAACkC,KAAK,CAAC,CAAC,CAAC;IAC9BD,MAAM,CAACE,gBAAgB,CAACC,MAAM,CAACV,SAAS,CAAC,CAAC;IAC1C,OAAOO,MAAM;EACf;AACF;AAAC,IAAAI,QAAA,GAAAC,OAAA,CAAA7E,OAAA,GAEc,IAAIC,cAAc,CAAC,CAAC","ignoreList":[]}
@@ -1,37 +1,125 @@
1
1
  "use strict";
2
2
 
3
- // Helper functions for accessing the github API.
4
- var Parse = require('parse/node').Parse;
5
- const httpsRequest = require('./httpsRequest');
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = void 0;
7
+ var _BaseCodeAuthAdapter = _interopRequireDefault(require("./BaseCodeAuthAdapter"));
8
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
9
+ /**
10
+ * Parse Server authentication adapter for GitHub.
11
+ * @class GitHubAdapter
12
+ * @param {Object} options - The adapter configuration options.
13
+ * @param {string} options.clientId - The GitHub App Client ID. Required for secure authentication.
14
+ * @param {string} options.clientSecret - The GitHub App Client Secret. Required for secure authentication.
15
+ * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).
16
+ *
17
+ * @param {Object} authData - The authentication data provided by the client.
18
+ * @param {string} authData.code - The authorization code from GitHub. Required for secure authentication.
19
+ * @param {string} [authData.id] - **[DEPRECATED]** The GitHub user ID (required for insecure authentication).
20
+ * @param {string} [authData.access_token] - **[DEPRECATED]** The GitHub access token (required for insecure authentication).
21
+ *
22
+ * @description
23
+ * ## Parse Server Configuration
24
+ * * To configure Parse Server for GitHub authentication, use the following structure:
25
+ * ```json
26
+ * {
27
+ * "auth": {
28
+ * "github": {
29
+ * "clientId": "12345",
30
+ * "clientSecret": "abcde"
31
+ * }
32
+ * }
33
+ * ```
34
+ *
35
+ * The GitHub adapter exchanges the `authData.code` provided by the client for an access token using GitHub's OAuth API. The following `authData` field is required:
36
+ * - `code`
37
+ *
38
+ * ## Insecure Authentication (Not Recommended)
39
+ * Insecure authentication uses the `authData.id` and `authData.access_token` provided by the client. This flow is insecure, deprecated, and poses potential security risks. The following `authData` fields are required:
40
+ * - `id` (**[DEPRECATED]**): The GitHub user ID.
41
+ * - `access_token` (**[DEPRECATED]**): The GitHub access token.
42
+ * To configure Parse Server for insecure authentication, use the following structure:
43
+ * ```json
44
+ * {
45
+ * "auth": {
46
+ * "github": {
47
+ * "enableInsecureAuth": true
48
+ * }
49
+ * }
50
+ * ```
51
+ *
52
+ * ### Deprecation Notice
53
+ * The `enableInsecureAuth` option and insecure `authData` fields (`id`, `access_token`) are deprecated and will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.
54
+ *
55
+ * @example <caption>Secure Authentication Example</caption>
56
+ * // Example authData for secure authentication:
57
+ * const authData = {
58
+ * github: {
59
+ * code: "abc123def456ghi789"
60
+ * }
61
+ * };
62
+ *
63
+ * @example <caption>Insecure Authentication Example (Not Recommended)</caption>
64
+ * // Example authData for insecure authentication:
65
+ * const authData = {
66
+ * github: {
67
+ * id: "1234567",
68
+ * access_token: "abc123def456ghi789" // Deprecated.
69
+ * }
70
+ * };
71
+ *
72
+ * @note `enableInsecureAuth` will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.
73
+ * @note Secure authentication exchanges the `code` provided by the client for an access token using GitHub's OAuth API.
74
+ *
75
+ * @see {@link https://docs.github.com/en/developers/apps/authorizing-oauth-apps GitHub OAuth Documentation}
76
+ */
6
77
 
7
- // Returns a promise that fulfills iff this user id is valid.
8
- function validateAuthData(authData) {
9
- return request('user', authData.access_token).then(data => {
10
- if (data && data.id == authData.id) {
11
- return;
78
+ class GitHubAdapter extends _BaseCodeAuthAdapter.default {
79
+ constructor() {
80
+ super('GitHub');
81
+ }
82
+ async getAccessTokenFromCode(authData) {
83
+ const tokenUrl = 'https://github.com/login/oauth/access_token';
84
+ const response = await fetch(tokenUrl, {
85
+ method: 'POST',
86
+ headers: {
87
+ 'Content-Type': 'application/json',
88
+ Accept: 'application/json'
89
+ },
90
+ body: JSON.stringify({
91
+ client_id: this.clientId,
92
+ client_secret: this.clientSecret,
93
+ code: authData.code
94
+ })
95
+ });
96
+ if (!response.ok) {
97
+ throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to exchange code for token: ${response.statusText}`);
12
98
  }
13
- throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Github auth is invalid for this user.');
14
- });
15
- }
16
-
17
- // Returns a promise that fulfills iff this app id is valid.
18
- function validateAppId() {
19
- return Promise.resolve();
20
- }
21
-
22
- // A promisey wrapper for api requests
23
- function request(path, access_token) {
24
- return httpsRequest.get({
25
- host: 'api.github.com',
26
- path: '/' + path,
27
- headers: {
28
- Authorization: 'bearer ' + access_token,
29
- 'User-Agent': 'parse-server'
99
+ const data = await response.json();
100
+ if (data.error) {
101
+ throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, data.error_description || data.error);
102
+ }
103
+ return data.access_token;
104
+ }
105
+ async getUserFromAccessToken(accessToken) {
106
+ const userApiUrl = 'https://api.github.com/user';
107
+ const response = await fetch(userApiUrl, {
108
+ method: 'GET',
109
+ headers: {
110
+ Authorization: `Bearer ${accessToken}`,
111
+ Accept: 'application/json'
112
+ }
113
+ });
114
+ if (!response.ok) {
115
+ throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to fetch GitHub user: ${response.statusText}`);
116
+ }
117
+ const userData = await response.json();
118
+ if (!userData.id || !userData.login) {
119
+ throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Invalid GitHub user data received.');
30
120
  }
31
- });
121
+ return userData;
122
+ }
32
123
  }
33
- module.exports = {
34
- validateAppId: validateAppId,
35
- validateAuthData: validateAuthData
36
- };
37
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJodHRwc1JlcXVlc3QiLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwidGhlbiIsImRhdGEiLCJpZCIsIkVycm9yIiwiT0JKRUNUX05PVF9GT1VORCIsInZhbGlkYXRlQXBwSWQiLCJQcm9taXNlIiwicmVzb2x2ZSIsInBhdGgiLCJnZXQiLCJob3N0IiwiaGVhZGVycyIsIkF1dGhvcml6YXRpb24iLCJtb2R1bGUiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL0FkYXB0ZXJzL0F1dGgvZ2l0aHViLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEhlbHBlciBmdW5jdGlvbnMgZm9yIGFjY2Vzc2luZyB0aGUgZ2l0aHViIEFQSS5cbnZhciBQYXJzZSA9IHJlcXVpcmUoJ3BhcnNlL25vZGUnKS5QYXJzZTtcbmNvbnN0IGh0dHBzUmVxdWVzdCA9IHJlcXVpcmUoJy4vaHR0cHNSZXF1ZXN0Jyk7XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEpIHtcbiAgcmV0dXJuIHJlcXVlc3QoJ3VzZXInLCBhdXRoRGF0YS5hY2Nlc3NfdG9rZW4pLnRoZW4oZGF0YSA9PiB7XG4gICAgaWYgKGRhdGEgJiYgZGF0YS5pZCA9PSBhdXRoRGF0YS5pZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCwgJ0dpdGh1YiBhdXRoIGlzIGludmFsaWQgZm9yIHRoaXMgdXNlci4nKTtcbiAgfSk7XG59XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgYXBwIGlkIGlzIHZhbGlkLlxuZnVuY3Rpb24gdmFsaWRhdGVBcHBJZCgpIHtcbiAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpO1xufVxuXG4vLyBBIHByb21pc2V5IHdyYXBwZXIgZm9yIGFwaSByZXF1ZXN0c1xuZnVuY3Rpb24gcmVxdWVzdChwYXRoLCBhY2Nlc3NfdG9rZW4pIHtcbiAgcmV0dXJuIGh0dHBzUmVxdWVzdC5nZXQoe1xuICAgIGhvc3Q6ICdhcGkuZ2l0aHViLmNvbScsXG4gICAgcGF0aDogJy8nICsgcGF0aCxcbiAgICBoZWFkZXJzOiB7XG4gICAgICBBdXRob3JpemF0aW9uOiAnYmVhcmVyICcgKyBhY2Nlc3NfdG9rZW4sXG4gICAgICAnVXNlci1BZ2VudCc6ICdwYXJzZS1zZXJ2ZXInLFxuICAgIH0sXG4gIH0pO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgdmFsaWRhdGVBcHBJZDogdmFsaWRhdGVBcHBJZCxcbiAgdmFsaWRhdGVBdXRoRGF0YTogdmFsaWRhdGVBdXRoRGF0YSxcbn07XG4iXSwibWFwcGluZ3MiOiI7O0FBQUE7QUFDQSxJQUFJQSxLQUFLLEdBQUdDLE9BQU8sQ0FBQyxZQUFZLENBQUMsQ0FBQ0QsS0FBSztBQUN2QyxNQUFNRSxZQUFZLEdBQUdELE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQzs7QUFFOUM7QUFDQSxTQUFTRSxnQkFBZ0JBLENBQUNDLFFBQVEsRUFBRTtFQUNsQyxPQUFPQyxPQUFPLENBQUMsTUFBTSxFQUFFRCxRQUFRLENBQUNFLFlBQVksQ0FBQyxDQUFDQyxJQUFJLENBQUNDLElBQUksSUFBSTtJQUN6RCxJQUFJQSxJQUFJLElBQUlBLElBQUksQ0FBQ0MsRUFBRSxJQUFJTCxRQUFRLENBQUNLLEVBQUUsRUFBRTtNQUNsQztJQUNGO0lBQ0EsTUFBTSxJQUFJVCxLQUFLLENBQUNVLEtBQUssQ0FBQ1YsS0FBSyxDQUFDVSxLQUFLLENBQUNDLGdCQUFnQixFQUFFLHVDQUF1QyxDQUFDO0VBQzlGLENBQUMsQ0FBQztBQUNKOztBQUVBO0FBQ0EsU0FBU0MsYUFBYUEsQ0FBQSxFQUFHO0VBQ3ZCLE9BQU9DLE9BQU8sQ0FBQ0MsT0FBTyxDQUFDLENBQUM7QUFDMUI7O0FBRUE7QUFDQSxTQUFTVCxPQUFPQSxDQUFDVSxJQUFJLEVBQUVULFlBQVksRUFBRTtFQUNuQyxPQUFPSixZQUFZLENBQUNjLEdBQUcsQ0FBQztJQUN0QkMsSUFBSSxFQUFFLGdCQUFnQjtJQUN0QkYsSUFBSSxFQUFFLEdBQUcsR0FBR0EsSUFBSTtJQUNoQkcsT0FBTyxFQUFFO01BQ1BDLGFBQWEsRUFBRSxTQUFTLEdBQUdiLFlBQVk7TUFDdkMsWUFBWSxFQUFFO0lBQ2hCO0VBQ0YsQ0FBQyxDQUFDO0FBQ0o7QUFFQWMsTUFBTSxDQUFDQyxPQUFPLEdBQUc7RUFDZlQsYUFBYSxFQUFFQSxhQUFhO0VBQzVCVCxnQkFBZ0IsRUFBRUE7QUFDcEIsQ0FBQyIsImlnbm9yZUxpc3QiOltdfQ==
124
+ var _default = exports.default = new GitHubAdapter();
125
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_BaseCodeAuthAdapter","_interopRequireDefault","require","e","__esModule","default","GitHubAdapter","BaseCodeAuthAdapter","constructor","getAccessTokenFromCode","authData","tokenUrl","response","fetch","method","headers","Accept","body","JSON","stringify","client_id","clientId","client_secret","clientSecret","code","ok","Parse","Error","VALIDATION_ERROR","statusText","data","json","error","OBJECT_NOT_FOUND","error_description","access_token","getUserFromAccessToken","accessToken","userApiUrl","Authorization","userData","id","login","_default","exports"],"sources":["../../../src/Adapters/Auth/github.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for GitHub.\n * @class GitHubAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {string} options.clientId - The GitHub App Client ID. Required for secure authentication.\n * @param {string} options.clientSecret - The GitHub App Client Secret. Required for secure authentication.\n * @param {boolean} [options.enableInsecureAuth=false] - **[DEPRECATED]** Enable insecure authentication (not recommended).\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {string} authData.code - The authorization code from GitHub. Required for secure authentication.\n * @param {string} [authData.id] - **[DEPRECATED]** The GitHub user ID (required for insecure authentication).\n * @param {string} [authData.access_token] - **[DEPRECATED]** The GitHub access token (required for insecure authentication).\n *\n * @description\n * ## Parse Server Configuration\n * * To configure Parse Server for GitHub authentication, use the following structure:\n * ```json\n * {\n *  \"auth\": {\n *   \"github\": {\n *     \"clientId\": \"12345\",\n *     \"clientSecret\": \"abcde\"\n *   }\n * }\n * ```\n *\n * The GitHub adapter exchanges the `authData.code` provided by the client for an access token using GitHub's OAuth API. The following `authData` field is required:\n * - `code`\n *\n * ## Insecure Authentication (Not Recommended)\n * Insecure authentication uses the `authData.id` and `authData.access_token` provided by the client. This flow is insecure, deprecated, and poses potential security risks. The following `authData` fields are required:\n * - `id` (**[DEPRECATED]**): The GitHub user ID.\n * - `access_token` (**[DEPRECATED]**): The GitHub access token.\n * To configure Parse Server for insecure authentication, use the following structure:\n * ```json\n * {\n *  \"auth\": {\n *    \"github\": {\n *    \"enableInsecureAuth\": true\n *  }\n * }\n * ```\n *\n * ### Deprecation Notice\n * The `enableInsecureAuth` option and insecure `authData` fields (`id`, `access_token`) are deprecated and will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.\n *\n * @example <caption>Secure Authentication Example</caption>\n * // Example authData for secure authentication:\n * const authData = {\n *   github: {\n *     code: \"abc123def456ghi789\"\n *   }\n * };\n *\n * @example <caption>Insecure Authentication Example (Not Recommended)</caption>\n * // Example authData for insecure authentication:\n * const authData = {\n *   github: {\n *     id: \"1234567\",\n *     access_token: \"abc123def456ghi789\" // Deprecated.\n *   }\n * };\n *\n * @note `enableInsecureAuth` will be removed in future versions. Use secure authentication with `clientId` and `clientSecret`.\n * @note Secure authentication exchanges the `code` provided by the client for an access token using GitHub's OAuth API.\n *\n * @see {@link https://docs.github.com/en/developers/apps/authorizing-oauth-apps GitHub OAuth Documentation}\n */\n\nimport BaseCodeAuthAdapter from './BaseCodeAuthAdapter';\nclass GitHubAdapter extends BaseCodeAuthAdapter {\n  constructor() {\n    super('GitHub');\n  }\n  async getAccessTokenFromCode(authData) {\n    const tokenUrl = 'https://github.com/login/oauth/access_token';\n    const response = await fetch(tokenUrl, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        Accept: 'application/json',\n      },\n      body: JSON.stringify({\n        client_id: this.clientId,\n        client_secret: this.clientSecret,\n        code: authData.code,\n      }),\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to exchange code for token: ${response.statusText}`);\n    }\n\n    const data = await response.json();\n    if (data.error) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, data.error_description || data.error);\n    }\n\n    return data.access_token;\n  }\n\n  async getUserFromAccessToken(accessToken) {\n    const userApiUrl = 'https://api.github.com/user';\n    const response = await fetch(userApiUrl, {\n      method: 'GET',\n      headers: {\n        Authorization: `Bearer ${accessToken}`,\n        Accept: 'application/json',\n      },\n    });\n\n    if (!response.ok) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, `Failed to fetch GitHub user: ${response.statusText}`);\n    }\n\n    const userData = await response.json();\n    if (!userData.id || !userData.login) {\n      throw new Parse.Error(Parse.Error.VALIDATION_ERROR, 'Invalid GitHub user data received.');\n    }\n\n    return userData;\n  }\n\n}\n\nexport default new GitHubAdapter();\n\n"],"mappings":"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","ignoreList":[]}