oncall-engine/grafana-plugin/src/pages/users/Users.tsx
Yulya Artyukhina 381520ee13
Get rid of installation token + add a bunch of tests (#624)
* Get rid of installation token (for OSS installations)

This is done by being required to supply the grafana API URL as an
environment variable on the backend. Additionally, optionally an OnCall
API URL environment variable can be passed in to the frontend (this basically
allows completely skipping the need to configure anything).
- deduplicated a lot of the sync logic on the frontend + made
error message more useful and consistent
- Split PluginConfigPage component into several subcomponents
(making it easier to test each individual component)
- Moved RootWithLoader (from plugin/GrafanaPluginRootPage) into its own
subcomponent (making it easier to test)
- Added tests for pre-existing components that were touched:
  - PluginConfigPage component (and its new subcomponents)
  - state/plugin and state/rootBaseStore functions
  - apps.grafana_plugin django app

Helm changes:
- add GRAFANA_API_URL to oncall.env
- some yaml autoformatting changes
- remove reference to python manage.py issue_invite_for_the_frontend --override

Co-authored-by: Joey Orlando <joseph.t.orlando@gmail.com>
2022-11-21 16:26:00 +01:00

389 lines
12 KiB
TypeScript

import React from 'react';
import { getLocationSrv } from '@grafana/runtime';
import { Alert, Button, HorizontalGroup, Icon, VerticalGroup } from '@grafana/ui';
import { PluginPage } from 'PluginPage';
import cn from 'classnames/bind';
import { debounce } from 'lodash-es';
import { observer } from 'mobx-react';
import LegacyNavHeading from 'navbar/LegacyNavHeading';
import { AppRootProps } from 'types';
import Avatar from 'components/Avatar/Avatar';
import GTable from 'components/GTable/GTable';
import PageErrorHandlingWrapper, { PageBaseState } from 'components/PageErrorHandlingWrapper/PageErrorHandlingWrapper';
import {
getWrongTeamResponseInfo,
initErrorDataState,
} from 'components/PageErrorHandlingWrapper/PageErrorHandlingWrapper.helpers';
import PluginLink from 'components/PluginLink/PluginLink';
import Text from 'components/Text/Text';
import UsersFilters from 'components/UsersFilters/UsersFilters';
import UserSettings from 'containers/UserSettings/UserSettings';
import { WithPermissionControl } from 'containers/WithPermissionControl/WithPermissionControl';
import { getRole } from 'models/user/user.helpers';
import { User as UserType, UserRole } from 'models/user/user.types';
import { pages } from 'pages';
import { getQueryParams } from 'plugin/GrafanaPluginRootPage.helpers';
import { WithStoreProps } from 'state/types';
import { UserAction } from 'state/userAction';
import { withMobXProviderContext } from 'state/withStore';
import { getRealFilters, getUserRowClassNameFn } from './Users.helpers';
import styles from './Users.module.css';
const cx = cn.bind(styles);
interface UsersProps extends WithStoreProps, AppRootProps {}
const ITEMS_PER_PAGE = 100;
interface UsersState extends PageBaseState {
page: number;
isWrongTeam: boolean;
userPkToEdit?: UserType['pk'] | 'new';
usersFilters?: {
searchTerm: string;
roles?: UserRole[];
};
}
@observer
class Users extends React.Component<UsersProps, UsersState> {
state: UsersState = {
page: 1,
isWrongTeam: false,
userPkToEdit: undefined,
usersFilters: {
searchTerm: '',
roles: [UserRole.ADMIN, UserRole.EDITOR, UserRole.VIEWER],
},
errorData: initErrorDataState(),
};
initialUsersLoaded = false;
private userId: string;
async componentDidMount() {
const { p } = getQueryParams();
this.setState({ page: p ? Number(p) : 1 }, this.updateUsers);
this.parseParams();
}
updateUsers = async () => {
const { store } = this.props;
const { usersFilters, page } = this.state;
const { userStore } = store;
if (!store.isUserActionAllowed(UserAction.ViewOtherUsers)) {
return;
}
getLocationSrv().update({ query: { p: page }, partial: true });
return await userStore.updateItems(getRealFilters(usersFilters), page);
};
componentDidUpdate() {
const { store } = this.props;
if (!this.initialUsersLoaded && store.isUserActionAllowed(UserAction.ViewOtherUsers)) {
this.updateUsers();
this.initialUsersLoaded = true;
}
if (this.userId !== getQueryParams()['id']) {
this.parseParams();
}
}
parseParams = async () => {
this.setState({ errorData: initErrorDataState() }); // reset wrong team error to false on query parse
const { store } = this.props;
const { id } = getQueryParams();
this.userId = id;
if (id) {
await (id === 'me' ? store.userStore.loadCurrentUser() : store.userStore.loadUser(String(id), true)).catch(
(error) => this.setState({ errorData: { ...getWrongTeamResponseInfo(error) } })
);
const userPkToEdit = String(id === 'me' ? store.userStore.currentUserPk : id);
if (store.userStore.items[userPkToEdit]) {
this.setState({ userPkToEdit });
}
}
};
render() {
const { usersFilters, userPkToEdit, page, errorData } = this.state;
const { store, query } = this.props;
const { userStore } = store;
const columns = [
{
width: '25%',
key: 'username',
title: 'User',
render: this.renderTitle,
},
{
width: '5%',
title: 'Role',
key: 'role',
render: this.renderRole,
},
{
width: '20%',
title: 'Status',
key: 'note',
render: this.renderNote,
},
{
width: '20%',
title: 'Default Notifications',
key: 'notifications-chain',
render: this.renderNotificationsChain,
},
{
width: '20%',
title: 'Important Notifications',
key: 'important-notifications-chain',
render: this.renderImportantNotificationsChain,
},
{
width: '5%',
key: 'buttons',
render: this.renderButtons,
},
];
const handleClear = () =>
this.setState(
{ usersFilters: { searchTerm: '', roles: [UserRole.ADMIN, UserRole.EDITOR, UserRole.VIEWER] } },
() => {
this.debouncedUpdateUsers();
}
);
const { count, results } = userStore.getSearchResult();
return (
<PluginPage pageNav={pages['users'].getPageNav()}>
<PageErrorHandlingWrapper
errorData={errorData}
objectName="user"
pageName="users"
itemNotFoundMessage={`User with id=${query?.id} is not found. Please select user from the list.`}
>
<>
<div className={cx('root')}>
<div className={cx('root', 'TEST-users-page')}>
<div className={cx('users-header')}>
<div style={{ display: 'flex', alignItems: 'baseline' }}>
<div>
<LegacyNavHeading>
<Text.Title level={3}>Users</Text.Title>
</LegacyNavHeading>
<Text type="secondary">
To manage permissions or add users, please visit{' '}
<a href="/org/users">Grafana user management</a>
</Text>
</div>
</div>
<PluginLink partial query={{ id: 'me' }}>
<Button variant="primary" icon="user">
View my profile
</Button>
</PluginLink>
</div>
{store.isUserActionAllowed(UserAction.ViewOtherUsers) ? (
<>
<div className={cx('user-filters-container')}>
<UsersFilters
className={cx('users-filters')}
value={usersFilters}
onChange={this.handleUsersFiltersChange}
/>
<Button
variant="secondary"
icon="times"
onClick={handleClear}
className={cx('searchIntegrationClear')}
>
Clear filters
</Button>
</div>
<GTable
emptyText={results ? 'No users found' : 'Loading...'}
rowKey="pk"
data={results}
columns={columns}
rowClassName={getUserRowClassNameFn(userPkToEdit, userStore.currentUserPk)}
pagination={{
page,
total: Math.ceil((count || 0) / ITEMS_PER_PAGE),
onChange: this.handleChangePage,
}}
/>
</>
) : (
<Alert
/* @ts-ignore */
title={
<>
You don't have enough permissions to view other users because you are not Admin.{' '}
<PluginLink query={{ page: 'users', id: 'me' }}>Click here</PluginLink> to open your profile
</>
}
severity="info"
/>
)}
</div>
{userPkToEdit && <UserSettings id={userPkToEdit} onHide={this.handleHideUserSettings} />}
</div>
</>
</PageErrorHandlingWrapper>
</PluginPage>
);
}
handleChangePage = (page: number) => {
this.setState({ page }, this.updateUsers);
};
renderTitle = (user: UserType) => {
return (
<HorizontalGroup>
<Avatar className={cx('user-avatar')} size="large" src={user.avatar} />
<div>
<div>{user.username}</div>
<Text type="secondary">{user.email}</Text>
<br />
<Text type="secondary">{user.verified_phone_number}</Text>
</div>
</HorizontalGroup>
);
};
renderRole = (user: UserType) => {
return getRole(user.role);
};
renderNotificationsChain = (user: UserType) => {
return user.notification_chain_verbal.default;
};
renderImportantNotificationsChain = (user: UserType) => {
return user.notification_chain_verbal.important;
};
renderContacts = (user: UserType) => {
return (
<div className={cx('contacts')}>
<div className={cx('contact')}>Slack: {user.slack_user_identity?.name || '-'}</div>
<div className={cx('contact')}>Telegram: {user.telegram_configuration?.telegram_nick_name || '-'}</div>
</div>
);
};
renderButtons = (user: UserType) => {
const { store } = this.props;
const { userStore } = store;
const isCurrent = userStore.currentUserPk === user.pk;
const action = isCurrent ? UserAction.UpdateOwnSettings : UserAction.UpdateOtherUsersSettings;
return (
<VerticalGroup justify="center">
<PluginLink partial query={{ id: user.pk }} disabled={!store.isUserActionAllowed(action)}>
<WithPermissionControl userAction={action}>
<Button
className={cx({
'TEST-edit-my-own-settings-button': isCurrent,
})}
fill="text"
>
Edit
</Button>
</WithPermissionControl>
</PluginLink>
</VerticalGroup>
);
};
renderNote = (user: UserType) => {
if (user.hidden_fields === true) {
return null;
}
let phone_verified = user.verified_phone_number !== null;
let phone_not_verified_message = 'Phone not verified';
if (user.cloud_connection_status !== null) {
phone_verified = false;
switch (user.cloud_connection_status) {
case 0:
phone_not_verified_message = 'Cloud is not synced';
break;
case 1:
phone_not_verified_message = 'User not matched with cloud';
break;
case 2:
phone_not_verified_message = 'Phone number is not verified in Grafana Cloud';
break;
case 3:
phone_verified = true;
break;
}
}
if (!phone_verified || !user.slack_user_identity || !user.telegram_configuration) {
let texts = [];
if (!phone_verified) {
texts.push(phone_not_verified_message);
}
if (!user.slack_user_identity) {
texts.push('Slack not verified');
}
if (!user.telegram_configuration) {
texts.push('Telegram not verified');
}
return (
<div>
<Icon className={cx('warning-message-icon')} name="exclamation-triangle" />
{texts.join(', ')}
</div>
);
}
return 'All contacts verified';
};
debouncedUpdateUsers = debounce(this.updateUsers, 500);
handleUsersFiltersChange = (usersFilters: any) => {
this.setState({ usersFilters, page: 1 }, () => {
this.debouncedUpdateUsers();
});
};
handleHideUserSettings = () => {
this.setState({ userPkToEdit: undefined });
getLocationSrv().update({ partial: true, query: { id: undefined } });
};
handleUserUpdate = () => {
this.updateUsers();
};
}
export default withMobXProviderContext(Users);