mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
286 lines
8.6 KiB
TypeScript
286 lines
8.6 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import "./dialog.scss";
|
|
|
|
import { computed, observable, makeObservable, action } from "mobx";
|
|
import { observer } from "mobx-react";
|
|
import React from "react";
|
|
|
|
import { rolesStore } from "../+roles/store";
|
|
import { serviceAccountsStore } from "../+service-accounts/store";
|
|
import { NamespaceSelect } from "../../+namespaces/namespace-select";
|
|
import { ClusterRole, Role, roleApi, RoleBinding, RoleBindingSubject, ServiceAccount } from "../../../../common/k8s-api/endpoints";
|
|
import { Dialog, DialogProps } from "../../dialog";
|
|
import { EditableList } from "../../editable-list";
|
|
import { Icon } from "../../icon";
|
|
import { showDetails } from "../../kube-detail-params";
|
|
import { SubTitle } from "../../layout/sub-title";
|
|
import { Notifications } from "../../notifications";
|
|
import { Select, SelectOption } from "../../select";
|
|
import { Wizard, WizardStep } from "../../wizard";
|
|
import { roleBindingsStore } from "./store";
|
|
import { clusterRolesStore } from "../+cluster-roles/store";
|
|
import { Input } from "../../input";
|
|
import { ObservableHashSet, nFircate } from "../../../utils";
|
|
|
|
interface Props extends Partial<DialogProps> {
|
|
}
|
|
|
|
interface DialogState {
|
|
isOpen: boolean;
|
|
data?: RoleBinding;
|
|
}
|
|
|
|
@observer
|
|
export class RoleBindingDialog extends React.Component<Props> {
|
|
static state = observable.object<DialogState>({
|
|
isOpen: false,
|
|
});
|
|
|
|
constructor(props: Props) {
|
|
super(props);
|
|
makeObservable(this);
|
|
}
|
|
|
|
static open(roleBinding?: RoleBinding) {
|
|
RoleBindingDialog.state.isOpen = true;
|
|
RoleBindingDialog.state.data = roleBinding;
|
|
}
|
|
|
|
static close() {
|
|
RoleBindingDialog.state.isOpen = false;
|
|
RoleBindingDialog.state.data = undefined;
|
|
}
|
|
|
|
get roleBinding(): RoleBinding {
|
|
return RoleBindingDialog.state.data;
|
|
}
|
|
|
|
@computed get isEditing() {
|
|
return !!this.roleBinding;
|
|
}
|
|
|
|
@observable.ref selectedRoleRef: Role | ClusterRole | undefined = undefined;
|
|
@observable bindingName = "";
|
|
@observable bindingNamespace = "";
|
|
selectedAccounts = new ObservableHashSet<ServiceAccount>([], sa => sa.metadata.uid);
|
|
selectedUsers = observable.set<string>([]);
|
|
selectedGroups = observable.set<string>([]);
|
|
|
|
@computed get selectedBindings(): RoleBindingSubject[] {
|
|
const serviceAccounts = Array.from(this.selectedAccounts, sa => ({
|
|
name: sa.getName(),
|
|
kind: "ServiceAccount" as const,
|
|
namespace: sa.getNs(),
|
|
}));
|
|
const users = Array.from(this.selectedUsers, user => ({
|
|
name: user,
|
|
kind: "User" as const,
|
|
}));
|
|
const groups = Array.from(this.selectedGroups, group => ({
|
|
name: group,
|
|
kind: "Group" as const,
|
|
}));
|
|
|
|
return [
|
|
...serviceAccounts,
|
|
...users,
|
|
...groups,
|
|
];
|
|
}
|
|
|
|
@computed get roleRefOptions(): SelectOption<Role | ClusterRole>[] {
|
|
const roles = rolesStore.items
|
|
.filter(role => role.getNs() === this.bindingNamespace)
|
|
.map(value => ({ value, label: value.getName() }));
|
|
const clusterRoles = clusterRolesStore.items
|
|
.map(value => ({ value, label: value.getName() }));
|
|
|
|
return [
|
|
...roles,
|
|
...clusterRoles,
|
|
];
|
|
}
|
|
|
|
@computed get serviceAccountOptions(): SelectOption<ServiceAccount>[] {
|
|
return serviceAccountsStore.items.map(account => ({
|
|
value: account,
|
|
label: `${account.getName()} (${account.getNs()})`,
|
|
}));
|
|
}
|
|
|
|
@computed get selectedServiceAccountOptions(): SelectOption<ServiceAccount>[] {
|
|
return this.serviceAccountOptions.filter(({ value }) => this.selectedAccounts.has(value));
|
|
}
|
|
|
|
@action
|
|
onOpen = () => {
|
|
const binding = this.roleBinding;
|
|
|
|
if (!binding) {
|
|
return this.reset();
|
|
}
|
|
|
|
this.selectedRoleRef = (binding.roleRef.kind === roleApi.kind ? rolesStore : clusterRolesStore)
|
|
.items
|
|
.find(item => item.getName() === binding.roleRef.name);
|
|
|
|
this.bindingName = binding.getName();
|
|
this.bindingNamespace = binding.getNs();
|
|
|
|
const [saSubjects, uSubjects, gSubjects] = nFircate(binding.getSubjects(), "kind", ["ServiceAccount", "User", "Group"]);
|
|
const accountNames = new Set(saSubjects.map(acc => acc.name));
|
|
|
|
this.selectedAccounts.replace(
|
|
serviceAccountsStore.items
|
|
.filter(sa => accountNames.has(sa.getName())),
|
|
);
|
|
this.selectedUsers.replace(uSubjects.map(user => user.name));
|
|
this.selectedGroups.replace(gSubjects.map(group => group.name));
|
|
};
|
|
|
|
@action
|
|
reset = () => {
|
|
this.selectedRoleRef = undefined;
|
|
this.bindingName = "";
|
|
this.bindingNamespace = "";
|
|
this.selectedAccounts.clear();
|
|
this.selectedUsers.clear();
|
|
this.selectedGroups.clear();
|
|
};
|
|
|
|
createBindings = async () => {
|
|
const { selectedRoleRef, bindingNamespace: namespace, selectedBindings } = this;
|
|
|
|
try {
|
|
const roleBinding = this.isEditing
|
|
? await roleBindingsStore.updateSubjects(this.roleBinding, selectedBindings)
|
|
: await roleBindingsStore.create({ name: this.bindingName, namespace }, {
|
|
subjects: selectedBindings,
|
|
roleRef: {
|
|
name: selectedRoleRef.getName(),
|
|
kind: selectedRoleRef.kind,
|
|
},
|
|
});
|
|
|
|
showDetails(roleBinding.selfLink);
|
|
RoleBindingDialog.close();
|
|
} catch (err) {
|
|
Notifications.error(err);
|
|
}
|
|
};
|
|
|
|
renderContents() {
|
|
return (
|
|
<>
|
|
<SubTitle title="Namespace" />
|
|
<NamespaceSelect
|
|
lightTheme
|
|
isDisabled={this.isEditing}
|
|
value={this.bindingNamespace}
|
|
autoFocus={!this.isEditing}
|
|
onChange={({ value }) => this.bindingNamespace = value}
|
|
/>
|
|
|
|
<SubTitle title="Role Reference" />
|
|
<Select
|
|
lightTheme
|
|
placeholder="Select role or cluster role ..."
|
|
isDisabled={this.isEditing}
|
|
options={this.roleRefOptions}
|
|
value={this.selectedRoleRef}
|
|
onChange={({ value }) => {
|
|
if (!this.selectedRoleRef || this.bindingName === this.selectedRoleRef.getName()) {
|
|
this.bindingName = value.getName();
|
|
}
|
|
|
|
this.selectedRoleRef = value;
|
|
}}
|
|
/>
|
|
|
|
<SubTitle title="Binding Name" />
|
|
<Input
|
|
lightTheme
|
|
disabled={this.isEditing}
|
|
value={this.bindingName}
|
|
onChange={value => this.bindingName = value}
|
|
/>
|
|
|
|
<SubTitle title="Binding targets" />
|
|
|
|
<b>Users</b>
|
|
<EditableList
|
|
lightTheme
|
|
placeholder="Bind to User Account ..."
|
|
add={(newUser) => this.selectedUsers.add(newUser)}
|
|
items={Array.from(this.selectedUsers)}
|
|
remove={({ oldItem }) => this.selectedUsers.delete(oldItem)}
|
|
/>
|
|
|
|
<b>Groups</b>
|
|
<EditableList
|
|
lightTheme
|
|
placeholder="Bind to User Group ..."
|
|
add={(newGroup) => this.selectedGroups.add(newGroup)}
|
|
items={Array.from(this.selectedGroups)}
|
|
remove={({ oldItem }) => this.selectedGroups.delete(oldItem)}
|
|
/>
|
|
|
|
<b>Service Accounts</b>
|
|
<Select
|
|
isMulti
|
|
lightTheme
|
|
placeholder="Select service accounts ..."
|
|
autoConvertOptions={false}
|
|
options={this.serviceAccountOptions}
|
|
value={this.selectedServiceAccountOptions}
|
|
formatOptionLabel={({ value }: SelectOption<ServiceAccount>) => (
|
|
<><Icon small material="account_box" /> {value.getName()} ({value.getNs()})</>
|
|
)}
|
|
onChange={(selected: SelectOption<ServiceAccount>[] | null) => {
|
|
if (selected) {
|
|
this.selectedAccounts.replace(selected.map(opt => opt.value));
|
|
} else {
|
|
this.selectedAccounts.clear();
|
|
}
|
|
}}
|
|
maxMenuHeight={200}
|
|
/>
|
|
</>
|
|
);
|
|
}
|
|
|
|
render() {
|
|
const { ...dialogProps } = this.props;
|
|
const [action, nextLabel] = this.isEditing ? ["Edit", "Update"] : ["Add", "Create"];
|
|
const disableNext = !this.selectedRoleRef || !this.selectedBindings.length || !this.bindingNamespace || !this.bindingName;
|
|
|
|
return (
|
|
<Dialog
|
|
{...dialogProps}
|
|
className="AddRoleBindingDialog"
|
|
isOpen={RoleBindingDialog.state.isOpen}
|
|
close={RoleBindingDialog.close}
|
|
onClose={this.reset}
|
|
onOpen={this.onOpen}
|
|
>
|
|
<Wizard
|
|
header={<h5>{action} RoleBinding</h5>}
|
|
done={RoleBindingDialog.close}
|
|
>
|
|
<WizardStep
|
|
nextLabel={nextLabel}
|
|
next={this.createBindings}
|
|
disabledNext={disableNext}
|
|
>
|
|
{this.renderContents()}
|
|
</WizardStep>
|
|
</Wizard>
|
|
</Dialog>
|
|
);
|
|
}
|
|
}
|