parse-server 9.5.2-alpha.3 → 9.5.2-alpha.5
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.
|
@@ -71,7 +71,9 @@
|
|
|
71
71
|
const {
|
|
72
72
|
Parse
|
|
73
73
|
} = require('parse/node');
|
|
74
|
-
const
|
|
74
|
+
const jwksClient = require('jwks-rsa');
|
|
75
|
+
const jwt = require('jsonwebtoken');
|
|
76
|
+
const authUtils = require('./utils');
|
|
75
77
|
const arraysEqual = (_arr1, _arr2) => {
|
|
76
78
|
if (!Array.isArray(_arr1) || !Array.isArray(_arr2) || _arr1.length !== _arr2.length) {
|
|
77
79
|
return false;
|
|
@@ -85,13 +87,30 @@ const arraysEqual = (_arr1, _arr2) => {
|
|
|
85
87
|
}
|
|
86
88
|
return true;
|
|
87
89
|
};
|
|
88
|
-
const
|
|
90
|
+
const getKeycloakKeyByKeyId = async (keyId, jwksUri, cacheMaxEntries, cacheMaxAge) => {
|
|
91
|
+
const client = jwksClient({
|
|
92
|
+
jwksUri,
|
|
93
|
+
cache: true,
|
|
94
|
+
cacheMaxEntries,
|
|
95
|
+
cacheMaxAge
|
|
96
|
+
});
|
|
97
|
+
let key;
|
|
98
|
+
try {
|
|
99
|
+
key = await authUtils.getSigningKey(client, keyId);
|
|
100
|
+
} catch {
|
|
101
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `Unable to find matching key for Key ID: ${keyId}`);
|
|
102
|
+
}
|
|
103
|
+
return key;
|
|
104
|
+
};
|
|
105
|
+
const verifyAccessToken = async ({
|
|
89
106
|
access_token,
|
|
90
107
|
id,
|
|
91
108
|
roles,
|
|
92
109
|
groups
|
|
93
110
|
} = {}, {
|
|
94
|
-
config
|
|
111
|
+
config,
|
|
112
|
+
cacheMaxEntries,
|
|
113
|
+
cacheMaxAge
|
|
95
114
|
} = {}) => {
|
|
96
115
|
if (!(access_token && id)) {
|
|
97
116
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing access token and/or User id');
|
|
@@ -99,45 +118,46 @@ const handleAuth = async ({
|
|
|
99
118
|
if (!config || !(config['auth-server-url'] && config['realm'])) {
|
|
100
119
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing keycloak configuration');
|
|
101
120
|
}
|
|
121
|
+
if (!config['client-id']) {
|
|
122
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Keycloak auth is not configured. Missing client-id.');
|
|
123
|
+
}
|
|
124
|
+
const expectedIssuer = `${config['auth-server-url']}/realms/${config['realm']}`;
|
|
125
|
+
const jwksUri = `${config['auth-server-url']}/realms/${config['realm']}/protocol/openid-connect/certs`;
|
|
126
|
+
const {
|
|
127
|
+
kid: keyId
|
|
128
|
+
} = authUtils.getHeaderFromToken(access_token);
|
|
129
|
+
const ONE_HOUR_IN_MS = 3600000;
|
|
130
|
+
cacheMaxAge = cacheMaxAge || ONE_HOUR_IN_MS;
|
|
131
|
+
cacheMaxEntries = cacheMaxEntries || 5;
|
|
132
|
+
const keycloakKey = await getKeycloakKeyByKeyId(keyId, jwksUri, cacheMaxEntries, cacheMaxAge);
|
|
133
|
+
const signingKey = keycloakKey.publicKey || keycloakKey.rsaPublicKey;
|
|
134
|
+
let jwtClaims;
|
|
102
135
|
try {
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
path: `/realms/${config['realm']}/protocol/openid-connect/userinfo`,
|
|
106
|
-
headers: {
|
|
107
|
-
Authorization: 'Bearer ' + access_token
|
|
108
|
-
}
|
|
136
|
+
jwtClaims = jwt.verify(access_token, signingKey, {
|
|
137
|
+
algorithms: ['RS256']
|
|
109
138
|
});
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
139
|
+
} catch (exception) {
|
|
140
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${exception.message}`);
|
|
141
|
+
}
|
|
142
|
+
if (jwtClaims.iss !== expectedIssuer) {
|
|
143
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `access token not issued by correct provider - expected: ${expectedIssuer} | from: ${jwtClaims.iss}`);
|
|
144
|
+
}
|
|
145
|
+
if (jwtClaims.azp !== config['client-id']) {
|
|
146
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `access token is not authorized for this client - expected: ${config['client-id']} | from: ${jwtClaims.azp}`);
|
|
147
|
+
}
|
|
148
|
+
if (jwtClaims.sub !== id) {
|
|
149
|
+
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'auth data is invalid for this user.');
|
|
150
|
+
}
|
|
151
|
+
const rolesMatch = jwtClaims.roles === roles || arraysEqual(jwtClaims.roles, roles);
|
|
152
|
+
const groupsMatch = jwtClaims.groups === groups || arraysEqual(jwtClaims.groups, groups);
|
|
153
|
+
if (!rolesMatch || !groupsMatch) {
|
|
113
154
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid authentication');
|
|
114
|
-
} catch (e) {
|
|
115
|
-
if (e instanceof Parse.Error) {
|
|
116
|
-
throw e;
|
|
117
|
-
}
|
|
118
|
-
const error = JSON.parse(e.text);
|
|
119
|
-
if (error.error_description) {
|
|
120
|
-
throw new Parse.Error(Parse.Error.HOSTING_ERROR, error.error_description);
|
|
121
|
-
} else {
|
|
122
|
-
throw new Parse.Error(Parse.Error.HOSTING_ERROR, 'Could not connect to the authentication server');
|
|
123
|
-
}
|
|
124
155
|
}
|
|
156
|
+
return jwtClaims;
|
|
125
157
|
};
|
|
126
|
-
|
|
127
|
-
/*
|
|
128
|
-
@param {Object} authData: the client provided authData
|
|
129
|
-
@param {string} authData.access_token: the access_token retrieved from client authentication in Keycloak
|
|
130
|
-
@param {string} authData.id: the id retrieved from client authentication in Keycloak
|
|
131
|
-
@param {Array} authData.roles: the roles retrieved from client authentication in Keycloak
|
|
132
|
-
@param {Array} authData.groups: the groups retrieved from client authentication in Keycloak
|
|
133
|
-
@param {Object} options: additional options
|
|
134
|
-
@param {Object} options.config: the config object passed during Parse Server instantiation
|
|
135
|
-
*/
|
|
136
158
|
function validateAuthData(authData, options = {}) {
|
|
137
|
-
return
|
|
159
|
+
return verifyAccessToken(authData, options);
|
|
138
160
|
}
|
|
139
|
-
|
|
140
|
-
// Returns a promise that fulfills if this app id is valid.
|
|
141
161
|
function validateAppId() {
|
|
142
162
|
return Promise.resolve();
|
|
143
163
|
}
|
|
@@ -145,4 +165,4 @@ module.exports = {
|
|
|
145
165
|
validateAppId,
|
|
146
166
|
validateAuthData
|
|
147
167
|
};
|
|
148
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","httpsRequest","arraysEqual","_arr1","_arr2","Array","isArray","length","arr1","concat","sort","arr2","i","handleAuth","access_token","id","roles","groups","config","Error","OBJECT_NOT_FOUND","response","get","host","path","headers","Authorization","data","sub","e","error","JSON","parse","text","error_description","HOSTING_ERROR","validateAuthData","authData","options","validateAppId","Promise","resolve","module","exports"],"sources":["../../../src/Adapters/Auth/keycloak.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Keycloak.\n *\n * @class KeycloakAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {Object} options.config - The Keycloak configuration object, typically loaded from a JSON file.\n * @param {String} options.config.auth-server-url - The Keycloak authentication server URL.\n * @param {String} options.config.realm - The Keycloak realm name.\n * @param {String} options.config.client-id - The Keycloak client ID.\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {String} authData.access_token - The Keycloak access token retrieved during client authentication.\n * @param {String} authData.id - The user ID retrieved from Keycloak during client authentication.\n * @param {Array} [authData.roles] - The roles assigned to the user in Keycloak (optional).\n * @param {Array} [authData.groups] - The groups assigned to the user in Keycloak (optional).\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for Keycloak authentication, use the following structure:\n * ```javascript\n * {\n *   \"auth\": {\n *     \"keycloak\": {\n *       \"config\": require('./auth/keycloak.json')\n *     }\n *   }\n * }\n * ```\n * Ensure the `keycloak.json` configuration file is generated from Keycloak's setup guide and includes:\n * - `auth-server-url`: The Keycloak authentication server URL.\n * - `realm`: The Keycloak realm name.\n * - `client-id`: The Keycloak client ID.\n *\n * ## Auth Data\n * The adapter requires the following `authData` fields:\n * - `access_token`: The Keycloak access token retrieved during client authentication.\n * - `id`: The user ID retrieved from Keycloak during client authentication.\n * - `roles` (optional): The roles assigned to the user in Keycloak.\n * - `groups` (optional): The groups assigned to the user in Keycloak.\n *\n * ## Auth Payload Example\n * ### Example Auth Data\n * ```json\n * {\n *   \"keycloak\": {\n *     \"access_token\": \"an authorized Keycloak access token for the user\",\n *     \"id\": \"user's Keycloak ID as a string\",\n *     \"roles\": [\"admin\", \"user\"],\n *     \"groups\": [\"group1\", \"group2\"]\n *   }\n * }\n * ```\n *\n * ## Notes\n * - Parse Server validates the provided `authData` by making a `userinfo` call to Keycloak and ensures the attributes match those returned by Keycloak.\n *\n * ## Keycloak Configuration\n * To configure Keycloak, copy the JSON configuration file generated from Keycloak's setup guide:\n * - [Keycloak Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/index.html#_javascript_adapter)\n *\n * Place the configuration file on your server, for example:\n * - `auth/keycloak.json`\n *\n * For more information on Keycloak authentication, see:\n * - [Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/)\n * - [Server Administration Documentation](https://www.keycloak.org/docs/latest/server_admin/)\n */\n\nconst { Parse } = require('parse/node');\nconst httpsRequest = require('./httpsRequest');\n\nconst arraysEqual = (_arr1, _arr2) => {\n  if (!Array.isArray(_arr1) || !Array.isArray(_arr2) || _arr1.length !== _arr2.length) { return false; }\n\n  var arr1 = _arr1.concat().sort();\n  var arr2 = _arr2.concat().sort();\n\n  for (var i = 0; i < arr1.length; i++) {\n    if (arr1[i] !== arr2[i]) { return false; }\n  }\n\n  return true;\n};\n\nconst handleAuth = async ({ access_token, id, roles, groups } = {}, { config } = {}) => {\n  if (!(access_token && id)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing access token and/or User id');\n  }\n  if (!config || !(config['auth-server-url'] && config['realm'])) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing keycloak configuration');\n  }\n  try {\n    const response = await httpsRequest.get({\n      host: config['auth-server-url'],\n      path: `/realms/${config['realm']}/protocol/openid-connect/userinfo`,\n      headers: {\n        Authorization: 'Bearer ' + access_token,\n      },\n    });\n    if (\n      response &&\n      response.data &&\n      response.data.sub == id &&\n      arraysEqual(response.data.roles, roles) &&\n      arraysEqual(response.data.groups, groups)\n    ) {\n      return;\n    }\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid authentication');\n  } catch (e) {\n    if (e instanceof Parse.Error) {\n      throw e;\n    }\n    const error = JSON.parse(e.text);\n    if (error.error_description) {\n      throw new Parse.Error(Parse.Error.HOSTING_ERROR, error.error_description);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.HOSTING_ERROR,\n        'Could not connect to the authentication server'\n      );\n    }\n  }\n};\n\n/*\n  @param {Object} authData: the client provided authData\n  @param {string} authData.access_token: the access_token retrieved from client authentication in Keycloak\n  @param {string} authData.id: the id retrieved from client authentication in Keycloak\n  @param {Array}  authData.roles: the roles retrieved from client authentication in Keycloak\n  @param {Array}  authData.groups: the groups retrieved from client authentication in Keycloak\n  @param {Object} options: additional options\n  @param {Object} options.config: the config object passed during Parse Server instantiation\n*/\nfunction validateAuthData(authData, options = {}) {\n  return handleAuth(authData, options);\n}\n\n// Returns a promise that fulfills if this app id is valid.\nfunction validateAppId() {\n  return Promise.resolve();\n}\n\nmodule.exports = {\n  validateAppId,\n  validateAuthData,\n};\n"],"mappings":"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","ignoreList":[]}
|
|
168
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","jwksClient","jwt","authUtils","arraysEqual","_arr1","_arr2","Array","isArray","length","arr1","concat","sort","arr2","i","getKeycloakKeyByKeyId","keyId","jwksUri","cacheMaxEntries","cacheMaxAge","client","cache","key","getSigningKey","Error","OBJECT_NOT_FOUND","verifyAccessToken","access_token","id","roles","groups","config","expectedIssuer","kid","getHeaderFromToken","ONE_HOUR_IN_MS","keycloakKey","signingKey","publicKey","rsaPublicKey","jwtClaims","verify","algorithms","exception","message","iss","azp","sub","rolesMatch","groupsMatch","validateAuthData","authData","options","validateAppId","Promise","resolve","module","exports"],"sources":["../../../src/Adapters/Auth/keycloak.js"],"sourcesContent":["/**\n * Parse Server authentication adapter for Keycloak.\n *\n * @class KeycloakAdapter\n * @param {Object} options - The adapter configuration options.\n * @param {Object} options.config - The Keycloak configuration object, typically loaded from a JSON file.\n * @param {String} options.config.auth-server-url - The Keycloak authentication server URL.\n * @param {String} options.config.realm - The Keycloak realm name.\n * @param {String} options.config.client-id - The Keycloak client ID.\n *\n * @param {Object} authData - The authentication data provided by the client.\n * @param {String} authData.access_token - The Keycloak access token retrieved during client authentication.\n * @param {String} authData.id - The user ID retrieved from Keycloak during client authentication.\n * @param {Array} [authData.roles] - The roles assigned to the user in Keycloak (optional).\n * @param {Array} [authData.groups] - The groups assigned to the user in Keycloak (optional).\n *\n * @description\n * ## Parse Server Configuration\n * To configure Parse Server for Keycloak authentication, use the following structure:\n * ```javascript\n * {\n *   \"auth\": {\n *     \"keycloak\": {\n *       \"config\": require('./auth/keycloak.json')\n *     }\n *   }\n * }\n * ```\n * Ensure the `keycloak.json` configuration file is generated from Keycloak's setup guide and includes:\n * - `auth-server-url`: The Keycloak authentication server URL.\n * - `realm`: The Keycloak realm name.\n * - `client-id`: The Keycloak client ID.\n *\n * ## Auth Data\n * The adapter requires the following `authData` fields:\n * - `access_token`: The Keycloak access token retrieved during client authentication.\n * - `id`: The user ID retrieved from Keycloak during client authentication.\n * - `roles` (optional): The roles assigned to the user in Keycloak.\n * - `groups` (optional): The groups assigned to the user in Keycloak.\n *\n * ## Auth Payload Example\n * ### Example Auth Data\n * ```json\n * {\n *   \"keycloak\": {\n *     \"access_token\": \"an authorized Keycloak access token for the user\",\n *     \"id\": \"user's Keycloak ID as a string\",\n *     \"roles\": [\"admin\", \"user\"],\n *     \"groups\": [\"group1\", \"group2\"]\n *   }\n * }\n * ```\n *\n * ## Notes\n * - Parse Server validates the provided `authData` by making a `userinfo` call to Keycloak and ensures the attributes match those returned by Keycloak.\n *\n * ## Keycloak Configuration\n * To configure Keycloak, copy the JSON configuration file generated from Keycloak's setup guide:\n * - [Keycloak Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/index.html#_javascript_adapter)\n *\n * Place the configuration file on your server, for example:\n * - `auth/keycloak.json`\n *\n * For more information on Keycloak authentication, see:\n * - [Securing Apps Documentation](https://www.keycloak.org/docs/latest/securing_apps/)\n * - [Server Administration Documentation](https://www.keycloak.org/docs/latest/server_admin/)\n */\n\nconst { Parse } = require('parse/node');\nconst jwksClient = require('jwks-rsa');\nconst jwt = require('jsonwebtoken');\nconst authUtils = require('./utils');\n\nconst arraysEqual = (_arr1, _arr2) => {\n  if (!Array.isArray(_arr1) || !Array.isArray(_arr2) || _arr1.length !== _arr2.length) { return false; }\n\n  var arr1 = _arr1.concat().sort();\n  var arr2 = _arr2.concat().sort();\n\n  for (var i = 0; i < arr1.length; i++) {\n    if (arr1[i] !== arr2[i]) { return false; }\n  }\n\n  return true;\n};\n\nconst getKeycloakKeyByKeyId = async (keyId, jwksUri, cacheMaxEntries, cacheMaxAge) => {\n  const client = jwksClient({\n    jwksUri,\n    cache: true,\n    cacheMaxEntries,\n    cacheMaxAge,\n  });\n\n  let key;\n  try {\n    key = await authUtils.getSigningKey(client, keyId);\n  } catch {\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 verifyAccessToken = async (\n  { access_token, id, roles, groups } = {},\n  { config, cacheMaxEntries, cacheMaxAge } = {}\n) => {\n  if (!(access_token && id)) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing access token and/or User id');\n  }\n  if (!config || !(config['auth-server-url'] && config['realm'])) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Missing keycloak configuration');\n  }\n  if (!config['client-id']) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      'Keycloak auth is not configured. Missing client-id.'\n    );\n  }\n\n  const expectedIssuer = `${config['auth-server-url']}/realms/${config['realm']}`;\n  const jwksUri = `${config['auth-server-url']}/realms/${config['realm']}/protocol/openid-connect/certs`;\n\n  const { kid: keyId } = authUtils.getHeaderFromToken(access_token);\n  const ONE_HOUR_IN_MS = 3600000;\n\n  cacheMaxAge = cacheMaxAge || ONE_HOUR_IN_MS;\n  cacheMaxEntries = cacheMaxEntries || 5;\n\n  const keycloakKey = await getKeycloakKeyByKeyId(keyId, jwksUri, cacheMaxEntries, cacheMaxAge);\n  const signingKey = keycloakKey.publicKey || keycloakKey.rsaPublicKey;\n\n  let jwtClaims;\n  try {\n    jwtClaims = jwt.verify(access_token, signingKey, {\n      algorithms: ['RS256'],\n    });\n  } catch (exception) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, `${exception.message}`);\n  }\n\n  if (jwtClaims.iss !== expectedIssuer) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `access token not issued by correct provider - expected: ${expectedIssuer} | from: ${jwtClaims.iss}`\n    );\n  }\n\n  if (jwtClaims.azp !== config['client-id']) {\n    throw new Parse.Error(\n      Parse.Error.OBJECT_NOT_FOUND,\n      `access token is not authorized for this client - expected: ${config['client-id']} | from: ${jwtClaims.azp}`\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\n  const rolesMatch = jwtClaims.roles === roles || arraysEqual(jwtClaims.roles, roles);\n  const groupsMatch = jwtClaims.groups === groups || arraysEqual(jwtClaims.groups, groups);\n\n  if (!rolesMatch || !groupsMatch) {\n    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid authentication');\n  }\n\n  return jwtClaims;\n};\n\nfunction validateAuthData(authData, options = {}) {\n  return verifyAccessToken(authData, options);\n}\n\nfunction validateAppId() {\n  return Promise.resolve();\n}\n\nmodule.exports = {\n  validateAppId,\n  validateAuthData,\n};\n"],"mappings":"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","ignoreList":[]}
|