mirror of
https://github.com/grafana/grafana.git
synced 2025-02-13 00:55:47 -06:00
* Add frontend test * Add frontend label component * Adjust backend tests * Retrieve auth IDP labels for users at org/users. Co-authored-by: Misi <mgyongyosi@users.noreply.github.com> Co-authored-by: Kalle <kalleep@users.noreply.github.com> Co-authored-by: Jo <Jguer@users.noreply.github.com>
153 lines
5.1 KiB
TypeScript
153 lines
5.1 KiB
TypeScript
import React, { FC, useEffect, useState } from 'react';
|
|
|
|
import { OrgRole } from '@grafana/data';
|
|
import { Button, ConfirmModal } from '@grafana/ui';
|
|
import { UserRolePicker } from 'app/core/components/RolePicker/UserRolePicker';
|
|
import { fetchRoleOptions } from 'app/core/components/RolePicker/api';
|
|
import { TagBadge } from 'app/core/components/TagFilter/TagBadge';
|
|
import { contextSrv } from 'app/core/core';
|
|
import { AccessControlAction, OrgUser, Role } from 'app/types';
|
|
|
|
import { OrgRolePicker } from '../admin/OrgRolePicker';
|
|
|
|
export interface Props {
|
|
users: OrgUser[];
|
|
orgId?: number;
|
|
onRoleChange: (role: OrgRole, user: OrgUser) => void;
|
|
onRemoveUser: (user: OrgUser) => void;
|
|
}
|
|
|
|
const UsersTable: FC<Props> = (props) => {
|
|
const { users, orgId, onRoleChange, onRemoveUser } = props;
|
|
const [userToRemove, setUserToRemove] = useState<OrgUser | null>(null);
|
|
const [roleOptions, setRoleOptions] = useState<Role[]>([]);
|
|
|
|
useEffect(() => {
|
|
async function fetchOptions() {
|
|
try {
|
|
if (contextSrv.hasPermission(AccessControlAction.ActionRolesList)) {
|
|
let options = await fetchRoleOptions(orgId);
|
|
setRoleOptions(options);
|
|
}
|
|
} catch (e) {
|
|
console.error('Error loading options');
|
|
}
|
|
}
|
|
if (contextSrv.licensedAccessControlEnabled()) {
|
|
fetchOptions();
|
|
}
|
|
}, [orgId]);
|
|
|
|
return (
|
|
<>
|
|
<table className="filter-table form-inline">
|
|
<thead>
|
|
<tr>
|
|
<th />
|
|
<th>Login</th>
|
|
<th>Email</th>
|
|
<th>Name</th>
|
|
<th>Seen</th>
|
|
<th>Role</th>
|
|
<th style={{ width: '34px' }} />
|
|
<th></th>
|
|
<th></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{users.map((user, index) => {
|
|
return (
|
|
<tr key={`${user.userId}-${index}`}>
|
|
<td className="width-2 text-center">
|
|
<img className="filter-table__avatar" src={user.avatarUrl} alt="User avatar" />
|
|
</td>
|
|
<td className="max-width-6">
|
|
<span className="ellipsis" title={user.login}>
|
|
{user.login}
|
|
</span>
|
|
</td>
|
|
|
|
<td className="max-width-5">
|
|
<span className="ellipsis" title={user.email}>
|
|
{user.email}
|
|
</span>
|
|
</td>
|
|
<td className="max-width-5">
|
|
<span className="ellipsis" title={user.name}>
|
|
{user.name}
|
|
</span>
|
|
</td>
|
|
<td className="width-1">{user.lastSeenAtAge}</td>
|
|
|
|
<td className="width-8">
|
|
{contextSrv.licensedAccessControlEnabled() ? (
|
|
<UserRolePicker
|
|
userId={user.userId}
|
|
orgId={orgId}
|
|
roleOptions={roleOptions}
|
|
basicRole={user.role}
|
|
onBasicRoleChange={(newRole) => onRoleChange(newRole, user)}
|
|
basicRoleDisabled={!contextSrv.hasPermissionInMetadata(AccessControlAction.OrgUsersWrite, user)}
|
|
/>
|
|
) : (
|
|
<OrgRolePicker
|
|
aria-label="Role"
|
|
value={user.role}
|
|
disabled={!contextSrv.hasPermissionInMetadata(AccessControlAction.OrgUsersWrite, user)}
|
|
onChange={(newRole) => onRoleChange(newRole, user)}
|
|
/>
|
|
)}
|
|
</td>
|
|
|
|
<td className="width-1 text-center">
|
|
{user.isDisabled && <span className="label label-tag label-tag--gray">Disabled</span>}
|
|
</td>
|
|
|
|
<td className="width-1">
|
|
{Array.isArray(user.authLabels) && user.authLabels.length > 0 && (
|
|
<TagBadge label={user.authLabels[0]} removeIcon={false} count={0} />
|
|
)}
|
|
</td>
|
|
|
|
{contextSrv.hasPermissionInMetadata(AccessControlAction.OrgUsersRemove, user) && (
|
|
<td className="text-right">
|
|
<Button
|
|
size="sm"
|
|
variant="destructive"
|
|
onClick={() => {
|
|
setUserToRemove(user);
|
|
}}
|
|
icon="times"
|
|
aria-label="Delete user"
|
|
/>
|
|
</td>
|
|
)}
|
|
</tr>
|
|
);
|
|
})}
|
|
</tbody>
|
|
</table>
|
|
{Boolean(userToRemove) && (
|
|
<ConfirmModal
|
|
body={`Are you sure you want to delete user ${userToRemove?.login}?`}
|
|
confirmText="Delete"
|
|
title="Delete"
|
|
onDismiss={() => {
|
|
setUserToRemove(null);
|
|
}}
|
|
isOpen={true}
|
|
onConfirm={() => {
|
|
if (!userToRemove) {
|
|
return;
|
|
}
|
|
onRemoveUser(userToRemove);
|
|
setUserToRemove(null);
|
|
}}
|
|
/>
|
|
)}
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default UsersTable;
|