Port Maintenance dialog to React. Fixes #7019
parent
65a6c18263
commit
ad862f4084
Binary file not shown.
Before Width: | Height: | Size: 43 KiB After Width: | Height: | Size: 87 KiB |
|
@ -15,6 +15,7 @@ Housekeeping
|
||||||
************
|
************
|
||||||
|
|
||||||
| `Issue #7018 <https://redmine.postgresql.org/issues/7018>`_ - Port Restore dialog to React.
|
| `Issue #7018 <https://redmine.postgresql.org/issues/7018>`_ - Port Restore dialog to React.
|
||||||
|
| `Issue #7019 <https://redmine.postgresql.org/issues/7019>`_ - Port Maintenance dialog to React.
|
||||||
|
|
||||||
Bug fixes
|
Bug fixes
|
||||||
*********
|
*********
|
||||||
|
|
|
@ -13,6 +13,7 @@ import {Accordion, AccordionSummary, AccordionDetails} from '@material-ui/core';
|
||||||
import ExpandMoreIcon from '@material-ui/icons/ExpandMore';
|
import ExpandMoreIcon from '@material-ui/icons/ExpandMore';
|
||||||
import SaveIcon from '@material-ui/icons/Save';
|
import SaveIcon from '@material-ui/icons/Save';
|
||||||
import PublishIcon from '@material-ui/icons/Publish';
|
import PublishIcon from '@material-ui/icons/Publish';
|
||||||
|
import DoneIcon from '@material-ui/icons/Done';
|
||||||
import SettingsBackupRestoreIcon from '@material-ui/icons/SettingsBackupRestore';
|
import SettingsBackupRestoreIcon from '@material-ui/icons/SettingsBackupRestore';
|
||||||
import CloseIcon from '@material-ui/icons/Close';
|
import CloseIcon from '@material-ui/icons/Close';
|
||||||
import InfoIcon from '@material-ui/icons/InfoRounded';
|
import InfoIcon from '@material-ui/icons/InfoRounded';
|
||||||
|
@ -687,6 +688,8 @@ function SchemaDialogView({
|
||||||
const getButtonIcon = () => {
|
const getButtonIcon = () => {
|
||||||
if(props.customSaveBtnIconType == 'upload') {
|
if(props.customSaveBtnIconType == 'upload') {
|
||||||
return <PublishIcon />;
|
return <PublishIcon />;
|
||||||
|
} else if(props.customSaveBtnIconType == 'done') {
|
||||||
|
return <DoneIcon />;
|
||||||
}
|
}
|
||||||
return <SaveIcon />;
|
return <SaveIcon />;
|
||||||
};
|
};
|
||||||
|
@ -720,7 +723,7 @@ function SchemaDialogView({
|
||||||
<DefaultButton data-test="Reset" onClick={onResetClick} startIcon={<SettingsBackupRestoreIcon />} disabled={!dirty || saving} className={classes.buttonMargin}>
|
<DefaultButton data-test="Reset" onClick={onResetClick} startIcon={<SettingsBackupRestoreIcon />} disabled={!dirty || saving} className={classes.buttonMargin}>
|
||||||
{gettext('Reset')}
|
{gettext('Reset')}
|
||||||
</DefaultButton>
|
</DefaultButton>
|
||||||
<PrimaryButton data-test="Save" onClick={onSaveClick} startIcon={ButtonIcon} disabled={!dirty || saving || Boolean(formErr.name) || !formReady}>
|
<PrimaryButton data-test="Save" onClick={onSaveClick} startIcon={ButtonIcon} disabled={ !(viewHelperProps.mode === 'edit' ? dirty : true) || saving || Boolean(formErr.name) || !formReady}>
|
||||||
{props.customSaveBtnName ? gettext(props.customSaveBtnName) : gettext('Save')}
|
{props.customSaveBtnName ? gettext(props.customSaveBtnName) : gettext('Save')}
|
||||||
</PrimaryButton>
|
</PrimaryButton>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
|
@ -127,7 +127,7 @@ define([
|
||||||
Alertify.grantWizardDialog().elements.modal.style.overflow='visible';
|
Alertify.grantWizardDialog().elements.modal.style.overflow='visible';
|
||||||
Alertify.grantWizardDialog().elements.dimmer.style.display='none';
|
Alertify.grantWizardDialog().elements.dimmer.style.display='none';
|
||||||
}
|
}
|
||||||
}, 500);
|
}, 10);
|
||||||
|
|
||||||
},
|
},
|
||||||
prepare: function () {
|
prepare: function () {
|
||||||
|
|
|
@ -93,7 +93,7 @@ export default class ImportExportServersModule {
|
||||||
Alertify.importExportWizardDialog().elements.modal.style.overflow='visible';
|
Alertify.importExportWizardDialog().elements.modal.style.overflow='visible';
|
||||||
Alertify.importExportWizardDialog().elements.dimmer.style.display='none';
|
Alertify.importExportWizardDialog().elements.dimmer.style.display='none';
|
||||||
}
|
}
|
||||||
}, 500);
|
}, 10);
|
||||||
|
|
||||||
},
|
},
|
||||||
prepare: function () {
|
prepare: function () {
|
||||||
|
|
|
@ -139,9 +139,9 @@ class Message(IProcessDesc):
|
||||||
res = _('VACUUM ({0})')
|
res = _('VACUUM ({0})')
|
||||||
|
|
||||||
opts = []
|
opts = []
|
||||||
if self.data['vacuum_full']:
|
if 'vacuum_full' in self.data and self.data['vacuum_full']:
|
||||||
opts.append(_('FULL'))
|
opts.append(_('FULL'))
|
||||||
if self.data['vacuum_freeze']:
|
if 'vacuum_freeze' in self.data and self.data['vacuum_freeze']:
|
||||||
opts.append(_('FREEZE'))
|
opts.append(_('FREEZE'))
|
||||||
if self.data['verbose']:
|
if self.data['verbose']:
|
||||||
opts.append(_('VERBOSE'))
|
opts.append(_('VERBOSE'))
|
||||||
|
|
|
@ -8,149 +8,27 @@
|
||||||
//////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
import Notify from '../../../../static/js/helpers/Notifier';
|
import Notify from '../../../../static/js/helpers/Notifier';
|
||||||
|
import {getUtilityView} from '../../../../browser/static/js/utility_view';
|
||||||
|
import getApiInstance from 'sources/api_instance';
|
||||||
|
import MaintenanceSchema, {getVacuumSchema} from './maintenance.ui';
|
||||||
|
|
||||||
define([
|
define([
|
||||||
'sources/gettext', 'sources/url_for', 'jquery', 'underscore',
|
'sources/gettext', 'sources/url_for', 'sources/pgadmin', 'pgadmin.browser',
|
||||||
'pgadmin.alertifyjs', 'sources/pgadmin', 'pgadmin.browser', 'backbone',
|
|
||||||
'backgrid', 'backform', 'sources/utils',
|
|
||||||
'tools/maintenance/static/js/menu_utils',
|
'tools/maintenance/static/js/menu_utils',
|
||||||
'sources/nodes/supported_database_node',
|
'sources/nodes/supported_database_node'
|
||||||
'pgadmin.backform', 'pgadmin.backgrid',
|
|
||||||
'pgadmin.browser.node.ui',
|
|
||||||
], function(
|
], function(
|
||||||
gettext, url_for, $, _, Alertify, pgAdmin, pgBrowser, Backbone, Backgrid,
|
gettext, url_for, pgAdmin, pgBrowser, menuUtils, supportedNodes
|
||||||
Backform, commonUtils,
|
|
||||||
menuUtils, supportedNodes
|
|
||||||
) {
|
) {
|
||||||
|
|
||||||
pgAdmin = pgAdmin || window.pgAdmin || {};
|
pgAdmin = pgAdmin || window.pgAdmin || {};
|
||||||
|
|
||||||
var pgTools = pgAdmin.Tools = pgAdmin.Tools || {};
|
var pgTools = pgAdmin.Tools = pgAdmin.Tools || {};
|
||||||
|
const api = getApiInstance();
|
||||||
|
|
||||||
// Return back, this has been called more than once
|
// Return back, this has been called more than once
|
||||||
if (pgAdmin.Tools.maintenance)
|
if (pgAdmin.Tools.maintenance)
|
||||||
return pgAdmin.Tools.maintenance;
|
return pgAdmin.Tools.maintenance;
|
||||||
|
|
||||||
// Main model for Maintenance functionality
|
|
||||||
var MaintenanceModel = Backbone.Model.extend({
|
|
||||||
defaults: {
|
|
||||||
op: 'VACUUM',
|
|
||||||
vacuum_full: false,
|
|
||||||
vacuum_freeze: false,
|
|
||||||
vacuum_analyze: false,
|
|
||||||
verbose: true,
|
|
||||||
},
|
|
||||||
initialize: function() {
|
|
||||||
var node_info = arguments[1]['node_info'];
|
|
||||||
// If node is Unique or Primary key then set op to reindex
|
|
||||||
if ('primary_key' in node_info || 'unique_constraint' in node_info ||
|
|
||||||
'index' in node_info) {
|
|
||||||
this.set('op', 'REINDEX');
|
|
||||||
this.set('verbose', false);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
schema: [{
|
|
||||||
id: 'op',
|
|
||||||
label: gettext('Maintenance operation'),
|
|
||||||
cell: 'string',
|
|
||||||
type: 'radioModern',
|
|
||||||
controlsClassName: 'pgadmin-controls col-12 col-sm-8',
|
|
||||||
controlLabelClassName: 'control-label col-sm-4 col-12',
|
|
||||||
group: gettext('Options'),
|
|
||||||
value: 'VACUUM',
|
|
||||||
options: [{
|
|
||||||
'label': 'VACUUM',
|
|
||||||
'value': 'VACUUM',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'label': 'ANALYZE',
|
|
||||||
'value': 'ANALYZE',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'label': 'REINDEX',
|
|
||||||
'value': 'REINDEX',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'label': 'CLUSTER',
|
|
||||||
'value': 'CLUSTER',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'nested',
|
|
||||||
control: 'fieldset',
|
|
||||||
label: gettext('Vacuum'),
|
|
||||||
group: gettext('Options'),
|
|
||||||
contentClass: 'row',
|
|
||||||
schema: [{
|
|
||||||
id: 'vacuum_full',
|
|
||||||
group: gettext('Vacuum'),
|
|
||||||
disabled: 'isDisabled',
|
|
||||||
type: 'switch',
|
|
||||||
extraToggleClasses: 'pg-el-sm-4',
|
|
||||||
controlLabelClassName: 'control-label pg-el-sm-5 pg-el-12',
|
|
||||||
controlsClassName: 'pgadmin-controls pg-el-sm-7 pg-el-12',
|
|
||||||
label: gettext('FULL'),
|
|
||||||
deps: ['op'],
|
|
||||||
}, {
|
|
||||||
id: 'vacuum_freeze',
|
|
||||||
deps: ['op'],
|
|
||||||
disabled: 'isDisabled',
|
|
||||||
type: 'switch',
|
|
||||||
extraToggleClasses: 'pg-el-sm-4',
|
|
||||||
controlLabelClassName: 'control-label pg-el-sm-5 pg-el-12',
|
|
||||||
controlsClassName: 'pgadmin-controls pg-el-sm-7 pg-el-12',
|
|
||||||
label: gettext('FREEZE'),
|
|
||||||
group: gettext('Vacuum'),
|
|
||||||
}, {
|
|
||||||
id: 'vacuum_analyze',
|
|
||||||
deps: ['op'],
|
|
||||||
disabled: 'isDisabled',
|
|
||||||
type: 'switch',
|
|
||||||
extraToggleClasses: 'pg-el-sm-4',
|
|
||||||
controlLabelClassName: 'control-label pg-el-sm-5 pg-el-12',
|
|
||||||
controlsClassName: 'pgadmin-controls pg-el-sm-7 pg-el-12',
|
|
||||||
label: gettext('ANALYZE'),
|
|
||||||
group: gettext('Vacuum'),
|
|
||||||
}],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'verbose',
|
|
||||||
group: gettext('Options'),
|
|
||||||
deps: ['op'],
|
|
||||||
type: 'switch',
|
|
||||||
label: gettext('Verbose Messages'),
|
|
||||||
disabled: 'isDisabled',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
// Enable/Disable the items based on the user maintenance operation
|
|
||||||
// selection.
|
|
||||||
isDisabled: function(m) {
|
|
||||||
var node_info = this.node_info;
|
|
||||||
|
|
||||||
switch (this.name) {
|
|
||||||
case 'vacuum_full':
|
|
||||||
case 'vacuum_freeze':
|
|
||||||
case 'vacuum_analyze':
|
|
||||||
return (m.get('op') != 'VACUUM');
|
|
||||||
case 'verbose':
|
|
||||||
if ('primary_key' in node_info || 'unique_constraint' in node_info ||
|
|
||||||
'index' in node_info) {
|
|
||||||
if (m.get('op') == 'REINDEX') {
|
|
||||||
setTimeout(function() {
|
|
||||||
m.set('verbose', false);
|
|
||||||
}, 10);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return m.get('op') == 'REINDEX';
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
pgTools.maintenance = {
|
pgTools.maintenance = {
|
||||||
init: function() {
|
init: function() {
|
||||||
|
|
||||||
|
@ -197,7 +75,33 @@ define([
|
||||||
}
|
}
|
||||||
pgBrowser.add_menus(menus);
|
pgBrowser.add_menus(menus);
|
||||||
},
|
},
|
||||||
|
getUISchema: function(treeItem) {
|
||||||
|
let treeNodeInfo = pgBrowser.tree.getTreeNodeHierarchy(treeItem);
|
||||||
|
|
||||||
|
return new MaintenanceSchema(
|
||||||
|
()=>getVacuumSchema(),
|
||||||
|
{
|
||||||
|
nodeInfo: treeNodeInfo
|
||||||
|
}
|
||||||
|
);
|
||||||
|
},
|
||||||
|
saveCallBack: function(data, dialog) {
|
||||||
|
if(data.errormsg) {
|
||||||
|
Notify.alert(
|
||||||
|
gettext('Utility not found'),
|
||||||
|
gettext(data.errormsg)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
Notify.success(data.data.info);
|
||||||
|
pgBrowser.Events.trigger('pgadmin-bgprocess:created', dialog);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
setExtraParameters(treeInfo) {
|
||||||
|
var extraData = {};
|
||||||
|
extraData['database'] = treeInfo.database._label;
|
||||||
|
extraData['save_btn_icon'] = 'done';
|
||||||
|
return extraData;
|
||||||
|
},
|
||||||
/*
|
/*
|
||||||
Open the dialog for the maintenance functionality
|
Open the dialog for the maintenance functionality
|
||||||
*/
|
*/
|
||||||
|
@ -224,12 +128,7 @@ define([
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!commonUtils.hasBinariesConfiguration(pgBrowser, server_data, Alertify)) {
|
var t = pgBrowser.tree;
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var self = this,
|
|
||||||
t = pgBrowser.tree;
|
|
||||||
|
|
||||||
i = item || t.selected();
|
i = item || t.selected();
|
||||||
|
|
||||||
|
@ -249,232 +148,50 @@ define([
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Alertify.MaintenanceDialog) {
|
|
||||||
Alertify.dialog('MaintenanceDialog', function factory() {
|
|
||||||
|
|
||||||
return {
|
|
||||||
main: function(title) {
|
|
||||||
this.set('title', title);
|
|
||||||
},
|
|
||||||
setup: function() {
|
|
||||||
return {
|
|
||||||
buttons: [{
|
|
||||||
text: '',
|
|
||||||
className: 'btn btn-primary-icon pull-left fa fa-info pg-alertify-icon-button',
|
|
||||||
attrs: {
|
|
||||||
name: 'object_help',
|
|
||||||
type: 'button',
|
|
||||||
url: 'maintenance.html',
|
|
||||||
label: gettext('Maintenance'),
|
|
||||||
'aria-label': gettext('Object Help'),
|
|
||||||
},
|
|
||||||
}, {
|
|
||||||
text: '',
|
|
||||||
key: 112,
|
|
||||||
className: 'btn btn-primary-icon pull-left fa fa-question pg-alertify-icon-button',
|
|
||||||
attrs: {
|
|
||||||
name: 'dialog_help',
|
|
||||||
type: 'button',
|
|
||||||
label: gettext('Maintenance'),
|
|
||||||
'aria-label': gettext('Help'),
|
|
||||||
url: url_for(
|
|
||||||
'help.static', {
|
|
||||||
'filename': 'maintenance_dialog.html',
|
|
||||||
}
|
|
||||||
),
|
|
||||||
},
|
|
||||||
}, {
|
|
||||||
text: gettext('Cancel'),
|
|
||||||
key: 27,
|
|
||||||
className: 'btn btn-secondary fa fa-lg fa-times pg-alertify-button',
|
|
||||||
'data-btn-name': 'cancel',
|
|
||||||
}, {
|
|
||||||
text: gettext('OK'),
|
|
||||||
key: 13,
|
|
||||||
className: 'btn btn-primary fa fa-lg fa-check pg-alertify-button',
|
|
||||||
'data-btn-name': 'ok',
|
|
||||||
}],
|
|
||||||
options: {
|
|
||||||
modal: 0,
|
|
||||||
pinnable: false,
|
|
||||||
//disable both padding and overflow control.
|
|
||||||
padding: !1,
|
|
||||||
overflow: !1,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
},
|
|
||||||
// Callback functions when click on the buttons of the Alertify dialogs
|
|
||||||
callback: function(e) {
|
|
||||||
var sel_item = pgBrowser.tree.selected(),
|
|
||||||
itemData = sel_item ? pgBrowser.tree.itemData(sel_item) : undefined,
|
|
||||||
sel_node = itemData && pgBrowser.Nodes[itemData._type];
|
|
||||||
|
|
||||||
if (e.button.element.name == 'dialog_help' || e.button.element.name == 'object_help') {
|
|
||||||
e.cancel = true;
|
|
||||||
pgBrowser.showHelp(e.button.element.name, e.button.element.getAttribute('url'),
|
|
||||||
sel_node, sel_item);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e.button['data-btn-name'] === 'ok') {
|
|
||||||
|
|
||||||
var schema = undefined,
|
|
||||||
table = undefined,
|
|
||||||
primary_key = undefined,
|
|
||||||
unique_constraint = undefined,
|
|
||||||
index = undefined;
|
|
||||||
|
|
||||||
if (!itemData)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var node_hierarchy = pgBrowser.tree.getTreeNodeHierarchy(sel_item);
|
|
||||||
|
|
||||||
if (node_hierarchy.schema != undefined) {
|
|
||||||
schema = node_hierarchy.schema._label;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (node_hierarchy.partition != undefined) {
|
|
||||||
table = node_hierarchy.partition._label;
|
|
||||||
} else if (node_hierarchy.table != undefined) {
|
|
||||||
table = node_hierarchy.table._label;
|
|
||||||
} else if (node_hierarchy.mview != undefined) {
|
|
||||||
table = node_hierarchy.mview._label;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (node_hierarchy.primary_key != undefined) {
|
|
||||||
primary_key = node_hierarchy.primary_key._label;
|
|
||||||
} else if (node_hierarchy.unique_constraint != undefined) {
|
|
||||||
unique_constraint = node_hierarchy.unique_constraint._label;
|
|
||||||
} else if (node_hierarchy.index != undefined) {
|
|
||||||
index = node_hierarchy.index._label;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.view.model.set({
|
|
||||||
'database': node_hierarchy.database._label,
|
|
||||||
'schema': schema,
|
|
||||||
'table': table,
|
|
||||||
'primary_key': primary_key,
|
|
||||||
'unique_constraint': unique_constraint,
|
|
||||||
'index': index,
|
|
||||||
});
|
|
||||||
|
|
||||||
$.ajax({
|
|
||||||
url: url_for(
|
|
||||||
'maintenance.create_job', {
|
|
||||||
'sid': node_hierarchy.server._id,
|
|
||||||
'did': node_hierarchy.database._id,
|
|
||||||
}),
|
|
||||||
method: 'POST',
|
|
||||||
data: {
|
|
||||||
'data': JSON.stringify(this.view.model.toJSON()),
|
|
||||||
},
|
|
||||||
})
|
|
||||||
.done(function(res) {
|
|
||||||
if (res.data && res.data.status) {
|
|
||||||
//Do nothing as we are creating the job and exiting from the main dialog
|
|
||||||
Notify.success(res.data.info);
|
|
||||||
pgBrowser.Events.trigger('pgadmin-bgprocess:created', self);
|
|
||||||
} else {
|
|
||||||
Notify.alert(
|
|
||||||
gettext('Maintenance job creation failed.'),
|
|
||||||
res.errormsg
|
|
||||||
);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.fail(function() {
|
|
||||||
Notify.alert(
|
|
||||||
gettext('Maintenance job creation failed.')
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
build: function() {
|
|
||||||
Alertify.pgDialogBuild.apply(this);
|
|
||||||
},
|
|
||||||
hooks: {
|
|
||||||
onclose: function() {
|
|
||||||
if (this.view) {
|
|
||||||
this.view.remove({
|
|
||||||
data: true,
|
|
||||||
internal: true,
|
|
||||||
silent: true,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
onshow: function() {
|
|
||||||
var container = $(this.elements.body).find('.tab-content:first > .tab-pane.active:first');
|
|
||||||
commonUtils.findAndSetFocus(container);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
prepare: function() {
|
|
||||||
// Main maintenance tool dialog container
|
|
||||||
var $container = $('<div class=\'maintenance_dlg\'></div>');
|
|
||||||
var tree = pgBrowser.tree,
|
|
||||||
sel_item = tree.selected(),
|
|
||||||
itemInfo = sel_item ? tree.itemData(sel_item) : undefined,
|
|
||||||
nodeData = itemInfo && pgBrowser.Nodes[itemInfo._type];
|
|
||||||
|
|
||||||
if (!itemInfo)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var treeData = tree.getTreeNodeHierarchy(sel_item);
|
|
||||||
|
|
||||||
var newModel = new MaintenanceModel({}, {
|
|
||||||
node_info: treeData,
|
|
||||||
}),
|
|
||||||
fields = Backform.generateViewSchema(
|
|
||||||
treeData, newModel, 'create', nodeData, treeData.server, true
|
|
||||||
);
|
|
||||||
|
|
||||||
var view = this.view = new Backform.Dialog({
|
|
||||||
el: $container,
|
|
||||||
model: newModel,
|
|
||||||
schema: fields,
|
|
||||||
});
|
|
||||||
|
|
||||||
$(this.elements.body.childNodes[0]).addClass('alertify_tools_dialog_properties obj_properties');
|
|
||||||
view.render();
|
|
||||||
|
|
||||||
// If node is Index, Unique or Primary key then disable vacuum & analyze button
|
|
||||||
if (itemInfo._type == 'primary_key' || itemInfo._type == 'unique_constraint' ||
|
|
||||||
itemInfo._type == 'index') {
|
|
||||||
var vacuum_analyze_btns = $container.find(
|
|
||||||
'.btn-group label.btn:lt(2)'
|
|
||||||
).addClass('disabled');
|
|
||||||
// Find reindex button element & add active class to it
|
|
||||||
var reindex_btn = vacuum_analyze_btns[1].nextElementSibling;
|
|
||||||
$(reindex_btn).trigger('click');
|
|
||||||
}
|
|
||||||
|
|
||||||
view.$el.attr('tabindex', -1);
|
|
||||||
this.elements.content.appendChild($container.get(0));
|
|
||||||
},
|
|
||||||
};
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const baseUrl = url_for('maintenance.utility_exists', {
|
const baseUrl = url_for('maintenance.utility_exists', {
|
||||||
'sid': server_data._id,
|
'sid': server_data._id,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let that = this;
|
||||||
// Check psql utility exists or not.
|
// Check psql utility exists or not.
|
||||||
$.ajax({
|
api({
|
||||||
url: baseUrl,
|
url: baseUrl,
|
||||||
type:'GET',
|
type:'GET',
|
||||||
})
|
})
|
||||||
.done(function(res) {
|
.then(function(res) {
|
||||||
if (!res.success) {
|
if (!res.data.success) {
|
||||||
Notify.alert(
|
Notify.alert(
|
||||||
gettext('Utility not found'),
|
gettext('Utility not found'),
|
||||||
res.errormsg
|
res.data.errormsg
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
|
} else{
|
||||||
|
|
||||||
|
pgBrowser.Node.registerUtilityPanel();
|
||||||
|
var panel = pgBrowser.Node.addUtilityPanel(pgBrowser.stdW.md),
|
||||||
|
j = panel.$container.find('.obj_properties').first();
|
||||||
|
|
||||||
|
var schema = that.getUISchema(item);
|
||||||
|
panel.title(gettext('Maintenance'));
|
||||||
|
panel.focus();
|
||||||
|
|
||||||
|
let urlShortcut = 'maintenance.create_job',
|
||||||
|
baseUrl = url_for(urlShortcut, {
|
||||||
|
'sid': treeInfo.server._id,
|
||||||
|
'did': treeInfo.database._id
|
||||||
|
});
|
||||||
|
let extraData = that.setExtraParameters(treeInfo);
|
||||||
|
|
||||||
|
var sqlHelpUrl = 'maintenance.html',
|
||||||
|
helpUrl = url_for('help.static', {
|
||||||
|
'filename': 'maintenance_dialog.html',
|
||||||
|
});
|
||||||
|
|
||||||
|
getUtilityView(
|
||||||
|
schema, treeInfo, 'select', 'dialog', j[0], panel, that.saveCallBack, extraData, 'OK', baseUrl, sqlHelpUrl, helpUrl);
|
||||||
}
|
}
|
||||||
// Open the Alertify dialog
|
|
||||||
Alertify.MaintenanceDialog(gettext('Maintenance...')).set('resizable', true)
|
|
||||||
.resizeTo(pgAdmin.Browser.stdW.md,pgAdmin.Browser.stdH.md);
|
|
||||||
})
|
})
|
||||||
.fail(function() {
|
.catch(function() {
|
||||||
Notify.alert(
|
Notify.alert(
|
||||||
gettext('Utility not found'),
|
gettext('Utility not found'),
|
||||||
gettext('Failed to fetch Utility information')
|
gettext('Failed to fetch Utility information')
|
||||||
|
|
|
@ -0,0 +1,158 @@
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// pgAdmin 4 - PostgreSQL Tools
|
||||||
|
//
|
||||||
|
// Copyright (C) 2013 - 2022, The pgAdmin Development Team
|
||||||
|
// This software is released under the PostgreSQL Licence
|
||||||
|
//
|
||||||
|
//////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
import BaseUISchema from 'sources/SchemaView/base_schema.ui';
|
||||||
|
import gettext from 'sources/gettext';
|
||||||
|
|
||||||
|
export class VacuumSchema extends BaseUISchema {
|
||||||
|
constructor(fieldOptions={}) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.fieldOptions = {
|
||||||
|
...fieldOptions,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
get idAttribute() {
|
||||||
|
return 'op';
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
get baseFields() {
|
||||||
|
return [{
|
||||||
|
id: 'vacuum_full',
|
||||||
|
group: gettext('Vacuum'),
|
||||||
|
disabled: function(state) {
|
||||||
|
if(state?.op) {
|
||||||
|
return (state.op != 'VACUUM');
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
type: 'switch',
|
||||||
|
label: gettext('FULL'),
|
||||||
|
deps: ['op'],
|
||||||
|
}, {
|
||||||
|
id: 'vacuum_freeze',
|
||||||
|
deps: ['op'],
|
||||||
|
disabled: function(state) {
|
||||||
|
if(state?.op) {
|
||||||
|
return (state.op != 'VACUUM');
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
type: 'switch',
|
||||||
|
label: gettext('FREEZE'),
|
||||||
|
group: gettext('Vacuum'),
|
||||||
|
}, {
|
||||||
|
id: 'vacuum_analyze',
|
||||||
|
deps: ['op'],
|
||||||
|
type: 'switch',
|
||||||
|
disabled: function(state) {
|
||||||
|
if(state?.op) {
|
||||||
|
return (state.op != 'VACUUM');
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
label: gettext('ANALYZE'),
|
||||||
|
group: gettext('Vacuum'),
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getVacuumSchema(fieldOptions) {
|
||||||
|
return new VacuumSchema(fieldOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//Maintenance Schema
|
||||||
|
export default class MaintenanceSchema extends BaseUISchema {
|
||||||
|
|
||||||
|
constructor(getVacuumSchema, fieldOptions = {}) {
|
||||||
|
super({
|
||||||
|
op: 'VACUUM',
|
||||||
|
verbose: true,
|
||||||
|
vacuum_full: false,
|
||||||
|
vacuum_freeze: false,
|
||||||
|
vacuum_analyze: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.fieldOptions = {
|
||||||
|
nodeInfo: null,
|
||||||
|
...fieldOptions,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.getVacuumSchema = getVacuumSchema;
|
||||||
|
this.nodeInfo = fieldOptions.nodeInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
get idAttribute() {
|
||||||
|
return 'id';
|
||||||
|
}
|
||||||
|
|
||||||
|
get baseFields() {
|
||||||
|
var obj = this;
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: 'op',
|
||||||
|
label: gettext('Maintenance operation'),
|
||||||
|
type: 'toggle',
|
||||||
|
group: gettext('Options'),
|
||||||
|
options: [
|
||||||
|
{
|
||||||
|
'label': gettext('VACUUM'),
|
||||||
|
value: 'VACUUM',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'label': gettext('ANALYZE'),
|
||||||
|
value: 'ANALYZE',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'label': gettext('REINDEX'),
|
||||||
|
value: 'REINDEX',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'label': gettext('CLUSTER'),
|
||||||
|
value: 'CLUSTER',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'nested-fieldset',
|
||||||
|
label: gettext('Type of objects'),
|
||||||
|
schema: obj.getVacuumSchema(),
|
||||||
|
group: gettext('Options'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'verbose',
|
||||||
|
group: gettext('Options'),
|
||||||
|
deps: ['op'],
|
||||||
|
type: 'switch',
|
||||||
|
label: gettext('Verbose Messages'),
|
||||||
|
disabled: function(state) {
|
||||||
|
var nodeInfo = this.nodeInfo;
|
||||||
|
if(state?.verbose) {
|
||||||
|
if ('primary_key' in nodeInfo || 'unique_constraint' in nodeInfo ||
|
||||||
|
'index' in nodeInfo) {
|
||||||
|
if (state.op == 'REINDEX') {
|
||||||
|
state.verbose = false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return state.op == 'REINDEX';
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,52 @@
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
//
|
||||||
|
// pgAdmin 4 - PostgreSQL Tools
|
||||||
|
//
|
||||||
|
// Copyright (C) 2013 - 2022, The pgAdmin Development Team
|
||||||
|
// This software is released under the PostgreSQL Licence
|
||||||
|
//
|
||||||
|
//////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
import React from 'react';
|
||||||
|
import '../helper/enzyme.helper';
|
||||||
|
import { createMount } from '@material-ui/core/test-utils';
|
||||||
|
import SchemaView from '../../../pgadmin/static/js/SchemaView';
|
||||||
|
import MaintenanceSchema, {getVacuumSchema} from '../../../pgadmin/tools/maintenance/static/js/maintenance.ui';
|
||||||
|
|
||||||
|
|
||||||
|
describe('MaintenanceSchema', ()=>{
|
||||||
|
let mount;
|
||||||
|
beforeAll(()=>{
|
||||||
|
mount = createMount();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
mount.cleanUp();
|
||||||
|
});
|
||||||
|
let backupSchemaObj = new MaintenanceSchema(
|
||||||
|
()=> getVacuumSchema(),
|
||||||
|
{
|
||||||
|
nodeInfo: {schema: {label: 'public'}, server: {version: 90400}}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
it('start maintenance', ()=>{
|
||||||
|
mount(<SchemaView
|
||||||
|
formType='dialog'
|
||||||
|
schema={backupSchemaObj}
|
||||||
|
viewHelperProps={{
|
||||||
|
mode: 'create',
|
||||||
|
}}
|
||||||
|
onSave={()=>{}}
|
||||||
|
onClose={()=>{}}
|
||||||
|
onHelp={()=>{}}
|
||||||
|
onDataChange={()=>{}}
|
||||||
|
confirmOnCloseReset={false}
|
||||||
|
hasSQL={false}
|
||||||
|
disableSqlHelp={false}
|
||||||
|
disableDialogHelp={false}
|
||||||
|
/>);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
Loading…
Reference in New Issue