From a0fe6cf69a80069dca6a5a410615340dfc13ddf5 Mon Sep 17 00:00:00 2001 From: Craig Perkins Date: Tue, 10 Jan 2023 18:19:14 -0500 Subject: [PATCH] Fix regression in jwt url parameter by awaiting async getAdditionalAuthHeader (#1292) * Fix issue with jwt as url param after getAdditionalAuthHeader switched to async Signed-off-by: Craig Perkins Signed-off-by: Ryan Liang Co-authored-by: Ryan Liang (cherry picked from commit 385377cbcfdb9dece8892312799aab53ac98d297) --- package.json | 1 + server/auth/types/authentication_type.ts | 4 +- .../{jwt_auth.test.ts => jwt_helper.test.ts} | 0 server/auth/types/multiple/multi_auth.ts | 2 +- test/jest.config.server.js | 5 +- test/jest_integration/jwt_auth.test.ts | 328 ++++++++++++++++++ test/setup/after_env.js | 18 + 7 files changed, 354 insertions(+), 4 deletions(-) rename server/auth/types/jwt/{jwt_auth.test.ts => jwt_helper.test.ts} (100%) create mode 100644 test/jest_integration/jwt_auth.test.ts create mode 100644 test/setup/after_env.js diff --git a/package.json b/package.json index 44bb50173..308111012 100644 --- a/package.json +++ b/package.json @@ -26,6 +26,7 @@ "@testing-library/react-hooks": "^7.0.2", "@types/hapi__wreck": "^15.0.1", "gulp-rename": "2.0.0", + "jose": "^4.11.2", "saml-idp": "^1.2.1", "selenium-webdriver": "^4.0.0-alpha.7", "selfsigned": "^2.0.1", diff --git a/server/auth/types/authentication_type.ts b/server/auth/types/authentication_type.ts index b1ea1a208..c26db20db 100755 --- a/server/auth/types/authentication_type.ts +++ b/server/auth/types/authentication_type.ts @@ -112,7 +112,7 @@ export abstract class AuthenticationType implements IAuthenticationType { // see https://www.elastic.co/guide/en/opensearch-dashboards/master/using-api.html if (this.requestIncludesAuthInfo(request)) { try { - const additonalAuthHeader = this.getAdditionalAuthHeader(request); + const additonalAuthHeader = await this.getAdditionalAuthHeader(request); Object.assign(authHeaders, additonalAuthHeader); authInfo = await this.securityClient.authinfo(request, additonalAuthHeader); cookie = this.getCookie(request, authInfo); @@ -162,7 +162,7 @@ export abstract class AuthenticationType implements IAuthenticationType { // build auth header const authHeadersFromCookie = this.buildAuthHeaderFromCookie(cookie!); Object.assign(authHeaders, authHeadersFromCookie); - const additonalAuthHeader = this.getAdditionalAuthHeader(request); + const additonalAuthHeader = await this.getAdditionalAuthHeader(request); Object.assign(authHeaders, additonalAuthHeader); } diff --git a/server/auth/types/jwt/jwt_auth.test.ts b/server/auth/types/jwt/jwt_helper.test.ts similarity index 100% rename from server/auth/types/jwt/jwt_auth.test.ts rename to server/auth/types/jwt/jwt_helper.test.ts diff --git a/server/auth/types/multiple/multi_auth.ts b/server/auth/types/multiple/multi_auth.ts index d119c18d6..26a3439ef 100644 --- a/server/auth/types/multiple/multi_auth.ts +++ b/server/auth/types/multiple/multi_auth.ts @@ -120,7 +120,7 @@ export class MultipleAuthentication extends AuthenticationType { const reqAuthType = cookie?.authType?.toLowerCase(); if (reqAuthType && this.authHandlers.has(reqAuthType)) { - return this.authHandlers.get(reqAuthType)!.getAdditionalAuthHeader(request); + return await this.authHandlers.get(reqAuthType)!.getAdditionalAuthHeader(request); } else { return {}; } diff --git a/test/jest.config.server.js b/test/jest.config.server.js index 85b3c81ad..783003154 100644 --- a/test/jest.config.server.js +++ b/test/jest.config.server.js @@ -22,7 +22,10 @@ export default { testPathIgnorePatterns: config.testPathIgnorePatterns.filter( (pattern) => !pattern.includes('integration_tests') ), - setupFilesAfterEnv: ['/src/dev/jest/setup/after_env.integration.js'], + setupFilesAfterEnv: [ + '/src/dev/jest/setup/after_env.integration.js', + '/plugins/security-dashboards-plugin/test/setup/after_env.js', + ], collectCoverageFrom: [ '/plugins/security-dashboards-plugin/server/**/*.{ts,tsx}', '!/plugins/security-dashboards-plugin/server/**/*.test.{ts,tsx}', diff --git a/test/jest_integration/jwt_auth.test.ts b/test/jest_integration/jwt_auth.test.ts new file mode 100644 index 000000000..371abe961 --- /dev/null +++ b/test/jest_integration/jwt_auth.test.ts @@ -0,0 +1,328 @@ +/* + * Copyright OpenSearch Contributors + * + * Licensed under the Apache License, Version 2.0 (the "License"). + * You may not use this file except in compliance with the License. + * A copy of the License is located at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * or in the "license" file accompanying this file. This file is distributed + * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either + * express or implied. See the License for the specific language governing + * permissions and limitations under the License. + */ + +import * as osdTestServer from '../../../../src/core/test_helpers/osd_server'; +import { Root } from '../../../../src/core/server/root'; +import { resolve } from 'path'; +import { describe, expect, it, beforeAll, afterAll } from '@jest/globals'; +import { SignJWT } from 'jose'; +import { + ADMIN_CREDENTIALS, + OPENSEARCH_DASHBOARDS_SERVER_USER, + OPENSEARCH_DASHBOARDS_SERVER_PASSWORD, +} from '../constant'; +import wreck from '@hapi/wreck'; +import { Builder, By, until } from 'selenium-webdriver'; +import { Options } from 'selenium-webdriver/firefox'; + +describe('start OpenSearch Dashboards server', () => { + let root: Root; + let config; + + // XPath Constants + const pageTitleXPath = '//*[@id="osdOverviewPageHeader__title"]'; + // Browser Settings + const browser = 'firefox'; + const options = new Options().headless(); + const rawKey = 'This is a very secure secret. No one will ever be able to guess it!'; + const b = Buffer.from(rawKey); + const signingKey = b.toString('base64'); + + beforeAll(async () => { + root = osdTestServer.createRootWithSettings( + { + plugins: { + scanDirs: [resolve(__dirname, '../..')], + }, + server: { + host: 'localhost', + port: 5601, + }, + logging: { + silent: true, + verbose: false, + }, + opensearch: { + hosts: ['https://localhost:9200'], + ignoreVersionMismatch: true, + ssl: { verificationMode: 'none' }, + username: OPENSEARCH_DASHBOARDS_SERVER_USER, + password: OPENSEARCH_DASHBOARDS_SERVER_PASSWORD, + requestHeadersWhitelist: ['authorization', 'securitytenant'], + }, + opensearch_security: { + auth: { + anonymous_auth_enabled: false, + type: 'jwt', + }, + jwt: { + url_param: 'token', + }, + }, + }, + { + // to make ignoreVersionMismatch setting work + // can be removed when we have corresponding ES version + dev: true, + } + ); + + console.log('Starting OpenSearchDashboards server..'); + await root.setup(); + await root.start(); + + await wreck.patch('https://localhost:9200/_plugins/_security/api/rolesmapping/all_access', { + payload: [ + { + op: 'add', + path: '/users', + value: ['jwt_test'], + }, + ], + rejectUnauthorized: false, + headers: { + 'Content-Type': 'application/json', + authorization: ADMIN_CREDENTIALS, + }, + }); + console.log('Starting to Download Flights Sample Data'); + await wreck.post('http://localhost:5601/api/sample_data/flights', { + payload: {}, + rejectUnauthorized: false, + headers: { + 'Content-Type': 'application/json', + authorization: ADMIN_CREDENTIALS, + security_tenant: 'global', + }, + }); + console.log('Downloaded Sample Data'); + const getConfigResponse = await wreck.get( + 'https://localhost:9200/_plugins/_security/api/securityconfig', + { + rejectUnauthorized: false, + headers: { + authorization: ADMIN_CREDENTIALS, + }, + } + ); + const responseBody = (getConfigResponse.payload as Buffer).toString(); + config = JSON.parse(responseBody).config; + const jwtConfig = { + http_enabled: true, + transport_enabled: false, + order: 5, + http_authenticator: { + challenge: true, + type: 'jwt', + config: { + signing_key: signingKey, + jwt_header: 'Authorization', + jwt_url_parameter: 'token', + subject_key: 'sub', + roles_key: 'roles', + }, + }, + authentication_backend: { + type: 'noop', + config: {}, + }, + }; + try { + config.dynamic!.authc!.jwt_auth_domain = jwtConfig; + config.dynamic!.authc!.basic_internal_auth_domain.http_authenticator.challenge = false; + config.dynamic!.http!.anonymous_auth_enabled = false; + await wreck.put('https://localhost:9200/_plugins/_security/api/securityconfig/config', { + payload: config, + rejectUnauthorized: false, + headers: { + 'Content-Type': 'application/json', + authorization: ADMIN_CREDENTIALS, + }, + }); + } catch (error) { + console.log('Got an error while updating security config!!', error.stack); + fail(error); + } + }); + + afterAll(async () => { + console.log('Remove the Sample Data'); + await wreck + .delete('http://localhost:5601/api/sample_data/flights', { + rejectUnauthorized: false, + headers: { + 'Content-Type': 'application/json', + authorization: ADMIN_CREDENTIALS, + }, + }) + .then((value) => { + Promise.resolve(value); + }) + .catch((value) => { + Promise.resolve(value); + }); + console.log('Remove the Role Mapping'); + await wreck + .patch('https://localhost:9200/_plugins/_security/api/rolesmapping/all_access', { + payload: [ + { + op: 'remove', + path: '/users', + users: ['jwt_test'], + }, + ], + rejectUnauthorized: false, + headers: { + 'Content-Type': 'application/json', + authorization: ADMIN_CREDENTIALS, + }, + }) + .then((value) => { + Promise.resolve(value); + }) + .catch((value) => { + Promise.resolve(value); + }); + console.log('Remove the Security Config'); + await wreck + .patch('https://localhost:9200/_plugins/_security/api/securityconfig', { + payload: [ + { + op: 'remove', + path: '/config/dynamic/authc/jwt_auth_domain', + }, + ], + rejectUnauthorized: false, + headers: { + 'Content-Type': 'application/json', + authorization: ADMIN_CREDENTIALS, + }, + }) + .then((value) => { + Promise.resolve(value); + }) + .catch((value) => { + Promise.resolve(value); + }); + // shutdown OpenSearchDashboards server + await root.shutdown(); + }); + + it('Login to app/opensearch_dashboards_overview#/ when JWT is enabled', async () => { + const payload = { + sub: 'jwt_test', + roles: 'admin,kibanauser', + }; + + const key = new TextEncoder().encode(rawKey); + + const token = await new SignJWT(payload) // details to encode in the token + .setProtectedHeader({ alg: 'HS256' }) // algorithm + .setIssuedAt() + .sign(key); + const driver = getDriver(browser, options).build(); + await driver.get(`http://localhost:5601/app/opensearch_dashboards_overview?token=${token}`); + await driver.wait(until.elementsLocated(By.xpath(pageTitleXPath)), 10000); + + const cookie = await driver.manage().getCookies(); + expect(cookie.length).toEqual(1); + await driver.manage().deleteAllCookies(); + await driver.quit(); + }); + + it('Login to app/dev_tools#/console when JWT is enabled', async () => { + const payload = { + sub: 'jwt_test', + roles: 'admin,kibanauser', + }; + + const key = new TextEncoder().encode(rawKey); + + const token = await new SignJWT(payload) // details to encode in the token + .setProtectedHeader({ alg: 'HS256' }) // algorithm + .setIssuedAt() + .sign(key); + const driver = getDriver(browser, options).build(); + await driver.get(`http://localhost:5601/app/dev_tools?token=${token}`); + + await driver.wait( + until.elementsLocated(By.xpath('//*[@data-test-subj="sendRequestButton"]')), + 10000 + ); + + const cookie = await driver.manage().getCookies(); + expect(cookie.length).toEqual(1); + await driver.manage().deleteAllCookies(); + await driver.quit(); + }); + + it('Login to app/opensearch_dashboards_overview#/ when JWT is enabled with invalid token', async () => { + const payload = { + sub: 'jwt_test', + roles: 'admin,kibanauser', + }; + + const key = new TextEncoder().encode('wrongKey'); + + const token = await new SignJWT(payload) // details to encode in the token + .setProtectedHeader({ alg: 'HS256' }) // algorithm + .setIssuedAt() + .sign(key); + const driver = getDriver(browser, options).build(); + await driver.get(`http://localhost:5601/app/opensearch_dashboards_overview?token=${token}`); + + const rep = await driver.getPageSource(); + expect(rep).toContain( + '"statusCode":401,"error":"Unauthorized","message":"Authentication Exception"' + ); + + const cookie = await driver.manage().getCookies(); + expect(cookie.length).toEqual(0); + + await driver.manage().deleteAllCookies(); + await driver.quit(); + }); + + it('Login to app/dev_tools#/console when JWT is enabled with invalid token', async () => { + const payload = { + sub: 'jwt_test', + roles: 'admin,kibanauser', + }; + + const key = new TextEncoder().encode('wrongKey'); + + const token = await new SignJWT(payload) // details to encode in the token + .setProtectedHeader({ alg: 'HS256' }) // algorithm + .setIssuedAt() + .sign(key); + const driver = getDriver(browser, options).build(); + await driver.get(`http://localhost:5601/app/dev_tools?token=${token}`); + + const rep = await driver.getPageSource(); + expect(rep).toContain( + '"statusCode":401,"error":"Unauthorized","message":"Authentication Exception"' + ); + + const cookie = await driver.manage().getCookies(); + expect(cookie.length).toEqual(0); + + await driver.manage().deleteAllCookies(); + await driver.quit(); + }); +}); + +function getDriver(browser: string, options: Options) { + return new Builder().forBrowser(browser).setFirefoxOptions(options); +} diff --git a/test/setup/after_env.js b/test/setup/after_env.js new file mode 100644 index 000000000..654f9134b --- /dev/null +++ b/test/setup/after_env.js @@ -0,0 +1,18 @@ +/* + * Copyright OpenSearch Contributors + * + * Licensed under the Apache License, Version 2.0 (the "License"). + * You may not use this file except in compliance with the License. + * A copy of the License is located at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * or in the "license" file accompanying this file. This file is distributed + * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either + * express or implied. See the License for the specific language governing + * permissions and limitations under the License. + */ + +import { TextEncoder, TextDecoder } from 'util'; +global.TextEncoder = TextEncoder; +global.TextDecoder = TextDecoder;