Skip to content

Commit

Permalink
[8.x] [FTR] enable roles management testing for Observability project (
Browse files Browse the repository at this point in the history
…#196514) (#197845)

# Backport

This will backport the following commits from `main` to `8.x`:
- [[FTR] enable roles management testing for Observability project
(#196514)](#196514)

<!--- Backport version: 9.4.3 -->

### Questions ?
Please refer to the [Backport tool
documentation](https://github.com/sqren/backport)

<!--BACKPORT [{"author":{"name":"Dzmitry
Lemechko","email":"[email protected]"},"sourceCommit":{"committedDate":"2024-10-18T12:17:13Z","message":"[FTR]
enable roles management testing for Observability project
(#196514)\n\n## Summary\r\n\r\nThis PR makes changes in FTR `saml_auth`
service to allow creating\r\ncustom role for Oblt serverless project,
when roles management is\r\nexplicitly enabled with
`--xpack.security.roleManagementEnabled=true` in\r\nKibana server
arguments.\r\n\r\nI also added
[role_management/custom_role_access.ts\r\n](x-pack/test_serverless/functional/test_suites/observability/role_management/custom_role_access.ts)\r\nas
a test example. Currently roles management is enabled
in\r\n`x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts`\r\nand
after this PR is merged, more tests with custom roles can be
added\r\nfor Oblt project.\r\n\r\nHow to run tests:\r\n\r\n```\r\nnode
scripts/functional_tests --config
x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts\r\n```","sha":"16c965f853f17565e2da996b1f2ab21e9e33a003","branchLabelMapping":{"^v9.0.0$":"main","^v8.17.0$":"8.x","^v(\\d+).(\\d+).\\d+$":"$1.$2"}},"sourcePullRequest":{"labels":["release_note:skip","v9.0.0","v8.16.0","backport:version","v8.17.0"],"title":"[FTR]
enable roles management testing for Observability
project","number":196514,"url":"https://github.com/elastic/kibana/pull/196514","mergeCommit":{"message":"[FTR]
enable roles management testing for Observability project
(#196514)\n\n## Summary\r\n\r\nThis PR makes changes in FTR `saml_auth`
service to allow creating\r\ncustom role for Oblt serverless project,
when roles management is\r\nexplicitly enabled with
`--xpack.security.roleManagementEnabled=true` in\r\nKibana server
arguments.\r\n\r\nI also added
[role_management/custom_role_access.ts\r\n](x-pack/test_serverless/functional/test_suites/observability/role_management/custom_role_access.ts)\r\nas
a test example. Currently roles management is enabled
in\r\n`x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts`\r\nand
after this PR is merged, more tests with custom roles can be
added\r\nfor Oblt project.\r\n\r\nHow to run tests:\r\n\r\n```\r\nnode
scripts/functional_tests --config
x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts\r\n```","sha":"16c965f853f17565e2da996b1f2ab21e9e33a003"}},"sourceBranch":"main","suggestedTargetBranches":["8.x"],"targetPullRequestStates":[{"branch":"main","label":"v9.0.0","branchLabelMappingKey":"^v9.0.0$","isSourceBranch":true,"state":"MERGED","url":"https://github.com/elastic/kibana/pull/196514","number":196514,"mergeCommit":{"message":"[FTR]
enable roles management testing for Observability project
(#196514)\n\n## Summary\r\n\r\nThis PR makes changes in FTR `saml_auth`
service to allow creating\r\ncustom role for Oblt serverless project,
when roles management is\r\nexplicitly enabled with
`--xpack.security.roleManagementEnabled=true` in\r\nKibana server
arguments.\r\n\r\nI also added
[role_management/custom_role_access.ts\r\n](x-pack/test_serverless/functional/test_suites/observability/role_management/custom_role_access.ts)\r\nas
a test example. Currently roles management is enabled
in\r\n`x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts`\r\nand
after this PR is merged, more tests with custom roles can be
added\r\nfor Oblt project.\r\n\r\nHow to run tests:\r\n\r\n```\r\nnode
scripts/functional_tests --config
x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts\r\n```","sha":"16c965f853f17565e2da996b1f2ab21e9e33a003"}},{"branch":"8.16","label":"v8.16.0","branchLabelMappingKey":"^v(\\d+).(\\d+).\\d+$","isSourceBranch":false,"url":"https://github.com/elastic/kibana/pull/196862","number":196862,"state":"MERGED","mergeCommit":{"sha":"e42cc1f8a4246b4aa34191429b272174ed9f852b","message":"[8.16]
[FTR] enable roles management testing for Observability project
(#196514) (#196862)\n\n# Backport\n\nThis will backport the following
commits from `main` to `8.16`:\n- [[FTR] enable roles management testing
for Observability
project\n(#196514)](https://github.com/elastic/kibana/pull/196514)\n\n<!---
Backport version: 9.4.3 -->\n\n### Questions ?\nPlease refer to the
[Backport
tool\ndocumentation](https://github.com/sqren/backport)\n\n<!--BACKPORT
[{\"author\":{\"name\":\"Dzmitry\nLemechko\",\"email\":\"[email protected]\"},\"sourceCommit\":{\"committedDate\":\"2024-10-18T12:17:13Z\",\"message\":\"[FTR]\nenable
roles management testing for Observability project\n(#196514)\\n\\n##
Summary\\r\\n\\r\\nThis PR makes changes in FTR `saml_auth`\nservice to
allow creating\\r\\ncustom role for Oblt serverless project,\nwhen roles
management is\\r\\nexplicitly enabled
with\n`--xpack.security.roleManagementEnabled=true` in\\r\\nKibana
server\narguments.\\r\\n\\r\\nI also
added\n[role_management/custom_role_access.ts\\r\\n](x-pack/test_serverless/functional/test_suites/observability/role_management/custom_role_access.ts)\\r\\nas\na
test example. Currently roles management is
enabled\nin\\r\\n`x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts`\\r\\nand\nafter
this PR is merged, more tests with custom roles can be\nadded\\r\\nfor
Oblt project.\\r\\n\\r\\nHow to run
tests:\\r\\n\\r\\n```\\r\\nnode\nscripts/functional_tests
--config\nx-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts\\r\\n```\",\"sha\":\"16c965f853f17565e2da996b1f2ab21e9e33a003\",\"branchLabelMapping\":{\"^v9.0.0$\":\"main\",\"^v8.17.0$\":\"8.x\",\"^v(\\\\d+).(\\\\d+).\\\\d+$\":\"$1.$2\"}},\"sourcePullRequest\":{\"labels\":[\"release_note:skip\",\"v9.0.0\",\"v8.16.0\",\"backport:version\"],\"title\":\"[FTR]\nenable
roles management testing for
Observability\nproject\",\"number\":196514,\"url\":\"https://github.com/elastic/kibana/pull/196514\",\"mergeCommit\":{\"message\":\"[FTR]\nenable
roles management testing for Observability project\n(#196514)\\n\\n##
Summary\\r\\n\\r\\nThis PR makes changes in FTR `saml_auth`\nservice to
allow creating\\r\\ncustom role for Oblt serverless project,\nwhen roles
management is\\r\\nexplicitly enabled
with\n`--xpack.security.roleManagementEnabled=true` in\\r\\nKibana
server\narguments.\\r\\n\\r\\nI also
added\n[role_management/custom_role_access.ts\\r\\n](x-pack/test_serverless/functional/test_suites/observability/role_management/custom_role_access.ts)\\r\\nas\na
test example. Currently roles management is
enabled\nin\\r\\n`x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts`\\r\\nand\nafter
this PR is merged, more tests with custom roles can be\nadded\\r\\nfor
Oblt project.\\r\\n\\r\\nHow to run
tests:\\r\\n\\r\\n```\\r\\nnode\nscripts/functional_tests
--config\nx-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts\\r\\n```\",\"sha\":\"16c965f853f17565e2da996b1f2ab21e9e33a003\"}},\"sourceBranch\":\"main\",\"suggestedTargetBranches\":[\"8.16\"],\"targetPullRequestStates\":[{\"branch\":\"main\",\"label\":\"v9.0.0\",\"branchLabelMappingKey\":\"^v9.0.0$\",\"isSourceBranch\":true,\"state\":\"MERGED\",\"url\":\"https://github.com/elastic/kibana/pull/196514\",\"number\":196514,\"mergeCommit\":{\"message\":\"[FTR]\nenable
roles management testing for Observability project\n(#196514)\\n\\n##
Summary\\r\\n\\r\\nThis PR makes changes in FTR `saml_auth`\nservice to
allow creating\\r\\ncustom role for Oblt serverless project,\nwhen roles
management is\\r\\nexplicitly enabled
with\n`--xpack.security.roleManagementEnabled=true` in\\r\\nKibana
server\narguments.\\r\\n\\r\\nI also
added\n[role_management/custom_role_access.ts\\r\\n](x-pack/test_serverless/functional/test_suites/observability/role_management/custom_role_access.ts)\\r\\nas\na
test example. Currently roles management is
enabled\nin\\r\\n`x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts`\\r\\nand\nafter
this PR is merged, more tests with custom roles can be\nadded\\r\\nfor
Oblt project.\\r\\n\\r\\nHow to run
tests:\\r\\n\\r\\n```\\r\\nnode\nscripts/functional_tests
--config\nx-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts\\r\\n```\",\"sha\":\"16c965f853f17565e2da996b1f2ab21e9e33a003\"}},{\"branch\":\"8.16\",\"label\":\"v8.16.0\",\"branchLabelMappingKey\":\"^v(\\\\d+).(\\\\d+).\\\\d+$\",\"isSourceBranch\":false,\"state\":\"NOT_CREATED\"}]}]\nBACKPORT-->\n\nCo-authored-by:
Dzmitry Lemechko
<[email protected]>"}},{"branch":"8.x","label":"v8.17.0","branchLabelMappingKey":"^v8.17.0$","isSourceBranch":false,"state":"NOT_CREATED"}]}]
BACKPORT-->

Co-authored-by: Dzmitry Lemechko <[email protected]>
  • Loading branch information
kibanamachine and dmlemeshko authored Oct 25, 2024
1 parent 0e9c0b0 commit b52af9c
Show file tree
Hide file tree
Showing 7 changed files with 165 additions and 4 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ export interface KibanaRoleDescriptors {
}

const throwIfRoleNotSet = (role: string, customRole: string, roleDescriptors: Map<string, any>) => {
if (role === customRole && !roleDescriptors.has(customRole)) {
if (role === customRole && !roleDescriptors.get(customRole)) {
throw new Error(
`Set privileges for '${customRole}' using 'samlAuth.setCustomRole' before authentication.`
);
Expand Down Expand Up @@ -179,7 +179,7 @@ export function SamlAuthProvider({ getService }: FtrProviderContext) {
if (!isCustomRoleEnabled) {
throw new Error(`Custom roles are not supported for the current deployment`);
}
log.debug(`Updating role ${CUSTOM_ROLE}`);
log.debug(`Updating role '${CUSTOM_ROLE}'`);
const adminCookieHeader = await getAdminCredentials();

const customRoleDescriptors = {
Expand All @@ -199,6 +199,28 @@ export function SamlAuthProvider({ getService }: FtrProviderContext) {
supportedRoleDescriptors.set(CUSTOM_ROLE, customRoleDescriptors);
},

async deleteCustomRole() {
if (!isCustomRoleEnabled) {
throw new Error(`Custom roles are not supported for the current deployment`);
}

if (supportedRoleDescriptors.get(CUSTOM_ROLE)) {
log.debug(`Deleting role '${CUSTOM_ROLE}'`);
const adminCookieHeader = await getAdminCredentials();

// Resetting descriptors for the custom role, even if role deletion fails
supportedRoleDescriptors.set(CUSTOM_ROLE, null);
log.debug(`'${CUSTOM_ROLE}' descriptors were reset`);

const { status } = await supertestWithoutAuth
.delete(`/api/security/role/${CUSTOM_ROLE}`)
.set(INTERNAL_REQUEST_HEADERS)
.set(adminCookieHeader);

expect(status).to.be(204);
}
},

getCommonRequestHeader() {
return COMMON_REQUEST_HEADERS;
},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,8 +34,18 @@ const getDefaultServerlessRole = (projectType: string) => {
}
};

const isRoleManagementExplicitlyEnabled = (args: string[]): boolean => {
const roleManagementArg = args.find((arg) =>
arg.startsWith('--xpack.security.roleManagementEnabled=')
);

// Return true if the value is explicitly set to 'true', otherwise false
return roleManagementArg?.split('=')[1] === 'true' || false;
};

export class ServerlessAuthProvider implements AuthProvider {
private readonly projectType: string;
private readonly roleManagementEnabled: boolean;
private readonly rolesDefinitionPath: string;

constructor(config: Config) {
Expand All @@ -45,6 +55,10 @@ export class ServerlessAuthProvider implements AuthProvider {
return acc + (match ? match[1] : '');
}, '') as ServerlessProjectType;

// Indicates whether role management was explicitly enabled using
// the `--xpack.security.roleManagementEnabled=true` flag.
this.roleManagementEnabled = isRoleManagementExplicitlyEnabled(kbnServerArgs);

if (!isServerlessProjectType(this.projectType)) {
throw new Error(`Unsupported serverless projectType: ${this.projectType}`);
}
Expand All @@ -70,7 +84,9 @@ export class ServerlessAuthProvider implements AuthProvider {
}

isCustomRoleEnabled() {
return projectTypesWithCustomRolesEnabled.includes(this.projectType);
return (
projectTypesWithCustomRolesEnabled.includes(this.projectType) || this.roleManagementEnabled
);
}

getCustomRole() {
Expand Down
8 changes: 7 additions & 1 deletion x-pack/test_serverless/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -207,6 +207,8 @@ describe("my internal APIs test suite", async function() {
With custom native roles now enabled for the Security and Search projects on MKI, the FTR supports
defining and authenticating with custom roles in both UI functional tests and API integration tests.

To test role management within the Observability project, you can execute the tests using the existing [config.feature_flags.ts](x-pack/test_serverless/functional/test_suites/observability/config.feature_flags.ts), where this functionality is explicitly enabled. Though the config is not run on MKI, it provides the ability to test custom roles in Kibana CI before the functionality is enabled in MKI. When roles management is enabled on MKI, these tests can be migrated to the regular FTR config and will be run on MKI.

For compatibility with MKI, the role name `customRole` is reserved for use in tests. The test user is automatically assigned to this role, but before logging in via the browser, generating a cookie header, or creating an API key in each test suite, the role’s privileges must be updated.

Note: We are still working on a solution to run these tests against MKI. In the meantime, please tag the suite with `skipMKI`.
Expand All @@ -229,6 +231,9 @@ await samlAuth.setCustomRole({
});
// Then, log in via the browser as a user with the newly defined privileges
await pageObjects.svlCommonPage.loginWithCustomRole();
// Make sure to delete the custom role in the 'after' hook
await samlAuth.deleteCustomRole();
```

FTR api_integration test example:
Expand All @@ -251,8 +256,9 @@ await samlAuth.setCustomRole({
// Then, generate an API key with the newly defined privileges
const roleAuthc = await samlAuth.createM2mApiKeyWithRoleScope('customRole');
// Remember to invalidate the API key after use
// Remember to invalidate the API key after use and delete the custom role
await samlAuth.invalidateM2mApiKeyWithRoleScope(roleAuthc);
await samlAuth.deleteCustomRole();
```

### Testing with feature flags
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import { FtrProviderContext } from '../../ftr_provider_context';
export default function ({ loadTestFile }: FtrProviderContext) {
describe('serverless observability UI - feature flags', function () {
// add tests that require feature flags, defined in config.feature_flags.ts
loadTestFile(require.resolve('./role_management'));
loadTestFile(require.resolve('./infra'));
loadTestFile(require.resolve('../common/platform_security/navigation/management_nav_cards.ts'));
loadTestFile(require.resolve('../common/platform_security/roles.ts'));
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import expect from '@kbn/expect';
import { FtrProviderContext } from '../../../ftr_provider_context';
import { RoleCredentials } from '../../../../shared/services';

export default function ({ getPageObjects, getService }: FtrProviderContext) {
const pageObjects = getPageObjects(['svlCommonPage', 'timePicker', 'common', 'header']);
const samlAuth = getService('samlAuth');
const supertestWithoutAuth = getService('supertestWithoutAuth');
const esArchiver = getService('esArchiver');
const kibanaServer = getService('kibanaServer');
const testSubjects = getService('testSubjects');
let roleAuthc: RoleCredentials;

describe('With custom role', function () {
// skipping on MKI while we are working on a solution
this.tags(['skipMKI']);
before(async () => {
await esArchiver.loadIfNeeded('test/functional/fixtures/es_archiver/logstash_functional');
await kibanaServer.importExport.load('test/functional/fixtures/kbn_archiver/discover');
await kibanaServer.uiSettings.update({
defaultIndex: 'logstash-*',
});
await samlAuth.setCustomRole({
elasticsearch: {
indices: [{ names: ['logstash-*'], privileges: ['read', 'view_index_metadata'] }],
},
kibana: [
{
feature: {
discover: ['read'],
},
spaces: ['*'],
},
],
});
// login with custom role
await pageObjects.svlCommonPage.loginWithCustomRole();
await pageObjects.svlCommonPage.assertUserAvatarExists();
});

after(async () => {
await esArchiver.unload('test/functional/fixtures/es_archiver/logstash_functional');
await kibanaServer.importExport.unload('test/functional/fixtures/kbn_archiver/discover');
await kibanaServer.uiSettings.replace({});
await kibanaServer.savedObjects.cleanStandardList();
if (roleAuthc) {
await samlAuth.invalidateM2mApiKeyWithRoleScope(roleAuthc);
}
// delete custom role
await samlAuth.deleteCustomRole();
});

it('should have limited navigation menu', async () => {
await pageObjects.svlCommonPage.assertUserAvatarExists();
// discover navigation link is present
await testSubjects.existOrFail('~nav-item-id-last-used-logs-viewer');

// all other links in navigation menu are hidden
await testSubjects.missingOrFail('~nav-item-id-dashboards');
await testSubjects.missingOrFail('~nav-item-id-observability-overview:alerts');
await testSubjects.missingOrFail('~nav-item-id-observability-overview:cases');
await testSubjects.missingOrFail('~nav-item-id-slo');
await testSubjects.missingOrFail('~nav-item-id-aiops');
await testSubjects.missingOrFail('~nav-item-id-inventory');
await testSubjects.missingOrFail('~nav-item-id-apm');
await testSubjects.missingOrFail('~nav-item-id-metrics');
await testSubjects.missingOrFail('~nav-item-id-synthetics');

// TODO: 'Add data' and 'Project Settings' should be hidden
// await testSubjects.missingOrFail('~nav-item-id-observabilityOnboarding');
// await testSubjects.missingOrFail('~nav-item-id-project_settings_project_nav');
});

it('should access Discover app', async () => {
await pageObjects.common.navigateToApp('discover');
await pageObjects.timePicker.setDefaultAbsoluteRange();
await pageObjects.header.waitUntilLoadingHasFinished();
expect(await testSubjects.exists('unifiedHistogramChart')).to.be(true);
expect(await testSubjects.exists('discoverQueryHits')).to.be(true);
});

it('should access console with API key', async () => {
roleAuthc = await samlAuth.createM2mApiKeyWithRoleScope('customRole');
const { body } = await supertestWithoutAuth
.get('/api/console/api_server')
.set(roleAuthc.apiKeyHeader)
.set(samlAuth.getInternalRequestHeader())
.set({ 'kbn-xsrf': 'true' })
.expect(200);
expect(body.es).to.be.ok();
});
});
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/

import { FtrProviderContext } from '../../../ftr_provider_context';

export default function ({ loadTestFile }: FtrProviderContext) {
describe('Role Management', function () {
loadTestFile(require.resolve('./custom_role_access'));
});
}
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,8 @@ export default function ({ getPageObjects, getService }: FtrProviderContext) {
if (roleAuthc) {
await samlAuth.invalidateM2mApiKeyWithRoleScope(roleAuthc);
}
// delete custom role
await samlAuth.deleteCustomRole();
});

it('should have limited navigation menu', async () => {
Expand Down

0 comments on commit b52af9c

Please sign in to comment.