/*
|
* CDDL HEADER START
|
*
|
* The contents of this file are subject to the terms of the
|
* Common Development and Distribution License, Version 1.0 only
|
* (the "License"). You may not use this file except in compliance
|
* with the License.
|
*
|
* You can obtain a copy of the license at legal-notices/CDDLv1_0.txt
|
* or http://forgerock.org/license/CDDLv1.0.html.
|
* See the License for the specific language governing permissions
|
* and limitations under the License.
|
*
|
* When distributing Covered Code, include this CDDL HEADER in each
|
* file and include the License file at legal-notices/CDDLv1_0.txt.
|
* If applicable, add the following below this CDDL HEADER, with the
|
* fields enclosed by brackets "[]" replaced with your own identifying
|
* information:
|
* Portions Copyright [yyyy] [name of copyright owner]
|
*
|
* CDDL HEADER END
|
*
|
*
|
* Copyright 2008-2009 Sun Microsystems, Inc.
|
* Portions Copyright 2014-2015 ForgeRock AS
|
*/
|
package org.opends.guitools.controlpanel.ui;
|
|
import static org.opends.messages.AdminToolMessages.*;
|
|
import java.awt.Component;
|
import java.awt.GridBagConstraints;
|
import java.awt.event.ItemEvent;
|
import java.awt.event.ItemListener;
|
import java.util.ArrayList;
|
import java.util.Collection;
|
import java.util.HashSet;
|
import java.util.LinkedHashSet;
|
import java.util.List;
|
import java.util.Set;
|
import java.util.SortedSet;
|
import java.util.TreeSet;
|
|
import javax.naming.ldap.InitialLdapContext;
|
import javax.swing.DefaultComboBoxModel;
|
import javax.swing.JCheckBox;
|
import javax.swing.SwingUtilities;
|
|
import org.forgerock.i18n.LocalizableMessage;
|
import org.opends.guitools.controlpanel.datamodel.BackendDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.CategorizedComboBoxElement;
|
import org.opends.guitools.controlpanel.datamodel.ControlPanelInfo;
|
import org.opends.guitools.controlpanel.datamodel.IndexDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.IndexTypeDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.ServerDescriptor;
|
import org.opends.guitools.controlpanel.event.ConfigurationChangeEvent;
|
import org.opends.guitools.controlpanel.task.Task;
|
import org.opends.guitools.controlpanel.util.ConfigReader;
|
import org.opends.guitools.controlpanel.util.Utilities;
|
import org.opends.server.admin.PropertyException;
|
import org.opends.server.admin.client.ManagementContext;
|
import org.opends.server.admin.client.ldap.JNDIDirContextAdaptor;
|
import org.opends.server.admin.client.ldap.LDAPManagementContext;
|
import org.opends.server.admin.std.client.BackendCfgClient;
|
import org.opends.server.admin.std.client.BackendIndexCfgClient;
|
import org.opends.server.admin.std.client.LocalDBBackendCfgClient;
|
import org.opends.server.admin.std.client.LocalDBIndexCfgClient;
|
import org.opends.server.admin.std.client.PluggableBackendCfgClient;
|
import org.opends.server.admin.std.meta.BackendIndexCfgDefn;
|
import org.opends.server.admin.std.meta.LocalDBIndexCfgDefn;
|
import org.opends.server.backends.jeb.RemoveOnceLocalDBBackendIsPluggable;
|
import org.opends.server.core.DirectoryServer;
|
import org.opends.server.types.AttributeType;
|
import org.opends.server.types.DN;
|
import org.opends.server.types.OpenDsException;
|
import org.opends.server.types.Schema;
|
|
/**
|
* Panel that appears when the user defines a new index.
|
*/
|
public class NewIndexPanel extends AbstractIndexPanel
|
{
|
private static final long serialVersionUID = -3516011638125862137L;
|
|
private final Component relativeComponent;
|
private Schema schema;
|
private IndexDescriptor newIndex;
|
|
/**
|
* Constructor of the panel.
|
*
|
* @param backendName
|
* the backend where the index will be created.
|
* @param relativeComponent
|
* the component relative to which the dialog containing this panel
|
* will be centered.
|
*/
|
public NewIndexPanel(final String backendName, final Component relativeComponent)
|
{
|
super();
|
this.backendName.setText(backendName);
|
this.relativeComponent = relativeComponent;
|
createLayout();
|
}
|
|
@Override
|
public LocalizableMessage getTitle()
|
{
|
return INFO_CTRL_PANEL_NEW_INDEX_TITLE.get();
|
}
|
|
@Override
|
public Component getPreferredFocusComponent()
|
{
|
return attributes;
|
}
|
|
/**
|
* Updates the contents of the panel with the provided backend.
|
*
|
* @param backend
|
* the backend where the index will be created.
|
*/
|
public void update(final BackendDescriptor backend)
|
{
|
backendName.setText(backend.getBackendID());
|
}
|
|
@Override
|
public void configurationChanged(final ConfigurationChangeEvent ev)
|
{
|
final ServerDescriptor desc = ev.getNewDescriptor();
|
|
Schema s = desc.getSchema();
|
final boolean[] repack = { false };
|
final boolean[] error = { false };
|
if (s != null)
|
{
|
schema = s;
|
repack[0] = attributes.getItemCount() == 0;
|
LinkedHashSet<CategorizedComboBoxElement> newElements = new LinkedHashSet<>();
|
|
BackendDescriptor backend = getBackendByID(backendName.getText());
|
|
TreeSet<String> standardAttrNames = new TreeSet<>();
|
TreeSet<String> configurationAttrNames = new TreeSet<>();
|
TreeSet<String> customAttrNames = new TreeSet<>();
|
for (AttributeType attr : schema.getAttributeTypes().values())
|
{
|
String name = attr.getPrimaryName();
|
if (!indexExists(backend, name))
|
{
|
if (Utilities.isStandard(attr))
|
{
|
standardAttrNames.add(name);
|
}
|
else if (Utilities.isConfiguration(attr))
|
{
|
configurationAttrNames.add(name);
|
}
|
else
|
{
|
customAttrNames.add(name);
|
}
|
}
|
}
|
if (!customAttrNames.isEmpty())
|
{
|
newElements.add(new CategorizedComboBoxElement(CUSTOM_ATTRIBUTES, CategorizedComboBoxElement.Type.CATEGORY));
|
for (String attrName : customAttrNames)
|
{
|
newElements.add(new CategorizedComboBoxElement(attrName, CategorizedComboBoxElement.Type.REGULAR));
|
}
|
}
|
if (!standardAttrNames.isEmpty())
|
{
|
newElements.add(new CategorizedComboBoxElement(STANDARD_ATTRIBUTES, CategorizedComboBoxElement.Type.CATEGORY));
|
for (String attrName : standardAttrNames)
|
{
|
newElements.add(new CategorizedComboBoxElement(attrName, CategorizedComboBoxElement.Type.REGULAR));
|
}
|
}
|
DefaultComboBoxModel model = (DefaultComboBoxModel) attributes.getModel();
|
updateComboBoxModel(newElements, model);
|
}
|
else
|
{
|
updateErrorPane(errorPane, ERR_CTRL_PANEL_SCHEMA_NOT_FOUND_SUMMARY.get(), ColorAndFontConstants.errorTitleFont,
|
ERR_CTRL_PANEL_SCHEMA_NOT_FOUND_DETAILS.get(), ColorAndFontConstants.defaultFont);
|
repack[0] = true;
|
error[0] = true;
|
}
|
|
SwingUtilities.invokeLater(new Runnable()
|
{
|
@Override
|
public void run()
|
{
|
setEnabledOK(!error[0]);
|
errorPane.setVisible(error[0]);
|
if (repack[0])
|
{
|
packParentDialog();
|
if (relativeComponent != null)
|
{
|
Utilities.centerGoldenMean(Utilities.getParentDialog(NewIndexPanel.this), relativeComponent);
|
}
|
}
|
}
|
});
|
if (!error[0])
|
{
|
updateErrorPaneAndOKButtonIfAuthRequired(desc, isLocal()
|
? INFO_CTRL_PANEL_AUTHENTICATION_REQUIRED_FOR_NEW_INDEX.get()
|
: INFO_CTRL_PANEL_CANNOT_CONNECT_TO_REMOTE_DETAILS.get(desc.getHostname()));
|
}
|
}
|
|
private boolean indexExists(BackendDescriptor backend, String indexName)
|
{
|
if (backend != null)
|
{
|
for (IndexDescriptor index : backend.getIndexes())
|
{
|
if (index.getName().equalsIgnoreCase(indexName))
|
{
|
return true;
|
}
|
}
|
}
|
return false;
|
}
|
|
private BackendDescriptor getBackendByID(String backendID)
|
{
|
for (BackendDescriptor b : getInfo().getServerDescriptor().getBackends())
|
{
|
if (b.getBackendID().equalsIgnoreCase(backendID))
|
{
|
return b;
|
}
|
}
|
return null;
|
}
|
|
@Override
|
public void okClicked()
|
{
|
setPrimaryValid(lAttribute);
|
setPrimaryValid(lEntryLimit);
|
setPrimaryValid(lType);
|
List<LocalizableMessage> errors = new ArrayList<>();
|
String attrName = getAttributeName();
|
if (attrName == null)
|
{
|
errors.add(ERR_INFO_CTRL_ATTRIBUTE_NAME_REQUIRED.get());
|
setPrimaryInvalid(lAttribute);
|
}
|
|
String v = entryLimit.getText();
|
try
|
{
|
int n = Integer.parseInt(v);
|
if (n < MIN_ENTRY_LIMIT || MAX_ENTRY_LIMIT < n)
|
{
|
errors.add(ERR_INFO_CTRL_PANEL_ENTRY_LIMIT_NOT_VALID.get(MIN_ENTRY_LIMIT, MAX_ENTRY_LIMIT));
|
setPrimaryInvalid(lEntryLimit);
|
}
|
}
|
catch (Throwable t)
|
{
|
errors.add(ERR_INFO_CTRL_PANEL_ENTRY_LIMIT_NOT_VALID.get(MIN_ENTRY_LIMIT, MAX_ENTRY_LIMIT));
|
setPrimaryInvalid(lEntryLimit);
|
}
|
|
if (!isSomethingSelected())
|
{
|
errors.add(ERR_INFO_ONE_INDEX_TYPE_MUST_BE_SELECTED.get());
|
setPrimaryInvalid(lType);
|
}
|
ProgressDialog dlg = new ProgressDialog(
|
Utilities.createFrame(), Utilities.getParentDialog(this), INFO_CTRL_PANEL_NEW_INDEX_TITLE.get(), getInfo());
|
NewIndexTask newTask = new NewIndexTask(getInfo(), dlg);
|
for (Task task : getInfo().getTasks())
|
{
|
task.canLaunch(newTask, errors);
|
}
|
if (errors.isEmpty())
|
{
|
launchOperation(newTask, INFO_CTRL_PANEL_CREATING_NEW_INDEX_SUMMARY.get(attrName),
|
INFO_CTRL_PANEL_CREATING_NEW_INDEX_SUCCESSFUL_SUMMARY.get(),
|
INFO_CTRL_PANEL_CREATING_NEW_INDEX_SUCCESSFUL_DETAILS.get(attrName),
|
ERR_CTRL_PANEL_CREATING_NEW_INDEX_ERROR_SUMMARY.get(),
|
ERR_CTRL_PANEL_CREATING_NEW_INDEX_ERROR_DETAILS.get(),
|
null, dlg);
|
dlg.setVisible(true);
|
Utilities.getParentDialog(this).setVisible(false);
|
}
|
else
|
{
|
displayErrorDialog(errors);
|
}
|
}
|
|
private boolean isSomethingSelected()
|
{
|
for (JCheckBox type : types)
|
{
|
boolean somethingSelected = type.isSelected() && type.isVisible();
|
if (somethingSelected)
|
{
|
return true;
|
}
|
}
|
return false;
|
}
|
|
private String getAttributeName()
|
{
|
CategorizedComboBoxElement o = (CategorizedComboBoxElement) attributes.getSelectedItem();
|
return o != null ? o.getValue().toString() : null;
|
}
|
|
/** Creates the layout of the panel (but the contents are not populated here). */
|
private void createLayout()
|
{
|
GridBagConstraints gbc = new GridBagConstraints();
|
createBasicLayout(this, gbc, false);
|
|
attributes.addItemListener(new ItemListener()
|
{
|
@Override
|
public void itemStateChanged(final ItemEvent ev)
|
{
|
String n = getAttributeName();
|
AttributeType attr = null;
|
if (n != null)
|
{
|
attr = schema.getAttributeType(n.toLowerCase());
|
}
|
repopulateTypesPanel(attr);
|
}
|
});
|
entryLimit.setText(String.valueOf(DEFAULT_ENTRY_LIMIT));
|
}
|
|
/** The task in charge of creating the index. */
|
private class NewIndexTask extends Task
|
{
|
private final Set<String> backendSet = new HashSet<>();
|
private final String attributeName;
|
private final int entryLimitValue;
|
private final SortedSet<IndexTypeDescriptor> indexTypes;
|
|
/**
|
* The constructor of the task.
|
*
|
* @param info
|
* the control panel info.
|
* @param dlg
|
* the progress dialog that shows the progress of the task.
|
*/
|
public NewIndexTask(final ControlPanelInfo info, final ProgressDialog dlg)
|
{
|
super(info, dlg);
|
backendSet.add(backendName.getText());
|
attributeName = getAttributeName();
|
entryLimitValue = Integer.parseInt(entryLimit.getText());
|
indexTypes = getTypes();
|
}
|
|
@Override
|
public Type getType()
|
{
|
return Type.NEW_INDEX;
|
}
|
|
@Override
|
public Set<String> getBackends()
|
{
|
return backendSet;
|
}
|
|
@Override
|
public LocalizableMessage getTaskDescription()
|
{
|
return INFO_CTRL_PANEL_NEW_INDEX_TASK_DESCRIPTION.get(attributeName, backendName.getText());
|
}
|
|
@Override
|
public boolean canLaunch(final Task taskToBeLaunched, final Collection<LocalizableMessage> incompatibilityReasons)
|
{
|
boolean canLaunch = true;
|
if (state == State.RUNNING && runningOnSameServer(taskToBeLaunched))
|
{
|
// All the operations are incompatible if they apply to this
|
// backend for safety. This is a short operation so the limitation
|
// has not a lot of impact.
|
Set<String> backends = new TreeSet<>(taskToBeLaunched.getBackends());
|
backends.retainAll(getBackends());
|
if (!backends.isEmpty())
|
{
|
incompatibilityReasons.add(getIncompatibilityMessage(this, taskToBeLaunched));
|
canLaunch = false;
|
}
|
}
|
return canLaunch;
|
}
|
|
private void updateConfiguration() throws OpenDsException
|
{
|
boolean configHandlerUpdated = false;
|
try
|
{
|
if (!isServerRunning())
|
{
|
configHandlerUpdated = true;
|
getInfo().stopPooling();
|
if (getInfo().mustDeregisterConfig())
|
{
|
DirectoryServer.deregisterBaseDN(DN.valueOf("cn=config"));
|
}
|
DirectoryServer.getInstance().initializeConfiguration(
|
org.opends.server.extensions.ConfigFileHandler.class.getName(), ConfigReader.configFile);
|
getInfo().setMustDeregisterConfig(true);
|
}
|
else
|
{
|
SwingUtilities.invokeLater(new Runnable()
|
{
|
@Override
|
public void run()
|
{
|
List<String> args = getObfuscatedCommandLineArguments(getDSConfigCommandLineArguments());
|
args.removeAll(getConfigCommandLineArguments());
|
printEquivalentCommandLine(
|
getConfigCommandLineName(), args, INFO_CTRL_PANEL_EQUIVALENT_CMD_TO_CREATE_INDEX.get());
|
}
|
});
|
}
|
SwingUtilities.invokeLater(new Runnable()
|
{
|
@Override
|
public void run()
|
{
|
getProgressDialog().appendProgressHtml(Utilities.getProgressWithPoints(
|
INFO_CTRL_PANEL_CREATING_NEW_INDEX_PROGRESS.get(attributeName), ColorAndFontConstants.progressFont));
|
}
|
});
|
|
if (isServerRunning())
|
{
|
createIndexOnline(getInfo().getDirContext());
|
}
|
else
|
{
|
createIndexOffline(backendName.getText(), attributeName, indexTypes, entryLimitValue);
|
}
|
SwingUtilities.invokeLater(new Runnable()
|
{
|
@Override
|
public void run()
|
{
|
getProgressDialog().appendProgressHtml(Utilities.getProgressDone(ColorAndFontConstants.progressFont));
|
}
|
});
|
}
|
finally
|
{
|
if (configHandlerUpdated)
|
{
|
DirectoryServer.getInstance().initializeConfiguration(ConfigReader.configClassName, ConfigReader.configFile);
|
getInfo().startPooling();
|
}
|
}
|
}
|
|
private void createIndexOnline(final InitialLdapContext ctx) throws OpenDsException
|
{
|
final ManagementContext mCtx = LDAPManagementContext.createFromContext(JNDIDirContextAdaptor.adapt(ctx));
|
final BackendCfgClient backend = mCtx.getRootConfiguration().getBackend(backendName.getText());
|
if (backend instanceof LocalDBBackendCfgClient)
|
{
|
createLocalDBIndexOnline((LocalDBBackendCfgClient) backend);
|
return;
|
}
|
createBackendIndexOnline((PluggableBackendCfgClient) backend);
|
}
|
|
private void createBackendIndexOnline(final PluggableBackendCfgClient backend) throws OpenDsException
|
{
|
final List<PropertyException> exceptions = new ArrayList<>();
|
final BackendIndexCfgClient index = backend.createBackendIndex(
|
BackendIndexCfgDefn.getInstance(), attributeName, exceptions);
|
index.setIndexType(IndexTypeDescriptor.toBackendIndexTypes(indexTypes));
|
if (entryLimitValue != index.getIndexEntryLimit())
|
{
|
index.setIndexEntryLimit(entryLimitValue);
|
}
|
index.commit();
|
Utilities.throwFirstFrom(exceptions);
|
}
|
|
@RemoveOnceLocalDBBackendIsPluggable
|
private void createLocalDBIndexOnline(final LocalDBBackendCfgClient backend) throws OpenDsException
|
{
|
final List<PropertyException> exceptions = new ArrayList<>();
|
final LocalDBIndexCfgClient index = backend.createLocalDBIndex(
|
LocalDBIndexCfgDefn.getInstance(), attributeName, exceptions);
|
index.setIndexType(IndexTypeDescriptor.toLocalDBIndexTypes(indexTypes));
|
if (entryLimitValue != index.getIndexEntryLimit())
|
{
|
index.setIndexEntryLimit(entryLimitValue);
|
}
|
index.commit();
|
Utilities.throwFirstFrom(exceptions);
|
}
|
|
@Override
|
protected String getCommandLinePath()
|
{
|
return null;
|
}
|
|
@Override
|
protected List<String> getCommandLineArguments()
|
{
|
return new ArrayList<>();
|
}
|
|
private String getConfigCommandLineName()
|
{
|
if (isServerRunning())
|
{
|
return getCommandLinePath("dsconfig");
|
}
|
return null;
|
}
|
|
@Override
|
public void runTask()
|
{
|
state = State.RUNNING;
|
lastException = null;
|
|
try
|
{
|
updateConfiguration();
|
for (BackendDescriptor backend : getInfo().getServerDescriptor().getBackends())
|
{
|
if (backend.getBackendID().equalsIgnoreCase(backendName.getText()))
|
{
|
newIndex = new IndexDescriptor(attributeName,
|
schema.getAttributeType(attributeName.toLowerCase()), backend, indexTypes, entryLimitValue);
|
getInfo().registerModifiedIndex(newIndex);
|
notifyConfigurationElementCreated(newIndex);
|
break;
|
}
|
}
|
state = State.FINISHED_SUCCESSFULLY;
|
}
|
catch (Throwable t)
|
{
|
lastException = t;
|
state = State.FINISHED_WITH_ERROR;
|
}
|
}
|
|
@Override
|
public void postOperation()
|
{
|
if (lastException == null && state == State.FINISHED_SUCCESSFULLY && newIndex != null)
|
{
|
rebuildIndexIfNecessary(newIndex, getProgressDialog());
|
}
|
}
|
|
private ArrayList<String> getDSConfigCommandLineArguments()
|
{
|
ArrayList<String> args = new ArrayList<>();
|
args.add("create-local-db-index");
|
args.add("--backend-name");
|
args.add(backendName.getText());
|
args.add("--type");
|
args.add("generic");
|
|
args.add("--index-name");
|
args.add(attributeName);
|
|
for (IndexTypeDescriptor type : indexTypes)
|
{
|
args.add("--set");
|
args.add("index-type:" + type.toLocalDBIndexType());
|
}
|
args.add("--set");
|
args.add("index-entry-limit:" + entryLimitValue);
|
args.addAll(getConnectionCommandLineArguments());
|
args.add(getNoPropertiesFileArgument());
|
args.add("--no-prompt");
|
return args;
|
}
|
}
|
}
|