parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (191) hide show
  1. package/README.md +45 -17
  2. package/lib/AccountLockout.js +11 -26
  3. package/lib/Adapters/AdapterLoader.js +8 -14
  4. package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
  5. package/lib/Adapters/Auth/AuthAdapter.js +7 -16
  6. package/lib/Adapters/Auth/OAuth1Client.js +32 -57
  7. package/lib/Adapters/Auth/apple.js +6 -22
  8. package/lib/Adapters/Auth/facebook.js +7 -37
  9. package/lib/Adapters/Auth/gcenter.js +8 -37
  10. package/lib/Adapters/Auth/github.js +7 -10
  11. package/lib/Adapters/Auth/google.js +11 -34
  12. package/lib/Adapters/Auth/gpgames.js +5 -8
  13. package/lib/Adapters/Auth/httpsRequest.js +1 -7
  14. package/lib/Adapters/Auth/index.js +20 -65
  15. package/lib/Adapters/Auth/instagram.js +5 -9
  16. package/lib/Adapters/Auth/janraincapture.js +8 -12
  17. package/lib/Adapters/Auth/janrainengage.js +7 -11
  18. package/lib/Adapters/Auth/keycloak.js +5 -19
  19. package/lib/Adapters/Auth/ldap.js +1 -15
  20. package/lib/Adapters/Auth/line.js +7 -10
  21. package/lib/Adapters/Auth/linkedin.js +7 -12
  22. package/lib/Adapters/Auth/meetup.js +7 -10
  23. package/lib/Adapters/Auth/microsoft.js +7 -10
  24. package/lib/Adapters/Auth/oauth2.js +6 -18
  25. package/lib/Adapters/Auth/phantauth.js +8 -10
  26. package/lib/Adapters/Auth/qq.js +7 -13
  27. package/lib/Adapters/Auth/spotify.js +7 -14
  28. package/lib/Adapters/Auth/twitter.js +5 -15
  29. package/lib/Adapters/Auth/vkontakte.js +9 -15
  30. package/lib/Adapters/Auth/wechat.js +7 -10
  31. package/lib/Adapters/Auth/weibo.js +7 -11
  32. package/lib/Adapters/Cache/CacheAdapter.js +4 -12
  33. package/lib/Adapters/Cache/InMemoryCache.js +5 -19
  34. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
  35. package/lib/Adapters/Cache/LRUCache.js +1 -11
  36. package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
  37. package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
  38. package/lib/Adapters/Cache/SchemaCache.js +1 -6
  39. package/lib/Adapters/Email/MailAdapter.js +2 -7
  40. package/lib/Adapters/Files/FilesAdapter.js +7 -21
  41. package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
  42. package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
  43. package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
  44. package/lib/Adapters/Logger/WinstonLogger.js +3 -30
  45. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
  46. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
  47. package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
  48. package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
  49. package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
  50. package/lib/Adapters/Push/PushAdapter.js +2 -8
  51. package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
  52. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
  53. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
  54. package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
  55. package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
  56. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
  57. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
  58. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
  59. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  60. package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
  61. package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
  62. package/lib/Auth.js +54 -121
  63. package/lib/ClientSDK.js +3 -11
  64. package/lib/Config.js +69 -113
  65. package/lib/Controllers/AdaptableController.js +6 -18
  66. package/lib/Controllers/AnalyticsController.js +1 -9
  67. package/lib/Controllers/CacheController.js +3 -23
  68. package/lib/Controllers/DatabaseController.js +147 -345
  69. package/lib/Controllers/FilesController.js +5 -34
  70. package/lib/Controllers/HooksController.js +1 -51
  71. package/lib/Controllers/LiveQueryController.js +4 -23
  72. package/lib/Controllers/LoggerController.js +15 -54
  73. package/lib/Controllers/ParseGraphQLController.js +49 -104
  74. package/lib/Controllers/PushController.js +20 -59
  75. package/lib/Controllers/SchemaController.js +154 -344
  76. package/lib/Controllers/UserController.js +11 -72
  77. package/lib/Controllers/index.js +19 -68
  78. package/lib/Controllers/types.js +1 -1
  79. package/lib/Deprecator/Deprecations.js +1 -8
  80. package/lib/Deprecator/Deprecator.js +9 -18
  81. package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
  82. package/lib/GraphQL/ParseGraphQLServer.js +2 -29
  83. package/lib/GraphQL/helpers/objectsMutations.js +2 -12
  84. package/lib/GraphQL/helpers/objectsQueries.js +18 -76
  85. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
  86. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
  87. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
  88. package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
  89. package/lib/GraphQL/loaders/filesMutations.js +2 -19
  90. package/lib/GraphQL/loaders/functionsMutations.js +6 -17
  91. package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
  92. package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
  93. package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
  94. package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
  95. package/lib/GraphQL/loaders/schemaMutations.js +1 -20
  96. package/lib/GraphQL/loaders/schemaQueries.js +1 -14
  97. package/lib/GraphQL/loaders/schemaTypes.js +2 -6
  98. package/lib/GraphQL/loaders/usersMutations.js +6 -28
  99. package/lib/GraphQL/loaders/usersQueries.js +4 -26
  100. package/lib/GraphQL/parseGraphQLUtils.js +6 -19
  101. package/lib/GraphQL/transformers/className.js +1 -4
  102. package/lib/GraphQL/transformers/constraintType.js +1 -20
  103. package/lib/GraphQL/transformers/inputType.js +1 -20
  104. package/lib/GraphQL/transformers/mutation.js +6 -51
  105. package/lib/GraphQL/transformers/outputType.js +1 -20
  106. package/lib/GraphQL/transformers/query.js +6 -42
  107. package/lib/GraphQL/transformers/schemaFields.js +7 -34
  108. package/lib/KeyPromiseQueue.js +1 -12
  109. package/lib/LiveQuery/Client.js +1 -25
  110. package/lib/LiveQuery/Id.js +1 -7
  111. package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
  112. package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
  113. package/lib/LiveQuery/ParsePubSub.js +1 -12
  114. package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
  115. package/lib/LiveQuery/QueryTools.js +14 -116
  116. package/lib/LiveQuery/RequestSchema.js +1 -1
  117. package/lib/LiveQuery/SessionTokenCache.js +1 -17
  118. package/lib/LiveQuery/Subscription.js +4 -18
  119. package/lib/LiveQuery/equalObjects.js +2 -14
  120. package/lib/Options/Definitions.js +79 -10
  121. package/lib/Options/docs.js +23 -3
  122. package/lib/Options/index.js +4 -12
  123. package/lib/Options/parsers.js +1 -18
  124. package/lib/Page.js +1 -9
  125. package/lib/ParseMessageQueue.js +1 -10
  126. package/lib/ParseServer.js +144 -182
  127. package/lib/ParseServerRESTController.js +6 -33
  128. package/lib/PromiseRouter.js +16 -50
  129. package/lib/Push/PushQueue.js +3 -15
  130. package/lib/Push/PushWorker.js +7 -32
  131. package/lib/Push/utils.js +9 -38
  132. package/lib/RestQuery.js +105 -242
  133. package/lib/RestWrite.js +212 -377
  134. package/lib/Routers/AggregateRouter.js +14 -51
  135. package/lib/Routers/AnalyticsRouter.js +2 -8
  136. package/lib/Routers/AudiencesRouter.js +1 -15
  137. package/lib/Routers/ClassesRouter.js +3 -53
  138. package/lib/Routers/CloudCodeRouter.js +1 -19
  139. package/lib/Routers/FeaturesRouter.js +1 -10
  140. package/lib/Routers/FilesRouter.js +29 -76
  141. package/lib/Routers/FunctionsRouter.js +5 -28
  142. package/lib/Routers/GlobalConfigRouter.js +4 -18
  143. package/lib/Routers/GraphQLRouter.js +1 -14
  144. package/lib/Routers/HooksRouter.js +1 -29
  145. package/lib/Routers/IAPValidationRouter.js +6 -29
  146. package/lib/Routers/InstallationsRouter.js +2 -12
  147. package/lib/Routers/LogsRouter.js +4 -16
  148. package/lib/Routers/PagesRouter.js +69 -129
  149. package/lib/Routers/PublicAPIRouter.js +3 -62
  150. package/lib/Routers/PurgeRouter.js +1 -15
  151. package/lib/Routers/PushRouter.js +2 -18
  152. package/lib/Routers/RolesRouter.js +1 -7
  153. package/lib/Routers/SchemasRouter.js +4 -34
  154. package/lib/Routers/SecurityRouter.js +1 -12
  155. package/lib/Routers/SessionsRouter.js +3 -19
  156. package/lib/Routers/UsersRouter.js +58 -155
  157. package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
  158. package/lib/SchemaMigrations/Migrations.js +2 -8
  159. package/lib/Security/Check.js +8 -16
  160. package/lib/Security/CheckGroup.js +4 -11
  161. package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
  162. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
  163. package/lib/Security/CheckGroups/CheckGroups.js +1 -4
  164. package/lib/Security/CheckRunner.js +22 -41
  165. package/lib/StatusHandler.js +12 -69
  166. package/lib/TestUtils.js +1 -6
  167. package/lib/Utils.js +27 -66
  168. package/lib/batch.js +17 -28
  169. package/lib/cache.js +1 -3
  170. package/lib/cli/definitions/parse-live-query-server.js +1 -3
  171. package/lib/cli/definitions/parse-server.js +1 -3
  172. package/lib/cli/parse-live-query-server.js +1 -6
  173. package/lib/cli/parse-server.js +11 -21
  174. package/lib/cli/utils/commander.js +13 -51
  175. package/lib/cli/utils/runner.js +1 -14
  176. package/lib/cloud-code/Parse.Cloud.js +71 -92
  177. package/lib/cryptoUtils.js +11 -19
  178. package/lib/defaults.js +2 -14
  179. package/lib/deprecated.js +1 -2
  180. package/lib/index.js +16 -34
  181. package/lib/logger.js +6 -13
  182. package/lib/middlewares.js +147 -151
  183. package/lib/password.js +6 -10
  184. package/lib/request.js +173 -2
  185. package/lib/requiredParameter.js +1 -3
  186. package/lib/rest.js +19 -41
  187. package/lib/triggers.js +54 -252
  188. package/lib/vendor/mongodbUrl.js +125 -305
  189. package/package.json +22 -19
  190. package/lib/cloud-code/HTTPResponse.js +0 -73
  191. package/lib/cloud-code/httpRequest.js +0 -192
@@ -2,62 +2,44 @@
2
2
 
3
3
  // Helper functions for accessing the Facebook Graph API.
4
4
  const Parse = require('parse/node').Parse;
5
-
6
5
  const crypto = require('crypto');
7
-
8
6
  const jwksClient = require('jwks-rsa');
9
-
10
7
  const util = require('util');
11
-
12
8
  const jwt = require('jsonwebtoken');
13
-
14
9
  const httpsRequest = require('./httpsRequest');
15
-
16
10
  const TOKEN_ISSUER = 'https://facebook.com';
17
-
18
11
  function getAppSecretPath(authData, options = {}) {
19
12
  const appSecret = options.appSecret;
20
-
21
13
  if (!appSecret) {
22
14
  return '';
23
15
  }
24
-
25
16
  const appsecret_proof = crypto.createHmac('sha256', appSecret).update(authData.access_token).digest('hex');
26
17
  return `&appsecret_proof=${appsecret_proof}`;
27
18
  }
28
-
29
19
  function validateGraphToken(authData, options) {
30
20
  return graphRequest('me?fields=id&access_token=' + authData.access_token + getAppSecretPath(authData, options)).then(data => {
31
21
  if (data && data.id == authData.id || process.env.TESTING && authData.id === 'test') {
32
22
  return;
33
23
  }
34
-
35
24
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');
36
25
  });
37
26
  }
38
-
39
27
  async function validateGraphAppId(appIds, authData, options) {
40
28
  var access_token = authData.access_token;
41
-
42
29
  if (process.env.TESTING && access_token === 'test') {
43
30
  return;
44
31
  }
45
-
46
32
  if (!Array.isArray(appIds)) {
47
33
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'appIds must be an array.');
48
34
  }
49
-
50
35
  if (!appIds.length) {
51
36
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is not configured.');
52
37
  }
53
-
54
38
  const data = await graphRequest(`app?access_token=${access_token}${getAppSecretPath(authData, options)}`);
55
-
56
39
  if (!data || !appIds.includes(data.id)) {
57
40
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');
58
41
  }
59
42
  }
60
-
61
43
  const getFacebookKeyByKeyId = async (keyId, cacheMaxEntries, cacheMaxAge) => {
62
44
  const client = jwksClient({
63
45
  jwksUri: `${TOKEN_ISSUER}/.well-known/oauth/openid/jwks/`,
@@ -67,28 +49,22 @@ const getFacebookKeyByKeyId = async (keyId, cacheMaxEntries, cacheMaxAge) => {
67
49
  });
68
50
  const asyncGetSigningKeyFunction = util.promisify(client.getSigningKey);
69
51
  let key;
70
-
71
52
  try {
72
53
  key = await asyncGetSigningKeyFunction(keyId);
73
54
  } catch (error) {
74
55
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Unable to find matching key for Key ID: ${keyId}`);
75
56
  }
76
-
77
57
  return key;
78
58
  };
79
-
80
59
  const getHeaderFromToken = token => {
81
60
  const decodedToken = jwt.decode(token, {
82
61
  complete: true
83
62
  });
84
-
85
63
  if (!decodedToken) {
86
64
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'provided token does not decode as JWT');
87
65
  }
88
-
89
66
  return decodedToken.header;
90
67
  };
91
-
92
68
  const verifyIdToken = async ({
93
69
  token,
94
70
  id
@@ -100,7 +76,6 @@ const verifyIdToken = async ({
100
76
  if (!token) {
101
77
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'id token is invalid for this user.');
102
78
  }
103
-
104
79
  const {
105
80
  kid: keyId,
106
81
  alg: algorithm
@@ -111,7 +86,6 @@ const verifyIdToken = async ({
111
86
  cacheMaxEntries = cacheMaxEntries || 5;
112
87
  const facebookKey = await getFacebookKeyByKeyId(keyId, cacheMaxEntries, cacheMaxAge);
113
88
  const signingKey = facebookKey.publicKey || facebookKey.rsaPublicKey;
114
-
115
89
  try {
116
90
  jwtClaims = jwt.verify(token, signingKey, {
117
91
  algorithms: algorithm,
@@ -122,43 +96,39 @@ const verifyIdToken = async ({
122
96
  const message = exception.message;
123
97
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);
124
98
  }
125
-
126
99
  if (jwtClaims.iss !== TOKEN_ISSUER) {
127
100
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `id token not issued by correct OpenID provider - expected: ${TOKEN_ISSUER} | from: ${jwtClaims.iss}`);
128
101
  }
129
-
130
102
  if (jwtClaims.sub !== id) {
131
103
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'auth data is invalid for this user.');
132
104
  }
133
-
134
105
  return jwtClaims;
135
- }; // Returns a promise that fulfills iff this user id is valid.
136
-
106
+ };
137
107
 
108
+ // Returns a promise that fulfills iff this user id is valid.
138
109
  function validateAuthData(authData, options) {
139
110
  if (authData.token) {
140
111
  return verifyIdToken(authData, options);
141
112
  } else {
142
113
  return validateGraphToken(authData, options);
143
114
  }
144
- } // Returns a promise that fulfills iff this app id is valid.
145
-
115
+ }
146
116
 
117
+ // Returns a promise that fulfills iff this app id is valid.
147
118
  function validateAppId(appIds, authData, options) {
148
119
  if (authData.token) {
149
120
  return Promise.resolve();
150
121
  } else {
151
122
  return validateGraphAppId(appIds, authData, options);
152
123
  }
153
- } // A promisey wrapper for FB graph requests.
154
-
124
+ }
155
125
 
126
+ // A promisey wrapper for FB graph requests.
156
127
  function graphRequest(path) {
157
128
  return httpsRequest.get('https://graph.facebook.com/' + path);
158
129
  }
159
-
160
130
  module.exports = {
161
131
  validateAppId: validateAppId,
162
132
  validateAuthData: validateAuthData
163
133
  };
164
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/Adapters/Auth/facebook.js"],"names":["Parse","require","crypto","jwksClient","util","jwt","httpsRequest","TOKEN_ISSUER","getAppSecretPath","authData","options","appSecret","appsecret_proof","createHmac","update","access_token","digest","validateGraphToken","graphRequest","then","data","id","process","env","TESTING","Error","OBJECT_NOT_FOUND","validateGraphAppId","appIds","Array","isArray","length","includes","getFacebookKeyByKeyId","keyId","cacheMaxEntries","cacheMaxAge","client","jwksUri","cache","asyncGetSigningKeyFunction","promisify","getSigningKey","key","error","getHeaderFromToken","token","decodedToken","decode","complete","header","verifyIdToken","clientId","kid","alg","algorithm","ONE_HOUR_IN_MS","jwtClaims","facebookKey","signingKey","publicKey","rsaPublicKey","verify","algorithms","audience","exception","message","iss","sub","validateAuthData","validateAppId","Promise","resolve","path","get","module","exports"],"mappings":";;AAAA;AACA,MAAMA,KAAK,GAAGC,OAAO,CAAC,YAAD,CAAP,CAAsBD,KAApC;;AACA,MAAME,MAAM,GAAGD,OAAO,CAAC,QAAD,CAAtB;;AACA,MAAME,UAAU,GAAGF,OAAO,CAAC,UAAD,CAA1B;;AACA,MAAMG,IAAI,GAAGH,OAAO,CAAC,MAAD,CAApB;;AACA,MAAMI,GAAG,GAAGJ,OAAO,CAAC,cAAD,CAAnB;;AACA,MAAMK,YAAY,GAAGL,OAAO,CAAC,gBAAD,CAA5B;;AAEA,MAAMM,YAAY,GAAG,sBAArB;;AAEA,SAASC,gBAAT,CAA0BC,QAA1B,EAAoCC,OAAO,GAAG,EAA9C,EAAkD;AAChD,QAAMC,SAAS,GAAGD,OAAO,CAACC,SAA1B;;AACA,MAAI,CAACA,SAAL,EAAgB;AACd,WAAO,EAAP;AACD;;AACD,QAAMC,eAAe,GAAGV,MAAM,CAC3BW,UADqB,CACV,QADU,EACAF,SADA,EAErBG,MAFqB,CAEdL,QAAQ,CAACM,YAFK,EAGrBC,MAHqB,CAGd,KAHc,CAAxB;AAKA,SAAQ,oBAAmBJ,eAAgB,EAA3C;AACD;;AAED,SAASK,kBAAT,CAA4BR,QAA5B,EAAsCC,OAAtC,EAA+C;AAC7C,SAAOQ,YAAY,CACjB,+BAA+BT,QAAQ,CAACM,YAAxC,GAAuDP,gBAAgB,CAACC,QAAD,EAAWC,OAAX,CADtD,CAAZ,CAELS,IAFK,CAEAC,IAAI,IAAI;AACb,QAAKA,IAAI,IAAIA,IAAI,CAACC,EAAL,IAAWZ,QAAQ,CAACY,EAA7B,IAAqCC,OAAO,CAACC,GAAR,CAAYC,OAAZ,IAAuBf,QAAQ,CAACY,EAAT,KAAgB,MAAhF,EAAyF;AACvF;AACD;;AACD,UAAM,IAAIrB,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,yCAA9C,CAAN;AACD,GAPM,CAAP;AAQD;;AAED,eAAeC,kBAAf,CAAkCC,MAAlC,EAA0CnB,QAA1C,EAAoDC,OAApD,EAA6D;AAC3D,MAAIK,YAAY,GAAGN,QAAQ,CAACM,YAA5B;;AACA,MAAIO,OAAO,CAACC,GAAR,CAAYC,OAAZ,IAAuBT,YAAY,KAAK,MAA5C,EAAoD;AAClD;AACD;;AACD,MAAI,CAACc,KAAK,CAACC,OAAN,CAAcF,MAAd,CAAL,EAA4B;AAC1B,UAAM,IAAI5B,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,0BAA9C,CAAN;AACD;;AACD,MAAI,CAACE,MAAM,CAACG,MAAZ,EAAoB;AAClB,UAAM,IAAI/B,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,kCAA9C,CAAN;AACD;;AACD,QAAMN,IAAI,GAAG,MAAMF,YAAY,CAC5B,oBAAmBH,YAAa,GAAEP,gBAAgB,CAACC,QAAD,EAAWC,OAAX,CAAoB,EAD1C,CAA/B;;AAGA,MAAI,CAACU,IAAD,IAAS,CAACQ,MAAM,CAACI,QAAP,CAAgBZ,IAAI,CAACC,EAArB,CAAd,EAAwC;AACtC,UAAM,IAAIrB,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,yCAA9C,CAAN;AACD;AACF;;AAED,MAAMO,qBAAqB,GAAG,OAAOC,KAAP,EAAcC,eAAd,EAA+BC,WAA/B,KAA+C;AAC3E,QAAMC,MAAM,GAAGlC,UAAU,CAAC;AACxBmC,IAAAA,OAAO,EAAG,GAAE/B,YAAa,iCADD;AAExBgC,IAAAA,KAAK,EAAE,IAFiB;AAGxBJ,IAAAA,eAHwB;AAIxBC,IAAAA;AAJwB,GAAD,CAAzB;AAOA,QAAMI,0BAA0B,GAAGpC,IAAI,CAACqC,SAAL,CAAeJ,MAAM,CAACK,aAAtB,CAAnC;AAEA,MAAIC,GAAJ;;AACA,MAAI;AACFA,IAAAA,GAAG,GAAG,MAAMH,0BAA0B,CAACN,KAAD,CAAtC;AACD,GAFD,CAEE,OAAOU,KAAP,EAAc;AACd,UAAM,IAAI5C,KAAK,CAACyB,KAAV,CACJzB,KAAK,CAACyB,KAAN,CAAYC,gBADR,EAEH,2CAA0CQ,KAAM,EAF7C,CAAN;AAID;;AACD,SAAOS,GAAP;AACD,CApBD;;AAsBA,MAAME,kBAAkB,GAAGC,KAAK,IAAI;AAClC,QAAMC,YAAY,GAAG1C,GAAG,CAAC2C,MAAJ,CAAWF,KAAX,EAAkB;AAAEG,IAAAA,QAAQ,EAAE;AAAZ,GAAlB,CAArB;;AACA,MAAI,CAACF,YAAL,EAAmB;AACjB,UAAM,IAAI/C,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,uCAA9C,CAAN;AACD;;AAED,SAAOqB,YAAY,CAACG,MAApB;AACD,CAPD;;AASA,MAAMC,aAAa,GAAG,OAAO;AAAEL,EAAAA,KAAF;AAASzB,EAAAA;AAAT,CAAP,EAAsB;AAAE+B,EAAAA,QAAF;AAAYjB,EAAAA,eAAZ;AAA6BC,EAAAA;AAA7B,CAAtB,KAAqE;AACzF,MAAI,CAACU,KAAL,EAAY;AACV,UAAM,IAAI9C,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,oCAA9C,CAAN;AACD;;AAED,QAAM;AAAE2B,IAAAA,GAAG,EAAEnB,KAAP;AAAcoB,IAAAA,GAAG,EAAEC;AAAnB,MAAiCV,kBAAkB,CAACC,KAAD,CAAzD;AACA,QAAMU,cAAc,GAAG,OAAvB;AACA,MAAIC,SAAJ;AAEArB,EAAAA,WAAW,GAAGA,WAAW,IAAIoB,cAA7B;AACArB,EAAAA,eAAe,GAAGA,eAAe,IAAI,CAArC;AAEA,QAAMuB,WAAW,GAAG,MAAMzB,qBAAqB,CAACC,KAAD,EAAQC,eAAR,EAAyBC,WAAzB,CAA/C;AACA,QAAMuB,UAAU,GAAGD,WAAW,CAACE,SAAZ,IAAyBF,WAAW,CAACG,YAAxD;;AAEA,MAAI;AACFJ,IAAAA,SAAS,GAAGpD,GAAG,CAACyD,MAAJ,CAAWhB,KAAX,EAAkBa,UAAlB,EAA8B;AACxCI,MAAAA,UAAU,EAAER,SAD4B;AAExC;AACAS,MAAAA,QAAQ,EAAEZ;AAH8B,KAA9B,CAAZ;AAKD,GAND,CAME,OAAOa,SAAP,EAAkB;AAClB,UAAMC,OAAO,GAAGD,SAAS,CAACC,OAA1B;AAEA,UAAM,IAAIlE,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA+C,GAAEwC,OAAQ,EAAzD,CAAN;AACD;;AAED,MAAIT,SAAS,CAACU,GAAV,KAAkB5D,YAAtB,EAAoC;AAClC,UAAM,IAAIP,KAAK,CAACyB,KAAV,CACJzB,KAAK,CAACyB,KAAN,CAAYC,gBADR,EAEH,8DAA6DnB,YAAa,YAAWkD,SAAS,CAACU,GAAI,EAFhG,CAAN;AAID;;AAED,MAAIV,SAAS,CAACW,GAAV,KAAkB/C,EAAtB,EAA0B;AACxB,UAAM,IAAIrB,KAAK,CAACyB,KAAV,CAAgBzB,KAAK,CAACyB,KAAN,CAAYC,gBAA5B,EAA8C,qCAA9C,CAAN;AACD;;AACD,SAAO+B,SAAP;AACD,CAtCD,C,CAwCA;;;AACA,SAASY,gBAAT,CAA0B5D,QAA1B,EAAoCC,OAApC,EAA6C;AAC3C,MAAID,QAAQ,CAACqC,KAAb,EAAoB;AAClB,WAAOK,aAAa,CAAC1C,QAAD,EAAWC,OAAX,CAApB;AACD,GAFD,MAEO;AACL,WAAOO,kBAAkB,CAACR,QAAD,EAAWC,OAAX,CAAzB;AACD;AACF,C,CAED;;;AACA,SAAS4D,aAAT,CAAuB1C,MAAvB,EAA+BnB,QAA/B,EAAyCC,OAAzC,EAAkD;AAChD,MAAID,QAAQ,CAACqC,KAAb,EAAoB;AAClB,WAAOyB,OAAO,CAACC,OAAR,EAAP;AACD,GAFD,MAEO;AACL,WAAO7C,kBAAkB,CAACC,MAAD,EAASnB,QAAT,EAAmBC,OAAnB,CAAzB;AACD;AACF,C,CAED;;;AACA,SAASQ,YAAT,CAAsBuD,IAAtB,EAA4B;AAC1B,SAAOnE,YAAY,CAACoE,GAAb,CAAiB,gCAAgCD,IAAjD,CAAP;AACD;;AAEDE,MAAM,CAACC,OAAP,GAAiB;AACfN,EAAAA,aAAa,EAAEA,aADA;AAEfD,EAAAA,gBAAgB,EAAEA;AAFH,CAAjB","sourcesContent":["// Helper functions for accessing the Facebook Graph API.\nconst Parse = require('parse/node').Parse;\nconst crypto = require('crypto');\nconst jwksClient = require('jwks-rsa');\nconst util = require('util');\nconst jwt = require('jsonwebtoken');\nconst httpsRequest = require('./httpsRequest');\n\nconst TOKEN_ISSUER = 'https://facebook.com';\n\nfunction getAppSecretPath(authData, options = {}) {\n  const appSecret = options.appSecret;\n  if (!appSecret) {\n    return '';\n  }\n  const appsecret_proof = crypto\n    .createHmac('sha256', appSecret)\n    .update(authData.access_token)\n    .digest('hex');\n\n  return `&appsecret_proof=${appsecret_proof}`;\n}\n\nfunction validateGraphToken(authData, options) {\n  return graphRequest(\n    'me?fields=id&access_token=' + authData.access_token + getAppSecretPath(authData, options)\n  ).then(data => {\n    if ((data && data.id == authData.id) || (process.env.TESTING && authData.id === 'test')) {\n      return;\n    }\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');\n  });\n}\n\nasync function validateGraphAppId(appIds, authData, options) {\n  var access_token = authData.access_token;\n  if (process.env.TESTING && access_token === 'test') {\n    return;\n  }\n  if (!Array.isArray(appIds)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'appIds must be an array.');\n  }\n  if (!appIds.length) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is not configured.');\n  }\n  const data = await graphRequest(\n    `app?access_token=${access_token}${getAppSecretPath(authData, options)}`\n  );\n  if (!data || !appIds.includes(data.id)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');\n  }\n}\n\nconst getFacebookKeyByKeyId = async (keyId, cacheMaxEntries, cacheMaxAge) => {\n  const client = jwksClient({\n    jwksUri: `${TOKEN_ISSUER}/.well-known/oauth/openid/jwks/`,\n    cache: true,\n    cacheMaxEntries,\n    cacheMaxAge,\n  });\n\n  const asyncGetSigningKeyFunction = util.promisify(client.getSigningKey);\n\n  let key;\n  try {\n    key = await asyncGetSigningKeyFunction(keyId);\n  } catch (error) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Unable to find matching key for Key ID: ${keyId}`\n    );\n  }\n  return key;\n};\n\nconst getHeaderFromToken = token => {\n  const decodedToken = jwt.decode(token, { complete: true });\n  if (!decodedToken) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'provided token does not decode as JWT');\n  }\n\n  return decodedToken.header;\n};\n\nconst verifyIdToken = async ({ token, id }, { clientId, cacheMaxEntries, cacheMaxAge }) => {\n  if (!token) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'id token is invalid for this user.');\n  }\n\n  const { kid: keyId, alg: algorithm } = getHeaderFromToken(token);\n  const ONE_HOUR_IN_MS = 3600000;\n  let jwtClaims;\n\n  cacheMaxAge = cacheMaxAge || ONE_HOUR_IN_MS;\n  cacheMaxEntries = cacheMaxEntries || 5;\n\n  const facebookKey = await getFacebookKeyByKeyId(keyId, cacheMaxEntries, cacheMaxAge);\n  const signingKey = facebookKey.publicKey || facebookKey.rsaPublicKey;\n\n  try {\n    jwtClaims = jwt.verify(token, signingKey, {\n      algorithms: algorithm,\n      // the audience can be checked against a string, a regular expression or a list of strings and/or regular expressions.\n      audience: clientId,\n    });\n  } catch (exception) {\n    const message = exception.message;\n\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);\n  }\n\n  if (jwtClaims.iss !== TOKEN_ISSUER) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `id token not issued by correct OpenID provider - expected: ${TOKEN_ISSUER} | from: ${jwtClaims.iss}`\n    );\n  }\n\n  if (jwtClaims.sub !== id) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'auth data is invalid for this user.');\n  }\n  return jwtClaims;\n};\n\n// Returns a promise that fulfills iff this user id is valid.\nfunction validateAuthData(authData, options) {\n  if (authData.token) {\n    return verifyIdToken(authData, options);\n  } else {\n    return validateGraphToken(authData, options);\n  }\n}\n\n// Returns a promise that fulfills iff this app id is valid.\nfunction validateAppId(appIds, authData, options) {\n  if (authData.token) {\n    return Promise.resolve();\n  } else {\n    return validateGraphAppId(appIds, authData, options);\n  }\n}\n\n// A promisey wrapper for FB graph requests.\nfunction graphRequest(path) {\n  return httpsRequest.get('https://graph.facebook.com/' + path);\n}\n\nmodule.exports = {\n  validateAppId: validateAppId,\n  validateAuthData: validateAuthData,\n};\n"]}
134
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","crypto","jwksClient","util","jwt","httpsRequest","TOKEN_ISSUER","getAppSecretPath","authData","options","appSecret","appsecret_proof","createHmac","update","access_token","digest","validateGraphToken","graphRequest","then","data","id","process","env","TESTING","Error","OBJECT_NOT_FOUND","validateGraphAppId","appIds","Array","isArray","length","includes","getFacebookKeyByKeyId","keyId","cacheMaxEntries","cacheMaxAge","client","jwksUri","cache","asyncGetSigningKeyFunction","promisify","getSigningKey","key","error","getHeaderFromToken","token","decodedToken","decode","complete","header","verifyIdToken","clientId","kid","alg","algorithm","ONE_HOUR_IN_MS","jwtClaims","facebookKey","signingKey","publicKey","rsaPublicKey","verify","algorithms","audience","exception","message","iss","sub","validateAuthData","validateAppId","Promise","resolve","path","get","module","exports"],"sources":["../../../src/Adapters/Auth/facebook.js"],"sourcesContent":["// Helper functions for accessing the Facebook Graph API.\nconst Parse = require('parse/node').Parse;\nconst crypto = require('crypto');\nconst jwksClient = require('jwks-rsa');\nconst util = require('util');\nconst jwt = require('jsonwebtoken');\nconst httpsRequest = require('./httpsRequest');\n\nconst TOKEN_ISSUER = 'https://facebook.com';\n\nfunction getAppSecretPath(authData, options = {}) {\n  const appSecret = options.appSecret;\n  if (!appSecret) {\n    return '';\n  }\n  const appsecret_proof = crypto\n    .createHmac('sha256', appSecret)\n    .update(authData.access_token)\n    .digest('hex');\n\n  return `&appsecret_proof=${appsecret_proof}`;\n}\n\nfunction validateGraphToken(authData, options) {\n  return graphRequest(\n    'me?fields=id&access_token=' + authData.access_token + getAppSecretPath(authData, options)\n  ).then(data => {\n    if ((data && data.id == authData.id) || (process.env.TESTING && authData.id === 'test')) {\n      return;\n    }\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');\n  });\n}\n\nasync function validateGraphAppId(appIds, authData, options) {\n  var access_token = authData.access_token;\n  if (process.env.TESTING && access_token === 'test') {\n    return;\n  }\n  if (!Array.isArray(appIds)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'appIds must be an array.');\n  }\n  if (!appIds.length) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is not configured.');\n  }\n  const data = await graphRequest(\n    `app?access_token=${access_token}${getAppSecretPath(authData, options)}`\n  );\n  if (!data || !appIds.includes(data.id)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Facebook auth is invalid for this user.');\n  }\n}\n\nconst getFacebookKeyByKeyId = async (keyId, cacheMaxEntries, cacheMaxAge) => {\n  const client = jwksClient({\n    jwksUri: `${TOKEN_ISSUER}/.well-known/oauth/openid/jwks/`,\n    cache: true,\n    cacheMaxEntries,\n    cacheMaxAge,\n  });\n\n  const asyncGetSigningKeyFunction = util.promisify(client.getSigningKey);\n\n  let key;\n  try {\n    key = await asyncGetSigningKeyFunction(keyId);\n  } catch (error) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `Unable to find matching key for Key ID: ${keyId}`\n    );\n  }\n  return key;\n};\n\nconst getHeaderFromToken = token => {\n  const decodedToken = jwt.decode(token, { complete: true });\n  if (!decodedToken) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'provided token does not decode as JWT');\n  }\n\n  return decodedToken.header;\n};\n\nconst verifyIdToken = async ({ token, id }, { clientId, cacheMaxEntries, cacheMaxAge }) => {\n  if (!token) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'id token is invalid for this user.');\n  }\n\n  const { kid: keyId, alg: algorithm } = getHeaderFromToken(token);\n  const ONE_HOUR_IN_MS = 3600000;\n  let jwtClaims;\n\n  cacheMaxAge = cacheMaxAge || ONE_HOUR_IN_MS;\n  cacheMaxEntries = cacheMaxEntries || 5;\n\n  const facebookKey = await getFacebookKeyByKeyId(keyId, cacheMaxEntries, cacheMaxAge);\n  const signingKey = facebookKey.publicKey || facebookKey.rsaPublicKey;\n\n  try {\n    jwtClaims = jwt.verify(token, signingKey, {\n      algorithms: algorithm,\n      // the audience can be checked against a string, a regular expression or a list of strings and/or regular expressions.\n      audience: clientId,\n    });\n  } catch (exception) {\n    const message = exception.message;\n\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${message}`);\n  }\n\n  if (jwtClaims.iss !== TOKEN_ISSUER) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `id token not issued by correct OpenID provider - expected: ${TOKEN_ISSUER} | from: ${jwtClaims.iss}`\n    );\n  }\n\n  if (jwtClaims.sub !== id) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'auth data is invalid for this user.');\n  }\n  return jwtClaims;\n};\n\n// Returns a promise that fulfills iff this user id is valid.\nfunction validateAuthData(authData, options) {\n  if (authData.token) {\n    return verifyIdToken(authData, options);\n  } else {\n    return validateGraphToken(authData, options);\n  }\n}\n\n// Returns a promise that fulfills iff this app id is valid.\nfunction validateAppId(appIds, authData, options) {\n  if (authData.token) {\n    return Promise.resolve();\n  } else {\n    return validateGraphAppId(appIds, authData, options);\n  }\n}\n\n// A promisey wrapper for FB graph requests.\nfunction graphRequest(path) {\n  return httpsRequest.get('https://graph.facebook.com/' + path);\n}\n\nmodule.exports = {\n  validateAppId: validateAppId,\n  validateAuthData: validateAuthData,\n};\n"],"mappings":"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"}
@@ -12,18 +12,15 @@ const authData = {
12
12
  id: 'playerId',
13
13
  };
14
14
  */
15
+
15
16
  const {
16
17
  Parse
17
18
  } = require('parse/node');
18
-
19
19
  const crypto = require('crypto');
20
-
21
20
  const https = require('https');
22
-
23
21
  const {
24
22
  pki
25
23
  } = require('node-forge');
26
-
27
24
  const ca = {
28
25
  cert: null,
29
26
  url: null
@@ -38,7 +35,6 @@ function verifyPublicKeyUrl(publicKeyUrl) {
38
35
  return false;
39
36
  }
40
37
  }
41
-
42
38
  function convertX509CertToPEM(X509Cert) {
43
39
  const pemPreFix = '-----BEGIN CERTIFICATE-----\n';
44
40
  const pemPostFix = '-----END CERTIFICATE-----';
@@ -46,16 +42,13 @@ function convertX509CertToPEM(X509Cert) {
46
42
  const certBody = base64.match(new RegExp('.{0,64}', 'g')).join('\n');
47
43
  return pemPreFix + certBody + pemPostFix;
48
44
  }
49
-
50
45
  async function getAppleCertificate(publicKeyUrl) {
51
46
  if (!verifyPublicKeyUrl(publicKeyUrl)) {
52
47
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
53
48
  }
54
-
55
49
  if (cache[publicKeyUrl]) {
56
50
  return cache[publicKeyUrl];
57
51
  }
58
-
59
52
  const url = new URL(publicKeyUrl);
60
53
  const headOptions = {
61
54
  hostname: url.hostname,
@@ -64,31 +57,25 @@ async function getAppleCertificate(publicKeyUrl) {
64
57
  };
65
58
  const cert_headers = await new Promise((resolve, reject) => https.get(headOptions, res => resolve(res.headers)).on('error', reject));
66
59
  const validContentTypes = ['application/x-x509-ca-cert', 'application/pkix-cert'];
67
-
68
60
  if (!validContentTypes.includes(cert_headers['content-type']) || cert_headers['content-length'] == null || cert_headers['content-length'] > 10000) {
69
61
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
70
62
  }
71
-
72
63
  const {
73
64
  certificate,
74
65
  headers
75
66
  } = await getCertificate(publicKeyUrl);
76
-
77
67
  if (headers['cache-control']) {
78
68
  const expire = headers['cache-control'].match(/max-age=([0-9]+)/);
79
-
80
69
  if (expire) {
81
- cache[publicKeyUrl] = certificate; // we'll expire the cache entry later, as per max-age
82
-
70
+ cache[publicKeyUrl] = certificate;
71
+ // we'll expire the cache entry later, as per max-age
83
72
  setTimeout(() => {
84
73
  delete cache[publicKeyUrl];
85
74
  }, parseInt(expire[1], 10) * 1000);
86
75
  }
87
76
  }
88
-
89
77
  return verifyPublicKeyIssuer(certificate, publicKeyUrl);
90
78
  }
91
-
92
79
  function getCertificate(url, buffer) {
93
80
  return new Promise((resolve, reject) => {
94
81
  https.get(url, res => {
@@ -104,13 +91,10 @@ function getCertificate(url, buffer) {
104
91
  });
105
92
  return;
106
93
  }
107
-
108
94
  let cert = '';
109
-
110
95
  for (const chunk of data) {
111
96
  cert += chunk.toString('base64');
112
97
  }
113
-
114
98
  const certificate = convertX509CertToPEM(cert);
115
99
  resolve({
116
100
  certificate,
@@ -120,7 +104,6 @@ function getCertificate(url, buffer) {
120
104
  }).on('error', reject);
121
105
  });
122
106
  }
123
-
124
107
  function convertTimestampToBigEndian(timestamp) {
125
108
  const buffer = Buffer.alloc(8);
126
109
  const high = ~~(timestamp / 0xffffffff);
@@ -129,26 +112,21 @@ function convertTimestampToBigEndian(timestamp) {
129
112
  buffer.writeUInt32BE(parseInt(low, 10), 4);
130
113
  return buffer;
131
114
  }
132
-
133
115
  function verifySignature(publicKey, authData) {
134
116
  const verifier = crypto.createVerify('sha256');
135
117
  verifier.update(authData.playerId, 'utf8');
136
118
  verifier.update(authData.bundleId, 'utf8');
137
119
  verifier.update(convertTimestampToBigEndian(authData.timestamp));
138
120
  verifier.update(authData.salt, 'base64');
139
-
140
121
  if (!verifier.verify(publicKey, authData.signature, 'base64')) {
141
122
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - invalid signature');
142
123
  }
143
124
  }
144
-
145
125
  function verifyPublicKeyIssuer(cert, publicKeyUrl) {
146
126
  const publicKeyCert = pki.certificateFromPem(cert);
147
-
148
127
  if (!ca.cert) {
149
128
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.');
150
129
  }
151
-
152
130
  try {
153
131
  if (!ca.cert.verify(publicKeyCert)) {
154
132
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
@@ -156,47 +134,40 @@ function verifyPublicKeyIssuer(cert, publicKeyUrl) {
156
134
  } catch (e) {
157
135
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Apple Game Center - invalid publicKeyUrl: ${publicKeyUrl}`);
158
136
  }
159
-
160
137
  return cert;
161
- } // Returns a promise that fulfills if this user id is valid.
162
-
138
+ }
163
139
 
140
+ // Returns a promise that fulfills if this user id is valid.
164
141
  async function validateAuthData(authData) {
165
142
  if (!authData.id) {
166
143
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center - authData id missing');
167
144
  }
168
-
169
145
  authData.playerId = authData.id;
170
146
  const publicKey = await getAppleCertificate(authData.publicKeyUrl);
171
147
  return verifySignature(publicKey, authData);
172
- } // Returns a promise that fulfills if this app id is valid.
173
-
148
+ }
174
149
 
150
+ // Returns a promise that fulfills if this app id is valid.
175
151
  async function validateAppId(appIds, authData, options = {}) {
176
152
  if (!options.rootCertificateUrl) {
177
153
  options.rootCertificateUrl = 'https://cacerts.digicert.com/DigiCertTrustedG4CodeSigningRSA4096SHA3842021CA1.crt.pem';
178
154
  }
179
-
180
155
  if (ca.url === options.rootCertificateUrl) {
181
156
  return;
182
157
  }
183
-
184
158
  const {
185
159
  certificate,
186
160
  headers
187
161
  } = await getCertificate(options.rootCertificateUrl, true);
188
-
189
162
  if (headers['content-type'] !== 'application/x-pem-file' || headers['content-length'] == null || headers['content-length'] > 10000) {
190
163
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Apple Game Center auth adapter parameter `rootCertificateURL` is invalid.');
191
164
  }
192
-
193
165
  ca.cert = pki.certificateFromPem(certificate);
194
166
  ca.url = options.rootCertificateUrl;
195
167
  }
196
-
197
168
  module.exports = {
198
169
  validateAppId,
199
170
  validateAuthData,
200
171
  cache
201
172
  };
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"]}
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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"}
@@ -2,26 +2,24 @@
2
2
 
3
3
  // Helper functions for accessing the github API.
4
4
  var Parse = require('parse/node').Parse;
5
+ const httpsRequest = require('./httpsRequest');
5
6
 
6
- const httpsRequest = require('./httpsRequest'); // Returns a promise that fulfills iff this user id is valid.
7
-
8
-
7
+ // Returns a promise that fulfills iff this user id is valid.
9
8
  function validateAuthData(authData) {
10
9
  return request('user', authData.access_token).then(data => {
11
10
  if (data && data.id == authData.id) {
12
11
  return;
13
12
  }
14
-
15
13
  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Github auth is invalid for this user.');
16
14
  });
17
- } // Returns a promise that fulfills iff this app id is valid.
18
-
15
+ }
19
16
 
17
+ // Returns a promise that fulfills iff this app id is valid.
20
18
  function validateAppId() {
21
19
  return Promise.resolve();
22
- } // A promisey wrapper for api requests
23
-
20
+ }
24
21
 
22
+ // A promisey wrapper for api requests
25
23
  function request(path, access_token) {
26
24
  return httpsRequest.get({
27
25
  host: 'api.github.com',
@@ -32,9 +30,8 @@ function request(path, access_token) {
32
30
  }
33
31
  });
34
32
  }
35
-
36
33
  module.exports = {
37
34
  validateAppId: validateAppId,
38
35
  validateAuthData: validateAuthData
39
36
  };
40
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BZGFwdGVycy9BdXRoL2dpdGh1Yi5qcyJdLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJodHRwc1JlcXVlc3QiLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwidGhlbiIsImRhdGEiLCJpZCIsIkVycm9yIiwiT0JKRUNUX05PVF9GT1VORCIsInZhbGlkYXRlQXBwSWQiLCJQcm9taXNlIiwicmVzb2x2ZSIsInBhdGgiLCJnZXQiLCJob3N0IiwiaGVhZGVycyIsIkF1dGhvcml6YXRpb24iLCJtb2R1bGUiLCJleHBvcnRzIl0sIm1hcHBpbmdzIjoiOztBQUFBO0FBQ0EsSUFBSUEsS0FBSyxHQUFHQyxPQUFPLENBQUMsWUFBRCxDQUFQLENBQXNCRCxLQUFsQzs7QUFDQSxNQUFNRSxZQUFZLEdBQUdELE9BQU8sQ0FBQyxnQkFBRCxDQUE1QixDLENBRUE7OztBQUNBLFNBQVNFLGdCQUFULENBQTBCQyxRQUExQixFQUFvQztBQUNsQyxTQUFPQyxPQUFPLENBQUMsTUFBRCxFQUFTRCxRQUFRLENBQUNFLFlBQWxCLENBQVAsQ0FBdUNDLElBQXZDLENBQTRDQyxJQUFJLElBQUk7QUFDekQsUUFBSUEsSUFBSSxJQUFJQSxJQUFJLENBQUNDLEVBQUwsSUFBV0wsUUFBUSxDQUFDSyxFQUFoQyxFQUFvQztBQUNsQztBQUNEOztBQUNELFVBQU0sSUFBSVQsS0FBSyxDQUFDVSxLQUFWLENBQWdCVixLQUFLLENBQUNVLEtBQU4sQ0FBWUMsZ0JBQTVCLEVBQThDLHVDQUE5QyxDQUFOO0FBQ0QsR0FMTSxDQUFQO0FBTUQsQyxDQUVEOzs7QUFDQSxTQUFTQyxhQUFULEdBQXlCO0FBQ3ZCLFNBQU9DLE9BQU8sQ0FBQ0MsT0FBUixFQUFQO0FBQ0QsQyxDQUVEOzs7QUFDQSxTQUFTVCxPQUFULENBQWlCVSxJQUFqQixFQUF1QlQsWUFBdkIsRUFBcUM7QUFDbkMsU0FBT0osWUFBWSxDQUFDYyxHQUFiLENBQWlCO0FBQ3RCQyxJQUFBQSxJQUFJLEVBQUUsZ0JBRGdCO0FBRXRCRixJQUFBQSxJQUFJLEVBQUUsTUFBTUEsSUFGVTtBQUd0QkcsSUFBQUEsT0FBTyxFQUFFO0FBQ1BDLE1BQUFBLGFBQWEsRUFBRSxZQUFZYixZQURwQjtBQUVQLG9CQUFjO0FBRlA7QUFIYSxHQUFqQixDQUFQO0FBUUQ7O0FBRURjLE1BQU0sQ0FBQ0MsT0FBUCxHQUFpQjtBQUNmVCxFQUFBQSxhQUFhLEVBQUVBLGFBREE7QUFFZlQsRUFBQUEsZ0JBQWdCLEVBQUVBO0FBRkgsQ0FBakIiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBIZWxwZXIgZnVuY3Rpb25zIGZvciBhY2Nlc3NpbmcgdGhlIGdpdGh1YiBBUEkuXG52YXIgUGFyc2UgPSByZXF1aXJlKCdwYXJzZS9ub2RlJykuUGFyc2U7XG5jb25zdCBodHRwc1JlcXVlc3QgPSByZXF1aXJlKCcuL2h0dHBzUmVxdWVzdCcpO1xuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIHVzZXIgaWQgaXMgdmFsaWQuXG5mdW5jdGlvbiB2YWxpZGF0ZUF1dGhEYXRhKGF1dGhEYXRhKSB7XG4gIHJldHVybiByZXF1ZXN0KCd1c2VyJywgYXV0aERhdGEuYWNjZXNzX3Rva2VuKS50aGVuKGRhdGEgPT4ge1xuICAgIGlmIChkYXRhICYmIGRhdGEuaWQgPT0gYXV0aERhdGEuaWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhyb3cgbmV3IFBhcnNlLkVycm9yKFBhcnNlLkVycm9yLk9CSkVDVF9OT1RfRk9VTkQsICdHaXRodWIgYXV0aCBpcyBpbnZhbGlkIGZvciB0aGlzIHVzZXIuJyk7XG4gIH0pO1xufVxuXG4vLyBSZXR1cm5zIGEgcHJvbWlzZSB0aGF0IGZ1bGZpbGxzIGlmZiB0aGlzIGFwcCBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXBwSWQoKSB7XG4gIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbn1cblxuLy8gQSBwcm9taXNleSB3cmFwcGVyIGZvciBhcGkgcmVxdWVzdHNcbmZ1bmN0aW9uIHJlcXVlc3QocGF0aCwgYWNjZXNzX3Rva2VuKSB7XG4gIHJldHVybiBodHRwc1JlcXVlc3QuZ2V0KHtcbiAgICBob3N0OiAnYXBpLmdpdGh1Yi5jb20nLFxuICAgIHBhdGg6ICcvJyArIHBhdGgsXG4gICAgaGVhZGVyczoge1xuICAgICAgQXV0aG9yaXphdGlvbjogJ2JlYXJlciAnICsgYWNjZXNzX3Rva2VuLFxuICAgICAgJ1VzZXItQWdlbnQnOiAncGFyc2Utc2VydmVyJyxcbiAgICB9LFxuICB9KTtcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gIHZhbGlkYXRlQXBwSWQ6IHZhbGlkYXRlQXBwSWQsXG4gIHZhbGlkYXRlQXV0aERhdGE6IHZhbGlkYXRlQXV0aERhdGEsXG59O1xuIl19
37
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJQYXJzZSIsInJlcXVpcmUiLCJodHRwc1JlcXVlc3QiLCJ2YWxpZGF0ZUF1dGhEYXRhIiwiYXV0aERhdGEiLCJyZXF1ZXN0IiwiYWNjZXNzX3Rva2VuIiwidGhlbiIsImRhdGEiLCJpZCIsIkVycm9yIiwiT0JKRUNUX05PVF9GT1VORCIsInZhbGlkYXRlQXBwSWQiLCJQcm9taXNlIiwicmVzb2x2ZSIsInBhdGgiLCJnZXQiLCJob3N0IiwiaGVhZGVycyIsIkF1dGhvcml6YXRpb24iLCJtb2R1bGUiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL0FkYXB0ZXJzL0F1dGgvZ2l0aHViLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEhlbHBlciBmdW5jdGlvbnMgZm9yIGFjY2Vzc2luZyB0aGUgZ2l0aHViIEFQSS5cbnZhciBQYXJzZSA9IHJlcXVpcmUoJ3BhcnNlL25vZGUnKS5QYXJzZTtcbmNvbnN0IGh0dHBzUmVxdWVzdCA9IHJlcXVpcmUoJy4vaHR0cHNSZXF1ZXN0Jyk7XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgdXNlciBpZCBpcyB2YWxpZC5cbmZ1bmN0aW9uIHZhbGlkYXRlQXV0aERhdGEoYXV0aERhdGEpIHtcbiAgcmV0dXJuIHJlcXVlc3QoJ3VzZXInLCBhdXRoRGF0YS5hY2Nlc3NfdG9rZW4pLnRoZW4oZGF0YSA9PiB7XG4gICAgaWYgKGRhdGEgJiYgZGF0YS5pZCA9PSBhdXRoRGF0YS5pZCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICB0aHJvdyBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuT0JKRUNUX05PVF9GT1VORCwgJ0dpdGh1YiBhdXRoIGlzIGludmFsaWQgZm9yIHRoaXMgdXNlci4nKTtcbiAgfSk7XG59XG5cbi8vIFJldHVybnMgYSBwcm9taXNlIHRoYXQgZnVsZmlsbHMgaWZmIHRoaXMgYXBwIGlkIGlzIHZhbGlkLlxuZnVuY3Rpb24gdmFsaWRhdGVBcHBJZCgpIHtcbiAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpO1xufVxuXG4vLyBBIHByb21pc2V5IHdyYXBwZXIgZm9yIGFwaSByZXF1ZXN0c1xuZnVuY3Rpb24gcmVxdWVzdChwYXRoLCBhY2Nlc3NfdG9rZW4pIHtcbiAgcmV0dXJuIGh0dHBzUmVxdWVzdC5nZXQoe1xuICAgIGhvc3Q6ICdhcGkuZ2l0aHViLmNvbScsXG4gICAgcGF0aDogJy8nICsgcGF0aCxcbiAgICBoZWFkZXJzOiB7XG4gICAgICBBdXRob3JpemF0aW9uOiAnYmVhcmVyICcgKyBhY2Nlc3NfdG9rZW4sXG4gICAgICAnVXNlci1BZ2VudCc6ICdwYXJzZS1zZXJ2ZXInLFxuICAgIH0sXG4gIH0pO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgdmFsaWRhdGVBcHBJZDogdmFsaWRhdGVBcHBJZCxcbiAgdmFsaWRhdGVBdXRoRGF0YTogdmFsaWRhdGVBdXRoRGF0YSxcbn07XG4iXSwibWFwcGluZ3MiOiI7O0FBQUE7QUFDQSxJQUFJQSxLQUFLLEdBQUdDLE9BQU8sQ0FBQyxZQUFZLENBQUMsQ0FBQ0QsS0FBSztBQUN2QyxNQUFNRSxZQUFZLEdBQUdELE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQzs7QUFFOUM7QUFDQSxTQUFTRSxnQkFBZ0IsQ0FBQ0MsUUFBUSxFQUFFO0VBQ2xDLE9BQU9DLE9BQU8sQ0FBQyxNQUFNLEVBQUVELFFBQVEsQ0FBQ0UsWUFBWSxDQUFDLENBQUNDLElBQUksQ0FBQ0MsSUFBSSxJQUFJO0lBQ3pELElBQUlBLElBQUksSUFBSUEsSUFBSSxDQUFDQyxFQUFFLElBQUlMLFFBQVEsQ0FBQ0ssRUFBRSxFQUFFO01BQ2xDO0lBQ0Y7SUFDQSxNQUFNLElBQUlULEtBQUssQ0FBQ1UsS0FBSyxDQUFDVixLQUFLLENBQUNVLEtBQUssQ0FBQ0MsZ0JBQWdCLEVBQUUsdUNBQXVDLENBQUM7RUFDOUYsQ0FBQyxDQUFDO0FBQ0o7O0FBRUE7QUFDQSxTQUFTQyxhQUFhLEdBQUc7RUFDdkIsT0FBT0MsT0FBTyxDQUFDQyxPQUFPLEVBQUU7QUFDMUI7O0FBRUE7QUFDQSxTQUFTVCxPQUFPLENBQUNVLElBQUksRUFBRVQsWUFBWSxFQUFFO0VBQ25DLE9BQU9KLFlBQVksQ0FBQ2MsR0FBRyxDQUFDO0lBQ3RCQyxJQUFJLEVBQUUsZ0JBQWdCO0lBQ3RCRixJQUFJLEVBQUUsR0FBRyxHQUFHQSxJQUFJO0lBQ2hCRyxPQUFPLEVBQUU7TUFDUEMsYUFBYSxFQUFFLFNBQVMsR0FBR2IsWUFBWTtNQUN2QyxZQUFZLEVBQUU7SUFDaEI7RUFDRixDQUFDLENBQUM7QUFDSjtBQUVBYyxNQUFNLENBQUNDLE9BQU8sR0FBRztFQUNmVCxhQUFhLEVBQUVBLGFBQWE7RUFDNUJULGdCQUFnQixFQUFFQTtBQUNwQixDQUFDIn0=