fleet/frontend/pages/admin/UserManagementPage/UserManagementPage.jsx
Zachary Wasserman d453ca3f8f
Enable eslint file resolution linting (#2119)
Previously this was disabled (perhaps unintentionally due to the
duplicate settings in the .eslintrc.js). Enable the lint rule and fix
the violations.

May fix JS build issues on case-sensitive filesystems.
2019-10-22 13:00:31 -07:00

332 lines
9.9 KiB
JavaScript

import React, { Component } from 'react';
import PropTypes from 'prop-types';
import { connect } from 'react-redux';
import { concat, includes, difference } from 'lodash';
import { push } from 'react-router-redux';
import Button from 'components/buttons/Button';
import configInterface from 'interfaces/config';
import deepDifference from 'utilities/deep_difference';
import entityGetter from 'redux/utilities/entityGetter';
import inviteActions from 'redux/nodes/entities/invites/actions';
import inviteInterface from 'interfaces/invite';
import InviteUserForm from 'components/forms/InviteUserForm';
import Modal from 'components/modals/Modal';
import paths from 'router/paths';
import { renderFlash } from 'redux/nodes/notifications/actions';
import WarningBanner from 'components/WarningBanner';
import { updateUser } from 'redux/nodes/auth/actions';
import userActions from 'redux/nodes/entities/users/actions';
import UserBlock from 'components/UserBlock';
import userInterface from 'interfaces/user';
const baseClass = 'user-management';
export class UserManagementPage extends Component {
static propTypes = {
appConfigLoading: PropTypes.bool,
config: configInterface,
currentUser: userInterface,
dispatch: PropTypes.func,
inviteErrors: PropTypes.shape({
base: PropTypes.string,
email: PropTypes.string,
}),
invites: PropTypes.arrayOf(inviteInterface),
loadingInvites: PropTypes.bool,
loadingUsers: PropTypes.bool,
userErrors: PropTypes.shape({
base: PropTypes.string,
name: PropTypes.string,
username: PropTypes.string,
}),
users: PropTypes.arrayOf(userInterface),
};
constructor (props) {
super(props);
this.state = {
showInviteUserModal: false,
usersEditing: [],
};
}
componentWillMount () {
const { dispatch } = this.props;
dispatch(userActions.loadAll());
dispatch(inviteActions.loadAll());
return false;
}
onUserActionSelect = (user, action) => {
const { currentUser, dispatch } = this.props;
const { enableUser, updateAdmin, requirePasswordReset } = userActions;
if (action) {
switch (action) {
case 'demote_user': {
if (currentUser.id === user.id) {
return dispatch(renderFlash('error', 'You cannot demote yourself'));
}
return dispatch(updateAdmin(user, { admin: false }))
.then(() => {
return dispatch(renderFlash('success', 'User demoted', updateAdmin(user, { admin: true })));
});
}
case 'disable_account': {
if (currentUser.id === user.id) {
return dispatch(renderFlash('error', 'You cannot disable your own account'));
}
return dispatch(userActions.enableUser(user, { enabled: false }))
.then(() => {
return dispatch(renderFlash('success', 'User account disabled', enableUser(user, { enabled: true })));
});
}
case 'enable_account':
return dispatch(enableUser(user, { enabled: true }))
.then(() => {
return dispatch(renderFlash('success', 'User account enabled', enableUser(user, { enabled: false })));
});
case 'promote_user':
return dispatch(updateAdmin(user, { admin: true }))
.then(() => {
return dispatch(renderFlash('success', 'User promoted to admin', updateAdmin(user, { admin: false })));
});
case 'reset_password':
return dispatch(requirePasswordReset(user, { require: true }))
.then(() => {
return dispatch(renderFlash('success', 'User required to reset password', requirePasswordReset(user, { require: false })));
});
case 'revert_invitation':
return dispatch(inviteActions.silentDestroy(user))
.then(() => dispatch(renderFlash('success', 'Invite revoked')))
.catch(() => dispatch(renderFlash('error', 'Invite could not be revoked')));
default:
return false;
}
}
return false;
}
onEditUser = (user, updatedUser) => {
const { currentUser, dispatch } = this.props;
const { onToggleEditUser } = this;
const { silentUpdate } = userActions;
const updatedAttrs = deepDifference(updatedUser, user);
if (currentUser.id === user.id) {
return dispatch(updateUser(user, updatedAttrs))
.then(() => {
dispatch(renderFlash('success', 'User updated', updateUser(user, user)));
onToggleEditUser(user);
return false;
})
.catch(() => false);
}
return dispatch(silentUpdate(user, updatedAttrs))
.then(() => {
dispatch(renderFlash('success', 'User updated', silentUpdate(user, user)));
onToggleEditUser(user);
return false;
})
.catch(() => false);
}
onInviteUserSubmit = (formData) => {
const { dispatch } = this.props;
dispatch(inviteActions.silentCreate(formData))
.then(() => {
return this.toggleInviteUserModal();
})
.catch(() => false);
}
onInviteCancel = (evt) => {
evt.preventDefault();
return this.toggleInviteUserModal();
}
onToggleEditUser = (user) => {
const { dispatch } = this.props;
const { usersEditing } = this.state;
let updatedUsersEditing = [];
dispatch(userActions.clearErrors());
if (includes(usersEditing, user.id)) {
updatedUsersEditing = difference(usersEditing, [user.id]);
} else {
updatedUsersEditing = concat(usersEditing, [user.id]);
}
this.setState({ usersEditing: updatedUsersEditing });
}
goToAppConfigPage = (evt) => {
evt.preventDefault();
const { ADMIN_SETTINGS } = paths;
const { dispatch } = this.props;
dispatch(push(ADMIN_SETTINGS));
}
toggleInviteUserModal = () => {
const { showInviteUserModal } = this.state;
this.setState({
showInviteUserModal: !showInviteUserModal,
});
return false;
}
renderUserBlock = (user, idx, options = { invite: false }) => {
const { currentUser, userErrors } = this.props;
const { invite } = options;
const { onEditUser, onToggleEditUser, onUserActionSelect } = this;
const { usersEditing } = this.state;
const isEditing = includes(usersEditing, user.id);
return (
<UserBlock
isEditing={isEditing}
isInvite={invite}
isCurrentUser={currentUser.id === user.id}
key={`${user.email}-${idx}-${invite ? 'invite' : 'user'}`}
onEditUser={onEditUser}
onSelect={onUserActionSelect}
onToggleEditUser={onToggleEditUser}
user={user}
userErrors={userErrors}
/>
);
}
renderModal = () => {
const { currentUser, inviteErrors } = this.props;
const { showInviteUserModal } = this.state;
const { onInviteCancel, onInviteUserSubmit, toggleInviteUserModal } = this;
const ssoEnabledForApp = this.props.config.enable_sso;
if (!showInviteUserModal) {
return false;
}
return (
<Modal
title="Invite New User"
onExit={toggleInviteUserModal}
className={`${baseClass}__invite-modal`}
>
<InviteUserForm
serverErrors={inviteErrors}
invitedBy={currentUser}
onCancel={onInviteCancel}
onSubmit={onInviteUserSubmit}
canUseSSO={ssoEnabledForApp}
/>
</Modal>
);
};
renderSmtpWarning = () => {
const { appConfigLoading, config } = this.props;
const { goToAppConfigPage } = this;
if (appConfigLoading) {
return false;
}
return (
<div className={`${baseClass}__smtp-warning-wrapper`}>
<WarningBanner
message="Email is not currently configured in Fleet. User management features require email."
onResolve={goToAppConfigPage}
shouldShowWarning={!config.configured}
/>
</div>
);
}
renderUsersAndInvites = () => {
const { invites, users } = this.props;
const { renderUserBlock } = this;
return (
<div className={`${baseClass}__users`}>
{users.map((user, idx) => {
return renderUserBlock(user, idx);
})}
{invites.map((user, idx) => {
return renderUserBlock(user, idx, { invite: true });
})}
</div>
);
}
render () {
const { renderModal, renderSmtpWarning, renderUsersAndInvites, toggleInviteUserModal } = this;
const { config, invites, loadingInvites, loadingUsers, users } = this.props;
const resourcesCount = users.length + invites.length;
if (loadingInvites || loadingUsers) {
return false;
}
return (
<div className={`${baseClass} body-wrap`}>
<div className={`${baseClass}__heading-wrapper`}>
<h1 className={`${baseClass}__user-count`}>Listing {resourcesCount} users</h1>
<div className={`${baseClass}__add-user-wrap`}>
<Button
className={`${baseClass}__add-user-btn`}
disabled={!config.configured}
onClick={toggleInviteUserModal}
title={config.configured ? 'Add User' : 'Email must be configured to add users'}
>
Add User
</Button>
</div>
</div>
{renderSmtpWarning()}
{renderUsersAndInvites()}
{renderModal()}
</div>
);
}
}
const mapStateToProps = (state) => {
const stateEntityGetter = entityGetter(state);
const { config } = state.app;
const { loading: appConfigLoading } = state.app;
const { user: currentUser } = state.auth;
const { entities: users } = stateEntityGetter.get('users');
const { entities: invites } = stateEntityGetter.get('invites');
const { errors: inviteErrors, loading: loadingInvites } = state.entities.invites;
const { errors: userErrors, loading: loadingUsers } = state.entities.users;
return {
appConfigLoading,
config,
currentUser,
inviteErrors,
invites,
loadingInvites,
loadingUsers,
userErrors,
users,
};
};
export default connect(mapStateToProps)(UserManagementPage);