import React, { PureComponent } from 'react'; import { connect, ConnectedProps } from 'react-redux'; import { NavModelItem } from '@grafana/data'; import { featureEnabled } from '@grafana/runtime'; import { Page } from 'app/core/components/Page/Page'; import config from 'app/core/config'; import { contextSrv } from 'app/core/core'; import { GrafanaRouteComponentProps } from 'app/core/navigation/types'; import { StoreState, UserDTO, UserOrg, UserSession, SyncInfo, UserAdminError, AccessControlAction } from 'app/types'; import { UserLdapSyncInfo } from './UserLdapSyncInfo'; import { UserOrgs } from './UserOrgs'; import { UserPermissions } from './UserPermissions'; import { UserProfile } from './UserProfile'; import { UserSessions } from './UserSessions'; import { loadAdminUserPage, revokeSession, revokeAllSessions, updateUser, setUserPassword, disableUser, enableUser, deleteUser, updateUserPermissions, addOrgUser, updateOrgUserRole, deleteOrgUser, syncLdapUser, } from './state/actions'; interface OwnProps extends GrafanaRouteComponentProps<{ id: string }> { user?: UserDTO; orgs: UserOrg[]; sessions: UserSession[]; ldapSyncInfo?: SyncInfo; isLoading: boolean; error?: UserAdminError; } const SyncedOAuthLabels: string[] = ['GitHub', 'GitLab', 'AzureAD', 'OAuth']; export class UserAdminPage extends PureComponent { async componentDidMount() { const { match, loadAdminUserPage } = this.props; loadAdminUserPage(parseInt(match.params.id, 10)); } onUserUpdate = (user: UserDTO) => { this.props.updateUser(user); }; onPasswordChange = (password: string) => { const { user, setUserPassword } = this.props; user && setUserPassword(user.id, password); }; onUserDelete = (userId: number) => { this.props.deleteUser(userId); }; onUserDisable = (userId: number) => { this.props.disableUser(userId); }; onUserEnable = (userId: number) => { this.props.enableUser(userId); }; onGrafanaAdminChange = (isGrafanaAdmin: boolean) => { const { user, updateUserPermissions } = this.props; user && updateUserPermissions(user.id, isGrafanaAdmin); }; onOrgRemove = (orgId: number) => { const { user, deleteOrgUser } = this.props; user && deleteOrgUser(user.id, orgId); }; onOrgRoleChange = (orgId: number, newRole: string) => { const { user, updateOrgUserRole } = this.props; user && updateOrgUserRole(user.id, orgId, newRole); }; onOrgAdd = (orgId: number, role: string) => { const { user, addOrgUser } = this.props; user && addOrgUser(user, orgId, role); }; onSessionRevoke = (tokenId: number) => { const { user, revokeSession } = this.props; user && revokeSession(tokenId, user.id); }; onAllSessionsRevoke = () => { const { user, revokeAllSessions } = this.props; user && revokeAllSessions(user.id); }; onUserSync = () => { const { user, syncLdapUser } = this.props; user && syncLdapUser(user.id); }; render() { const { user, orgs, sessions, ldapSyncInfo, isLoading } = this.props; const isLDAPUser = user?.isExternal && user?.authLabels?.includes('LDAP'); const canReadSessions = contextSrv.hasPermission(AccessControlAction.UsersAuthTokenList); const canReadLDAPStatus = contextSrv.hasPermission(AccessControlAction.LDAPStatusRead); const isOAuthUserWithSkippableSync = user?.isExternal && user?.authLabels?.some((r) => SyncedOAuthLabels.includes(r)); const isSAMLUser = user?.isExternal && user?.authLabels?.includes('SAML'); const isGoogleUser = user?.isExternal && user?.authLabels?.includes('Google'); const isAuthProxyUser = user?.isExternal && user?.authLabels?.includes('Auth Proxy'); const isUserSynced = !config.auth.DisableSyncLock && ((user?.isExternal && !(isAuthProxyUser || isGoogleUser || isOAuthUserWithSkippableSync || isSAMLUser || isLDAPUser)) || (!config.auth.OAuthSkipOrgRoleUpdateSync && isOAuthUserWithSkippableSync) || (!config.auth.SAMLSkipOrgRoleSync && isSAMLUser) || (!config.auth.LDAPSkipOrgRoleSync && isLDAPUser)); const pageNav: NavModelItem = { text: user?.login ?? '', icon: 'shield', breadcrumbs: [{ title: 'Users', url: 'admin/users' }], subTitle: 'Manage settings for an individual user.', }; return ( {user && ( <> {!config.auth.LDAPSkipOrgRoleSync && isLDAPUser && featureEnabled('ldapsync') && ldapSyncInfo && canReadLDAPStatus && ( )} )} {orgs && ( )} {sessions && canReadSessions && ( )} ); } } const mapStateToProps = (state: StoreState) => ({ user: state.userAdmin.user, sessions: state.userAdmin.sessions, orgs: state.userAdmin.orgs, ldapSyncInfo: state.ldap.syncInfo, isLoading: state.userAdmin.isLoading, error: state.userAdmin.error, }); const mapDispatchToProps = { loadAdminUserPage, updateUser, setUserPassword, disableUser, enableUser, deleteUser, updateUserPermissions, addOrgUser, updateOrgUserRole, deleteOrgUser, revokeSession, revokeAllSessions, syncLdapUser, }; const connector = connect(mapStateToProps, mapDispatchToProps); type Props = OwnProps & ConnectedProps; export default connector(UserAdminPage);