mirror of
https://github.com/pgadmin-org/pgadmin4.git
synced 2025-02-25 18:55:31 -06:00
1) String literals should not be duplicated. 2) Prefer using an optional chain expression instead, as it's more concise and easier to read. 3) Expected the Promise rejection reason to be an Error.
439 lines
14 KiB
JavaScript
439 lines
14 KiB
JavaScript
/////////////////////////////////////////////////////////////
|
|
//
|
|
// pgAdmin 4 - PostgreSQL Tools
|
|
//
|
|
// Copyright (C) 2013 - 2024, The pgAdmin Development Team
|
|
// This software is released under the PostgreSQL Licence
|
|
//
|
|
//////////////////////////////////////////////////////////////
|
|
import React from 'react';
|
|
import { styled } from '@mui/material/styles';
|
|
import SchemaView from '../../../../static/js/SchemaView';
|
|
import BaseUISchema from '../../../../static/js/SchemaView/base_schema.ui';
|
|
import pgAdmin from 'sources/pgadmin';
|
|
import gettext from 'sources/gettext';
|
|
import url_for from 'sources/url_for';
|
|
import PropTypes from 'prop-types';
|
|
import getApiInstance, { parseApiError } from '../../../../static/js/api_instance';
|
|
import {AUTH_METHODS} from 'pgadmin.browser.constants';
|
|
import current_user from 'pgadmin.user_management.current_user';
|
|
import { isEmptyString } from '../../../../static/js/validators';
|
|
import { showChangeOwnership } from '../../../../static/js/Dialogs/index';
|
|
import { BROWSER_PANELS } from '../../../../browser/static/js/constants';
|
|
import _ from 'lodash';
|
|
|
|
const StyledSchemaView = styled(SchemaView)(({theme}) => ({
|
|
'& .UserManagementDialog-root': {
|
|
...theme.mixins.tabPanel,
|
|
padding: 0,
|
|
}
|
|
}));
|
|
|
|
class UserManagementCollection extends BaseUISchema {
|
|
constructor(authSources, roleOptions) {
|
|
super({
|
|
id: undefined,
|
|
username: undefined,
|
|
email: undefined,
|
|
active: true,
|
|
role: '2',
|
|
newPassword: undefined,
|
|
confirmPassword: undefined,
|
|
locked: false,
|
|
auth_source: AUTH_METHODS['INTERNAL']
|
|
});
|
|
|
|
this.authOnlyInternal = (current_user['auth_sources'].length == 1 &&
|
|
current_user['auth_sources'].includes(AUTH_METHODS['INTERNAL']));
|
|
this.authSources = authSources;
|
|
this.roleOptions = roleOptions;
|
|
}
|
|
|
|
get idAttribute() {
|
|
return 'id';
|
|
}
|
|
|
|
isUserNameEnabled(state) {
|
|
return !(this.authOnlyInternal || state.auth_source == AUTH_METHODS['INTERNAL']);
|
|
}
|
|
|
|
isEditable(state) {
|
|
return state.id != current_user['id'];
|
|
}
|
|
|
|
get baseFields() {
|
|
let obj = this;
|
|
return [
|
|
{
|
|
id: 'auth_source', label: gettext('Authentication source'),
|
|
cell: (state)=> {
|
|
return {
|
|
cell: 'select',
|
|
options: ()=> {
|
|
if (obj.isNew(state)) {
|
|
return Promise.resolve(obj.authSources.filter((s)=> current_user['auth_sources'].includes(s.value)));
|
|
}
|
|
return Promise.resolve(obj.authSources);
|
|
},
|
|
optionsReloadBasis: obj.isNew(state)
|
|
};
|
|
},
|
|
minWidth: 110, width: 110,
|
|
controlProps: {
|
|
allowClear: false,
|
|
openOnEnter: false,
|
|
first_empty: false,
|
|
},
|
|
visible: function() {
|
|
return !obj.authOnlyInternal;
|
|
},
|
|
editable: function(state) {
|
|
return (obj.isNew(state) && !obj.authOnlyInternal);
|
|
}
|
|
}, {
|
|
id: 'username', label: gettext('Username'), cell: 'text',
|
|
minWidth: 90, width: 90,
|
|
deps: ['auth_source'],
|
|
depChange: (state)=>{
|
|
if (obj.isUserNameEnabled(state) && obj.isNew(state) && !isEmptyString(obj.username)) {
|
|
return {username: undefined};
|
|
}
|
|
},
|
|
editable: (state)=> {
|
|
return obj.isUserNameEnabled(state);
|
|
}
|
|
}, {
|
|
id: 'email', label: gettext('Email'), cell: 'text',
|
|
minWidth: 90, width: 90, deps: ['id'],
|
|
editable: (state)=> {
|
|
if (obj.isNew(state))
|
|
return true;
|
|
|
|
return obj.isEditable(state) && state.auth_source != AUTH_METHODS['INTERNAL'];
|
|
}
|
|
}, {
|
|
id: 'role', label: gettext('Role'), cell: 'select',
|
|
options: obj.roleOptions, minWidth: 95, width: 95,
|
|
controlProps: {
|
|
allowClear: false,
|
|
openOnEnter: false,
|
|
first_empty: false,
|
|
},
|
|
editable: (state)=> {
|
|
return obj.isEditable(state);
|
|
}
|
|
}, {
|
|
id: 'active', label: gettext('Active'), cell: 'switch', width: 60, enableResizing: false,
|
|
editable: (state)=> {
|
|
return obj.isEditable(state);
|
|
}
|
|
}, {
|
|
id: 'newPassword', label: gettext('New password'), cell: 'password',
|
|
minWidth: 90, width: 90, deps: ['auth_source'], controlProps: {
|
|
autoComplete: 'new-password',
|
|
},
|
|
editable: (state)=> {
|
|
return obj.isEditable(state) && state.auth_source == AUTH_METHODS['INTERNAL'];
|
|
}
|
|
}, {
|
|
id: 'confirmPassword', label: gettext('Confirm password'), cell: 'password',
|
|
minWidth: 90, width: 90, deps: ['auth_source'], controlProps: {
|
|
autoComplete: 'new-password',
|
|
},
|
|
editable: (state)=> {
|
|
return obj.isEditable(state) && state.auth_source == AUTH_METHODS['INTERNAL'];
|
|
}
|
|
}, {
|
|
id: 'locked', label: gettext('Locked'), cell: 'switch', width: 60, enableResizing: false,
|
|
editable: (state)=> {
|
|
return state.locked;
|
|
}
|
|
}
|
|
];
|
|
}
|
|
|
|
validate(state, setError) {
|
|
let msg;
|
|
let obj = this;
|
|
let minPassLen = pgAdmin.password_length_min;
|
|
if (obj.isUserNameEnabled(state) && isEmptyString(state.username)) {
|
|
msg = gettext('Username cannot be empty');
|
|
setError('username', msg);
|
|
return true;
|
|
} else {
|
|
setError('username', null);
|
|
}
|
|
|
|
if (state.auth_source != AUTH_METHODS['INTERNAL']) {
|
|
if (obj.isNew(state) && obj.top?._sessData?.userManagement) {
|
|
for (let i=0; i < obj.top._sessData.userManagement.length; i++) {
|
|
if (obj.top._sessData.userManagement[i]?.id &&
|
|
obj.top._sessData.userManagement[i].username.toLowerCase() == state.username.toLowerCase() &&
|
|
obj.top._sessData.userManagement[i].auth_source == state.auth_source) {
|
|
msg = gettext('User name \'%s\' already exists', state.username);
|
|
setError('username', msg);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (state.auth_source == AUTH_METHODS['INTERNAL']) {
|
|
let email_filter = /^[a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/;
|
|
if (isEmptyString(state.email)) {
|
|
msg = gettext('Email cannot be empty');
|
|
setError('email', msg);
|
|
return true;
|
|
} else if (!email_filter.test(state.email)) {
|
|
msg = gettext('Invalid email address: %s', state.email);
|
|
setError('email', msg);
|
|
return true;
|
|
} else {
|
|
setError('email', null);
|
|
}
|
|
|
|
if (obj.isNew(state) && obj.top?._sessData?.userManagement) {
|
|
for (let i=0; i < obj.top._sessData.userManagement.length; i++) {
|
|
if (obj.top._sessData.userManagement[i]?.id &&
|
|
obj.top._sessData.userManagement[i].email?.toLowerCase() == state.email?.toLowerCase()) {
|
|
msg = gettext('Email address \'%s\' already exists', state.email);
|
|
setError('email', msg);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (obj.isNew(state) && isEmptyString(state.newPassword)) {
|
|
msg = gettext('Password cannot be empty for user %s', state.email);
|
|
setError('newPassword', msg);
|
|
return true;
|
|
} else if (state.newPassword?.length < minPassLen) {
|
|
msg = gettext('Password must be at least %s characters for user %s', minPassLen, state.email);
|
|
setError('newPassword', msg);
|
|
return true;
|
|
} else {
|
|
setError('newPassword', null);
|
|
}
|
|
|
|
if (obj.isNew(state) && isEmptyString(state.confirmPassword)) {
|
|
msg = gettext('Confirm Password cannot be empty for user %s', state.email);
|
|
setError('confirmPassword', msg);
|
|
return true;
|
|
} else {
|
|
setError('confirmPassword', null);
|
|
}
|
|
|
|
if (state.newPassword !== state.confirmPassword) {
|
|
msg = gettext('Passwords do not match for user %s', state.email);
|
|
setError('confirmPassword', msg);
|
|
return true;
|
|
} else {
|
|
setError('confirmPassword', null);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
class UserManagementSchema extends BaseUISchema {
|
|
constructor(authSources, roleOptions) {
|
|
super({refreshBrowserTree: false});
|
|
this.userManagementCollObj = new UserManagementCollection(authSources, roleOptions);
|
|
this.changeOwnership = false;
|
|
}
|
|
|
|
deleteUser(deleteRow) {
|
|
pgAdmin.Browser.notifier.confirm(
|
|
gettext('Delete user?'),
|
|
gettext('Are you sure you wish to delete this user?'),
|
|
deleteRow,
|
|
function() {
|
|
return true;
|
|
}
|
|
);
|
|
}
|
|
|
|
get baseFields() {
|
|
let obj = this;
|
|
const api = getApiInstance();
|
|
return [
|
|
{
|
|
id: 'userManagement', label: '', type: 'collection', schema: obj.userManagementCollObj,
|
|
canAdd: true, canDelete: true, isFullTab: true, group: 'temp_user',
|
|
addOnTop: true,
|
|
canDeleteRow: (row)=>{
|
|
return row['id'] != current_user['id'];
|
|
},
|
|
onDelete: (row, deleteRow)=> {
|
|
if (_.isUndefined(row['id'])) {
|
|
deleteRow();
|
|
return;
|
|
}
|
|
let deletedUser = {'id': row['id'], 'name': !isEmptyString(row['email']) ? row['email'] : row['username']};
|
|
api.get(url_for('user_management.shared_servers', {'uid': row['id']}))
|
|
.then((res)=>{
|
|
if (res.data?.data?.shared_servers > 0) {
|
|
api.get(url_for('user_management.admin_users', {'uid': row['id']}))
|
|
.then((result)=>{
|
|
showChangeOwnership(gettext('Change ownership'),
|
|
result?.data?.data?.result?.data,
|
|
res?.data?.data?.shared_servers,
|
|
deletedUser,
|
|
()=> {
|
|
this.changeOwnership = true;
|
|
deleteRow();
|
|
}
|
|
);
|
|
})
|
|
.catch((err)=>{
|
|
pgAdmin.Browser.notifier.error(parseApiError(err));
|
|
});
|
|
} else {
|
|
obj.deleteUser(deleteRow);
|
|
}
|
|
})
|
|
.catch((err)=>{
|
|
pgAdmin.Browser.notifier.error(parseApiError(err));
|
|
obj.deleteUser(deleteRow);
|
|
});
|
|
},
|
|
canSearch: true
|
|
},
|
|
{
|
|
id: 'refreshBrowserTree', visible: false, type: 'switch',
|
|
deps: ['userManagement'], depChange: ()=> {
|
|
return { refreshBrowserTree: this.changeOwnership };
|
|
}
|
|
}
|
|
];
|
|
}
|
|
}
|
|
|
|
function UserManagementDialog({onClose}) {
|
|
|
|
const [authSources, setAuthSources] = React.useState([]);
|
|
const [roles, setRoles] = React.useState([]);
|
|
const api = getApiInstance();
|
|
|
|
React.useEffect(async ()=>{
|
|
try {
|
|
api.get(url_for('user_management.auth_sources'))
|
|
.then(res=>{
|
|
setAuthSources(res.data);
|
|
})
|
|
.catch((err)=>{
|
|
pgAdmin.Browser.notifier.error(err);
|
|
});
|
|
|
|
api.get(url_for('user_management.roles'))
|
|
.then(res=>{
|
|
setRoles(res.data);
|
|
})
|
|
.catch((err)=>{
|
|
pgAdmin.Browser.notifier.error(parseApiError(err));
|
|
});
|
|
} catch (error) {
|
|
pgAdmin.Browser.notifier.error(parseApiError(error));
|
|
}
|
|
}, []);
|
|
|
|
const onSaveClick = (_isNew, changeData)=>{
|
|
return new Promise((resolve, reject)=>{
|
|
try {
|
|
if (changeData['refreshBrowserTree']) {
|
|
// Confirmation dialog to refresh the browser tree.
|
|
pgAdmin.Browser.notifier.confirm(
|
|
gettext('Object explorer tree refresh required'),
|
|
gettext('The ownership of the shared server was changed or the shared server was deleted, so the object explorer tree refresh is required. Do you wish to refresh the tree?'),
|
|
function () {
|
|
pgAdmin.Browser.tree.destroy();
|
|
},
|
|
function () {
|
|
return true;
|
|
},
|
|
gettext('Refresh'),
|
|
gettext('Later')
|
|
);
|
|
}
|
|
api.post(url_for('user_management.save'), changeData['userManagement'])
|
|
.then(()=>{
|
|
pgAdmin.Browser.notifier.success('Users Saved Successfully');
|
|
resolve();
|
|
onClose();
|
|
})
|
|
.catch((err)=>{
|
|
reject(new Error(err));
|
|
});
|
|
} catch (error) {
|
|
reject(parseApiError(error));
|
|
}
|
|
});
|
|
};
|
|
|
|
const authSourcesOptions = authSources.map((m)=>({
|
|
label: m.label,
|
|
value: m.value,
|
|
}));
|
|
|
|
if(authSourcesOptions.length <= 0) {
|
|
return <></>;
|
|
}
|
|
|
|
const roleOptions = roles.map((m)=>({
|
|
label: m.name,
|
|
value: m.id,
|
|
}));
|
|
|
|
if(roleOptions.length <= 0) {
|
|
return <></>;
|
|
}
|
|
|
|
const onDialogHelp = () => {
|
|
window.open(url_for('help.static', { 'filename': 'user_management.html' }), 'pgadmin_help');
|
|
};
|
|
|
|
return <StyledSchemaView
|
|
formType={'dialog'}
|
|
getInitData={()=>{ return new Promise((resolve, reject)=>{
|
|
api.get(url_for('user_management.users'))
|
|
.then((res)=>{
|
|
resolve({userManagement:res.data});
|
|
})
|
|
.catch((err)=>{
|
|
reject(new Error(err));
|
|
});
|
|
}); }}
|
|
schema={new UserManagementSchema(authSourcesOptions, roleOptions)}
|
|
viewHelperProps={{
|
|
mode: 'edit',
|
|
}}
|
|
onSave={onSaveClick}
|
|
onClose={onClose}
|
|
onHelp={onDialogHelp}
|
|
hasSQL={false}
|
|
disableSqlHelp={true}
|
|
isTabView={false}
|
|
formClassName='UserManagementDialog-root'
|
|
/>;
|
|
}
|
|
|
|
UserManagementDialog.propTypes = {
|
|
onClose: PropTypes.func
|
|
};
|
|
|
|
export function showUserManagement() {
|
|
const panelTitle = gettext('User Management');
|
|
const panelId = BROWSER_PANELS.USER_MANAGEMENT;
|
|
pgAdmin.Browser.docker.openDialog({
|
|
id: panelId,
|
|
title: panelTitle,
|
|
manualClose: false,
|
|
content: (
|
|
<UserManagementDialog
|
|
onClose={()=>{pgAdmin.Browser.docker.close(panelId);}}
|
|
/>
|
|
)
|
|
}, pgAdmin.Browser.stdW.lg, pgAdmin.Browser.stdH.md);
|
|
}
|