Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

dashboard admin(groups/users) implementation and integrating with dynamic application config #303

Merged
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
72fc938
dashboard admin(groups/users) implementation and add unit/integration…
yubonluo Mar 20, 2024
29236e3
Add test cases of user Id matching dashboard admin
yubonluo Mar 20, 2024
22323c1
delete useless code
yubonluo Mar 20, 2024
f9666a0
change isDashboard function name to isRequestByDashboardAdmin
yubonluo Mar 21, 2024
123c87e
Merge branch 'workspace-pr-integr' of github.com:ruanyl/OpenSearch-Da…
yubonluo Apr 1, 2024
166451a
dashboard admin config integrating with dynamic application config
yubonluo Apr 2, 2024
363aeaa
Optimize the code
yubonluo Apr 2, 2024
c8bb0b5
fix test error
yubonluo Apr 2, 2024
74d9a26
delete useless code
yubonluo Apr 3, 2024
a07dde0
optimize code
yubonluo Apr 3, 2024
518cd75
optimize code and add unit test
yubonluo Apr 3, 2024
dc77999
optimize code according to comments
yubonluo Apr 3, 2024
71c20d9
change dashboardAdmin yml config to openseachDashboard
yubonluo Apr 3, 2024
c61dac9
add missed code
yubonluo Apr 3, 2024
0bf41b3
solve conflict
yubonluo Apr 8, 2024
f0c1b5d
optimize code
yubonluo Apr 8, 2024
2018bdf
Merge branch 'workspace-pr-integr' into pr-integr-dashboard-admin
yubonluo Apr 9, 2024
979851a
solve conflict
yubonluo Apr 9, 2024
3a70922
solve conflict
yubonluo Apr 9, 2024
a69cc58
delete useless code
yubonluo Apr 9, 2024
7d83f48
delete useless code
yubonluo Apr 9, 2024
f50093f
optimize code
yubonluo Apr 10, 2024
e9fd21a
delete useless code
yubonluo Apr 11, 2024
909af1c
add utils unit tests
yubonluo Apr 11, 2024
b342950
Merge branch 'workspace-pr-integr' of github.com:ruanyl/OpenSearch-Da…
yubonluo Apr 11, 2024
27d95fe
optimize code
yubonluo Apr 11, 2024
acfb166
Fix the wrong reference
yubonluo Apr 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions config/opensearch_dashboards.yml
Original file line number Diff line number Diff line change
Expand Up @@ -290,3 +290,8 @@

# Set the value to true to enable workspace feature
# workspace.enabled: false

# Set the backend roles in groups or users, whoever has the backend roles or exactly match the user ids defined in this config will be regard as dashboard admin.
# Dashboard admin will have the access to all the workspaces and objects inside OpenSearch Dashboards.
# workspace.dashboardAdmin.groups: ["dashboard_admin"]
# workspace.dashboardAdmin.users: ["dashboard_admin"]
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The naming workspace.dashboardAdmin may confuse me, it sounds like the admin for dashboard type only. I come few names like workspace.dashboardsAdmin,
workspace.admin or workpsace.superAdmin

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

workspace.admin may be confusing as admin of workspace stands for the the users who have write permission on specific workspaces, vote for superAdmin.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If there is no configuration named workspace.admin, would workspace.superAdmin also be confused?

1 change: 1 addition & 0 deletions src/legacy/server/osd_server.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,7 @@ declare module '@hapi/hapi' {
interface PluginsStates {
workspace?: {
id?: string;
isDashboardAdmin?: boolean;
};
}
}
Expand Down
8 changes: 8 additions & 0 deletions src/plugins/workspace/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,14 @@ export const configSchema = schema.object({
permission: schema.object({
enabled: schema.boolean({ defaultValue: true }),
}),
dashboardAdmin: schema.object({
groups: schema.arrayOf(schema.string(), {
defaultValue: [],
}),
users: schema.arrayOf(schema.string(), {
defaultValue: [],
}),
}),
});

export type WorkspacePluginConfigType = TypeOf<typeof configSchema>;
2 changes: 1 addition & 1 deletion src/plugins/workspace/opensearch_dashboards.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,6 @@
"requiredPlugins": [
"savedObjects"
],
"optionalPlugins": ["savedObjectsManagement"],
"optionalPlugins": ["savedObjectsManagement", "applicationConfig"],
"requiredBundles": ["opensearchDashboardsReact"]
}
10 changes: 9 additions & 1 deletion src/plugins/workspace/server/plugin.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@

import { coreMock } from '../../../core/server/mocks';
import { WorkspacePlugin } from './plugin';
import { AppPluginSetupDependencies } from './types';

describe('Workspace server plugin', () => {
it('#setup', async () => {
Expand All @@ -16,9 +17,16 @@ describe('Workspace server plugin', () => {
enabled: true,
},
});
const mockApplicationConfig = {
getConfigurationClient: jest.fn().mockResolvedValue({}),
registerConfigurationClient: jest.fn().mockResolvedValue({}),
};
const mockDependencies: AppPluginSetupDependencies = {
applicationConfig: mockApplicationConfig,
};
setupMock.capabilities.registerProvider.mockImplementationOnce((fn) => (value = fn()));
const workspacePlugin = new WorkspacePlugin(initializerContextConfigMock);
await workspacePlugin.setup(setupMock);
await workspacePlugin.setup(setupMock, mockDependencies);
expect(value).toMatchInlineSnapshot(`
Object {
"workspaces": Object {
Expand Down
111 changes: 95 additions & 16 deletions src/plugins/workspace/server/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,16 +11,21 @@ import {
Plugin,
Logger,
CoreStart,
OpenSearchDashboardsRequest,
} from '../../../core/server';
import {
WORKSPACE_SAVED_OBJECTS_CLIENT_WRAPPER_ID,
WORKSPACE_CONFLICT_CONTROL_SAVED_OBJECTS_CLIENT_WRAPPER_ID,
} from '../common/constants';
import { IWorkspaceClientImpl } from './types';
import { AppPluginSetupDependencies, IWorkspaceClientImpl } from './types';
import { WorkspaceClient } from './workspace_client';
import { registerRoutes } from './routes';
import { WorkspaceSavedObjectsClientWrapper } from './saved_objects';
import { cleanWorkspaceId, getWorkspaceIdFromUrl } from '../../../core/server/utils';
import {
cleanWorkspaceId,
getWorkspaceIdFromUrl,
updateWorkspaceState,
} from '../../../core/server/utils';
import { WorkspaceConflictSavedObjectsClientWrapper } from './saved_objects/saved_objects_wrapper_for_check_workspace_conflict';
import {
SavedObjectsPermissionControl,
Expand Down Expand Up @@ -55,12 +60,98 @@ export class WorkspacePlugin implements Plugin<{}, {}> {
});
}

private isRequestByDashboardAdmin(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can be a util function inside another file.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Have moved to src/plugins/workspace/server/utils.ts and add unit test.

request: OpenSearchDashboardsRequest,
groups: string[],
users: string[],
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nit: I'd recommend combine these two params into a param named principals: Principals.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think using deconstructed code is more conducive to subsequent use.

configGroups: string[],
configUsers: string[]
) {
if (configGroups.length === 0 && configUsers.length === 0) {
updateWorkspaceState(request, {
isDashboardAdmin: false,
});
return;
}
const groupMatchAny = groups.some((group) => configGroups.includes(group)) || false;
const userMatchAny = users.some((user) => configUsers.includes(user)) || false;
updateWorkspaceState(request, {
isDashboardAdmin: groupMatchAny || userMatchAny,
});
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

From the function name, I do not think it appropriate to update anything inside this function.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, I will change the function name to updateDashboardAdminStateForRequest

}

private async setupPermission(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No need to declare as a async function here I think

core: CoreSetup,
config: WorkspacePluginConfigType,
Copy link
Collaborator

@SuZhou-Joe SuZhou-Joe Apr 3, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we leverage the existing property config$?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done

{ applicationConfig }: AppPluginSetupDependencies
) {
this.permissionControl = new SavedObjectsPermissionControl(this.logger);

core.http.registerOnPostAuth(async (request, response, toolkit) => {
let groups: string[];
let users: string[];

// There may be calls to saved objects client before user get authenticated, need to add a try catch here as `getPrincipalsFromRequest` will throw error when user is not authenticated.
try {
({ groups = [], users = [] } = this.permissionControl!.getPrincipalsFromRequest(request));
} catch (e) {
return toolkit.next();
}
if (groups.length === 0 && users.length === 0) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If the cluster does not require authentication, anyone can be dashboardAdmin I think.
@ruanyl @wanglam @raintygao what do you think?

Copy link
Collaborator

@raintygao raintygao Apr 3, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Makes sense, a bit like OSD without security plugin installed that will have almost admin permissions. Also want to hear other's thoughts.

Copy link
Author

@yubonluo yubonluo Apr 3, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You mean to update isDashboardAdmin state to true if groups.length === 0 && users.length === 0 when both the security plugin and savedObject.permission.enabled are open.

updateWorkspaceState(request, {
isDashboardAdmin: false,
});
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why don't we use updateDashboardAdminStateForRequest method here?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If users/groups is null, we can prevent the program from running down in advance. Meanwhile, configGroups/configUsers has not been obtained.

return toolkit.next();
}

this.logger.info('Dynamic application configuration enabled:' + !!applicationConfig);
if (!!applicationConfig) {
const [coreStart] = await core.getStartServices();
const scopeClient = coreStart.opensearch.client.asScoped(request);
const applicationConfigClient = applicationConfig.getConfigurationClient(scopeClient);

const [configGroups, configUsers] = await Promise.all([
applicationConfigClient
.getEntityConfig('workspace.dashboardAdmin.groups')
.catch(() => undefined),
applicationConfigClient
.getEntityConfig('workspace.dashboardAdmin.users')
.catch(() => undefined),
]);

this.isRequestByDashboardAdmin(
request,
groups,
users,
configGroups ? [configGroups] : [],
configUsers ? [configUsers] : []
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
configGroups ? [configGroups] : [],
configUsers ? [configUsers] : []
configGroups ? configGroups : [],
configUsers ? configUsers : []

configGroups and configUsers should already be array I think?

);
return toolkit.next();
}

const configGroups = config.dashboardAdmin.groups || [];
const configUsers = config.dashboardAdmin.users || [];
this.isRequestByDashboardAdmin(request, groups, users, configGroups, configUsers);
return toolkit.next();
});

this.workspaceSavedObjectsClientWrapper = new WorkspaceSavedObjectsClientWrapper(
this.permissionControl
);

core.savedObjects.addClientWrapper(
0,
WORKSPACE_SAVED_OBJECTS_CLIENT_WRAPPER_ID,
this.workspaceSavedObjectsClientWrapper.wrapperFactory
);
}

constructor(initializerContext: PluginInitializerContext) {
this.logger = initializerContext.logger.get('plugins', 'workspace');
this.config$ = initializerContext.config.create<WorkspacePluginConfigType>();
}

public async setup(core: CoreSetup) {
public async setup(core: CoreSetup, { applicationConfig }: AppPluginSetupDependencies) {
this.logger.debug('Setting up Workspaces service');
const config: WorkspacePluginConfigType = await this.config$.pipe(first()).toPromise();
const isPermissionControlEnabled =
Expand All @@ -80,19 +171,7 @@ export class WorkspacePlugin implements Plugin<{}, {}> {
);

this.logger.info('Workspace permission control enabled:' + isPermissionControlEnabled);
if (isPermissionControlEnabled) {
this.permissionControl = new SavedObjectsPermissionControl(this.logger);

this.workspaceSavedObjectsClientWrapper = new WorkspaceSavedObjectsClientWrapper(
this.permissionControl
);

core.savedObjects.addClientWrapper(
0,
WORKSPACE_SAVED_OBJECTS_CLIENT_WRAPPER_ID,
this.workspaceSavedObjectsClientWrapper.wrapperFactory
);
}
if (isPermissionControlEnabled) this.setupPermission(core, config, { applicationConfig });

registerRoutes({
http: core.http,
Expand Down
Loading
Loading