/*
|
* 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
|
* trunk/opends/resource/legal-notices/OpenDS.LICENSE
|
* or https://OpenDS.dev.java.net/OpenDS.LICENSE.
|
* 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
|
* trunk/opends/resource/legal-notices/OpenDS.LICENSE. 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-2010 Sun Microsystems, Inc.
|
*/
|
|
|
package org.opends.guitools.controlpanel.ui;
|
|
import static org.opends.messages.AdminToolMessages.*;
|
|
import java.awt.Component;
|
import java.awt.Container;
|
import java.awt.Dimension;
|
import java.awt.GridBagConstraints;
|
import java.awt.GridBagLayout;
|
import java.awt.event.ActionEvent;
|
import java.awt.event.ActionListener;
|
import java.awt.event.ItemEvent;
|
import java.awt.event.ItemListener;
|
import java.util.ArrayList;
|
import java.util.Collection;
|
import java.util.LinkedHashSet;
|
import java.util.List;
|
import java.util.TreeSet;
|
|
import javax.swing.Box;
|
import javax.swing.ButtonGroup;
|
import javax.swing.DefaultComboBoxModel;
|
import javax.swing.DefaultListModel;
|
import javax.swing.JButton;
|
import javax.swing.JComboBox;
|
import javax.swing.JLabel;
|
import javax.swing.JList;
|
import javax.swing.JPanel;
|
import javax.swing.JRadioButton;
|
import javax.swing.JTextField;
|
import javax.swing.ListModel;
|
import javax.swing.SwingUtilities;
|
import javax.swing.event.ListSelectionEvent;
|
import javax.swing.event.ListSelectionListener;
|
|
import org.opends.guitools.controlpanel.datamodel.BackendDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.BaseDNDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.CategorizedComboBoxElement;
|
import org.opends.guitools.controlpanel.datamodel.IndexDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.ServerDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.VLVIndexDescriptor;
|
import org.opends.guitools.controlpanel.datamodel.VLVSortOrder;
|
import org.opends.guitools.controlpanel.ui.components.TitlePanel;
|
import org.opends.guitools.controlpanel.ui.renderer.CustomListCellRenderer;
|
import org.opends.guitools.controlpanel.ui.renderer.IndexComboBoxCellRenderer;
|
import org.opends.guitools.controlpanel.ui.renderer.VLVSortOrderRenderer;
|
import org.opends.guitools.controlpanel.util.Utilities;
|
import org.opends.messages.Message;
|
import org.opends.server.admin.DefinedDefaultBehaviorProvider;
|
import org.opends.server.admin.std.meta.LocalDBVLVIndexCfgDefn;
|
import org.opends.server.admin.std.meta.LocalDBIndexCfgDefn.IndexType;
|
import org.opends.server.admin.std.meta.LocalDBVLVIndexCfgDefn.Scope;
|
import org.opends.server.protocols.ldap.LDAPFilter;
|
import org.opends.server.types.AttributeType;
|
import org.opends.server.types.DN;
|
import org.opends.server.types.FilterType;
|
import org.opends.server.types.LDAPException;
|
import org.opends.server.types.OpenDsException;
|
import org.opends.server.types.RawFilter;
|
import org.opends.server.types.Schema;
|
import org.opends.server.util.ServerConstants;
|
|
/**
|
* Abstract class used to re-factor some code between the classes that are used
|
* to edit/create a VLV index.
|
*
|
*/
|
public abstract class AbstractVLVIndexPanel extends StatusGenericPanel
|
{
|
private static final long serialVersionUID = -82857384664911898L;
|
/**
|
* Title panel.
|
*/
|
protected TitlePanel titlePanel = new TitlePanel(Message.EMPTY,
|
Message.EMPTY);
|
/**
|
* Name label.
|
*/
|
protected JLabel lName = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_VLV_INDEX_NAME_LABEL.get());
|
/**
|
* Base DN label.
|
*/
|
protected JLabel lBaseDN = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_VLV_INDEX_BASE_DN_LABEL.get());
|
/**
|
* Search scope label.
|
*/
|
protected JLabel lSearchScope = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_VLV_INDEX_SEARCH_SCOPE_LABEL.get());
|
/**
|
* Search filter label.
|
*/
|
protected JLabel lFilter = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_VLV_INDEX_FILTER_LABEL.get());
|
/**
|
* Sort order label.
|
*/
|
protected JLabel lSortOrder = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_VLV_INDEX_SORT_ORDER_LABEL.get());
|
/**
|
* Backends label.
|
*/
|
protected JLabel lBackend = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_BACKEND_LABEL.get());
|
/**
|
* Max block size label.
|
*/
|
protected JLabel lMaxBlockSize = Utilities.createPrimaryLabel(
|
INFO_CTRL_PANEL_VLV_INDEX_MAX_BLOCK_SIZE_LABEL.get());
|
|
/**
|
* Name text field.
|
*/
|
protected JTextField name = Utilities.createMediumTextField();
|
/**
|
* Read-only name label.
|
*/
|
protected JLabel readOnlyName = Utilities.createDefaultLabel();
|
/**
|
* Read-only backend name label.
|
*/
|
protected JLabel backendName = Utilities.createDefaultLabel();
|
/**
|
* Base DNs combo box.
|
*/
|
protected JComboBox baseDNs = Utilities.createComboBox();
|
/**
|
* Subtree text field.
|
*/
|
protected JTextField baseDN = Utilities.createLongTextField();
|
/**
|
* Base Object scope radio button.
|
*/
|
protected JRadioButton baseObject = Utilities.createRadioButton(
|
INFO_CTRL_PANEL_VLV_INDEX_BASE_OBJECT_LABEL.get());
|
/**
|
* Single Level scope radio button.
|
*/
|
protected JRadioButton singleLevel = Utilities.createRadioButton(
|
INFO_CTRL_PANEL_VLV_INDEX_SINGLE_LEVEL_LABEL.get());
|
/**
|
* Subordinate subtree scope radio button.
|
*/
|
protected JRadioButton subordinateSubtree = Utilities.createRadioButton(
|
INFO_CTRL_PANEL_VLV_INDEX_SUBORDINATE_SUBTREE_LABEL.get());
|
/**
|
* Whole subtree scope radio button.
|
*/
|
protected JRadioButton wholeSubtree = Utilities.createRadioButton(
|
INFO_CTRL_PANEL_VLV_INDEX_WHOLE_SUBTREE_LABEL.get());
|
|
/**
|
* Filter text field.
|
*/
|
protected JTextField filter = Utilities.createLongTextField();
|
/**
|
* Max block size text field.
|
*/
|
protected JTextField maxBlockSize = Utilities.createShortTextField();
|
/**
|
* Attributes combo box.
|
*/
|
protected JComboBox attributes = Utilities.createComboBox();
|
/**
|
* The list containing the sort order elements.
|
*/
|
protected JList sortOrder = new JList();
|
/**
|
* The add button.
|
*/
|
protected JButton add = Utilities.createButton(
|
INFO_CTRL_PANEL_VLV_INDEX_ADD_BUTTON_LABEL.get());
|
/**
|
* The move up button.
|
*/
|
protected JButton moveUp = Utilities.createButton(
|
INFO_CTRL_PANEL_VLV_INDEX_MOVE_UP_BUTTON_LABEL.get());
|
/**
|
* The move down button.
|
*/
|
protected JButton moveDown = Utilities.createButton(
|
INFO_CTRL_PANEL_VLV_INDEX_MOVE_DOWN_BUTTON_LABEL.get());
|
/**
|
* The remove button.
|
*/
|
protected JButton remove = Utilities.createButton (
|
INFO_CTRL_PANEL_VLV_INDEX_REMOVE_BUTTON_LABEL.get());
|
/**
|
* Ascending order combo box.
|
*/
|
protected JComboBox ascendingOrder = Utilities.createComboBox();
|
|
/**
|
* Combo box containing the sort order.
|
*/
|
protected DefaultListModel sortOrderModel;
|
|
/**
|
* The list of labels.
|
*/
|
protected JLabel[] labels = {lName, lBaseDN, lSearchScope, lFilter,
|
lSortOrder, lBackend, lMaxBlockSize};
|
|
/**
|
* The relative component that must be used to center the parent dialog of
|
* this panel.
|
*/
|
protected Component relativeComponent;
|
|
/**
|
* Other base DN message.
|
*/
|
protected final Message OTHER_BASE_DN =
|
INFO_CTRL_PANEL_VLV_OTHER_BASE_DN_LABEL.get();
|
/**
|
* Ascending message.
|
*/
|
protected final Message ASCENDING =
|
INFO_CTRL_PANEL_VLV_ASCENDING_LABEL.get();
|
/**
|
* Descending message.
|
*/
|
protected final Message DESCENDING =
|
INFO_CTRL_PANEL_VLV_DESCENDING_LABEL.get();
|
|
/**
|
* Custom attributes message.
|
*/
|
protected Message CUSTOM_ATTRIBUTES =
|
INFO_CTRL_PANEL_CUSTOM_ATTRIBUTES_LABEL.get();
|
|
/**
|
* Standard attributes message.
|
*/
|
protected Message STANDARD_ATTRIBUTES =
|
INFO_CTRL_PANEL_STANDARD_ATTRIBUTES_LABEL.get();
|
|
/**
|
* The list of standard attribute names.
|
*/
|
protected TreeSet<String> standardAttrNames = new TreeSet<String>();
|
/**
|
* The list of configuration attribute names.
|
*/
|
protected TreeSet<String> configurationAttrNames = new TreeSet<String>();
|
/**
|
* The list of custom attribute names.
|
*/
|
protected TreeSet<String> customAttrNames = new TreeSet<String>();
|
|
private int defaultVLVEntryLimitValue;
|
{
|
DefinedDefaultBehaviorProvider<Integer> provider =
|
(DefinedDefaultBehaviorProvider<Integer>)
|
LocalDBVLVIndexCfgDefn.getInstance().getMaxBlockSizePropertyDefinition().
|
getDefaultBehaviorProvider();
|
defaultVLVEntryLimitValue =
|
Integer.parseInt(provider.getDefaultValues().iterator().next());
|
}
|
|
/**
|
* Minimum value for max block size.
|
*/
|
protected final int MIN_MAX_BLOCK_SIZE =
|
LocalDBVLVIndexCfgDefn.getInstance().getMaxBlockSizePropertyDefinition().
|
getLowerLimit();
|
/**
|
* Maximum value for max block size.
|
*/
|
protected final int MAX_MAX_BLOCK_SIZE = 2147483647;
|
/**
|
* Default value for max block size.
|
*/
|
protected final int DEFAULT_MAX_BLOCK_SIZE = defaultVLVEntryLimitValue;
|
|
|
/**
|
* Constructor.
|
* @param backendID the backend ID where the index is defined (or will be
|
* defined).
|
* @param relativeComponent the relative component where the dialog containing
|
* this panel must be centered.
|
*/
|
protected AbstractVLVIndexPanel(String backendID, Component relativeComponent)
|
{
|
super();
|
if (backendID != null)
|
{
|
backendName.setText(backendID);
|
}
|
this.relativeComponent = relativeComponent;
|
}
|
|
/**
|
* Sets the name of the backend where the index is defined or will be defined.
|
* @param backendID the ID of the backend.
|
*/
|
public void setBackendName(String backendID)
|
{
|
backendName.setText(backendID);
|
}
|
|
/**
|
* Returns the LDIF representing the new index.
|
* @param indexName the name of the index.
|
* @return the LDIF representing the new index.
|
*/
|
protected String getIndexLDIF(String indexName)
|
{
|
String dn = Utilities.getRDNString("ds-cfg-backend-id",
|
backendName.getText())+",cn=Backends,cn=config";
|
ArrayList<String> lines = new ArrayList<String>();
|
lines.add("dn: "+Utilities.getRDNString("ds-cfg-name", indexName)+
|
",cn=VLV Index,"+dn);
|
lines.add("objectClass: ds-cfg-local-db-vlv-index");
|
lines.add("objectClass: top");
|
lines.add("ds-cfg-name: "+indexName);
|
lines.add("ds-cfg-filter: "+filter.getText().trim());
|
lines.add("ds-cfg-sort-order: "+getSortOrderStringValue(getSortOrder()));
|
lines.add("ds-cfg-base-dn: "+getBaseDN());
|
lines.add("ds-cfg-scope: "+getScope().toString());
|
lines.add("ds-cfg-max-block-size: "+maxBlockSize.getText().trim());
|
StringBuilder sb = new StringBuilder();
|
for (String line : lines)
|
{
|
sb.append(line+ServerConstants.EOL);
|
}
|
return sb.toString();
|
}
|
|
/**
|
* Returns the scope of the VLV index as it appears on the panel.
|
* @return the scope of the VLV index as it appears on the panel.
|
*/
|
protected Scope getScope()
|
{
|
Scope scope;
|
if (baseObject.isSelected())
|
{
|
scope = Scope.BASE_OBJECT;
|
}
|
else if (singleLevel.isSelected())
|
{
|
scope = Scope.SINGLE_LEVEL;
|
}
|
else if (subordinateSubtree.isSelected())
|
{
|
scope = Scope.SUBORDINATE_SUBTREE;
|
}
|
else
|
{
|
scope = Scope.WHOLE_SUBTREE;
|
}
|
return scope;
|
}
|
|
/**
|
* Returns the list of VLV sort order elements as they are displayed in the
|
* panel.
|
* @return the list of VLV sort order elements as they are displayed in the
|
* panel.
|
*/
|
protected List<VLVSortOrder> getSortOrder()
|
{
|
ArrayList<VLVSortOrder> sortOrder = new ArrayList<VLVSortOrder>();
|
for (int i=0; i<sortOrderModel.getSize(); i++)
|
{
|
sortOrder.add((VLVSortOrder)sortOrderModel.get(i));
|
}
|
return sortOrder;
|
}
|
|
/**
|
* Returns the string representation for the provided list of VLV sort order.
|
* @param sortOrder the list of VLV sort order elements.
|
* @return the string representation for the provided list of VLV sort order.
|
*/
|
protected String getSortOrderStringValue(List<VLVSortOrder> sortOrder)
|
{
|
StringBuilder sb = new StringBuilder();
|
for (VLVSortOrder s : sortOrder)
|
{
|
if (sb.length() > 0)
|
{
|
sb.append(" ");
|
}
|
if (s.isAscending())
|
{
|
sb.append("+");
|
}
|
else
|
{
|
sb.append("-");
|
}
|
sb.append(s.getAttributeName());
|
}
|
return sb.toString();
|
}
|
|
|
/**
|
* Updates the layout with the provided server descriptor.
|
* @param desc the server descriptor.
|
* @return <CODE>true</CODE> if an error has been displayed and
|
* <CODE>false</CODE> otherwise.
|
*/
|
protected boolean updateLayout(final ServerDescriptor desc)
|
{
|
Schema schema = desc.getSchema();
|
BackendDescriptor backend = getBackend();
|
final boolean[] repack = {false};
|
final boolean[] error = {false};
|
if (backend != null)
|
{
|
updateBaseDNCombo(backend);
|
}
|
|
if (schema != null)
|
{
|
repack[0] = attributes.getItemCount() == 0;
|
LinkedHashSet<CategorizedComboBoxElement> newElements =
|
new LinkedHashSet<CategorizedComboBoxElement>();
|
|
synchronized(standardAttrNames)
|
{
|
standardAttrNames.clear();
|
configurationAttrNames.clear();
|
customAttrNames.clear();
|
|
for (AttributeType attr : schema.getAttributeTypes().values())
|
{
|
String name = attr.getPrimaryName();
|
boolean defined = false;
|
ListModel model = sortOrder.getModel();
|
for (int i=0; i < model.getSize(); i++)
|
{
|
VLVSortOrder s = (VLVSortOrder)model.getElementAt(i);
|
if (name.equalsIgnoreCase(s.getAttributeName()))
|
{
|
defined = true;
|
break;
|
}
|
}
|
if (!defined)
|
{
|
if (Utilities.isStandard(attr))
|
{
|
standardAttrNames.add(name);
|
}
|
else if (Utilities.isConfiguration(attr))
|
{
|
configurationAttrNames.add(name);
|
}
|
else
|
{
|
customAttrNames.add(name);
|
}
|
}
|
}
|
}
|
if (customAttrNames.size() > 0)
|
{
|
newElements.add(new CategorizedComboBoxElement(
|
CUSTOM_ATTRIBUTES,
|
CategorizedComboBoxElement.Type.CATEGORY));
|
for (String attrName : customAttrNames)
|
{
|
newElements.add(new CategorizedComboBoxElement(
|
attrName,
|
CategorizedComboBoxElement.Type.REGULAR));
|
}
|
}
|
if (standardAttrNames.size() > 0)
|
{
|
newElements.add(new CategorizedComboBoxElement(
|
STANDARD_ATTRIBUTES,
|
CategorizedComboBoxElement.Type.CATEGORY));
|
for (String attrName : standardAttrNames)
|
{
|
newElements.add(new CategorizedComboBoxElement(
|
attrName,
|
CategorizedComboBoxElement.Type.REGULAR));
|
}
|
}
|
// Ignore configuration attr names
|
/*
|
if (configurationAttrNames.size() > 0)
|
{
|
newElements.add(new CategorizedComboBoxDescriptor(
|
"Configuration Attributes",
|
CategorizedComboBoxDescriptor.Type.CATEGORY));
|
for (String attrName : configurationAttrNames)
|
{
|
newElements.add(new CategorizedComboBoxDescriptor(
|
attrName,
|
CategorizedComboBoxDescriptor.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()
|
{
|
public void run()
|
{
|
if ((getButtonType() == GenericDialog.ButtonType.OK) ||
|
(getButtonType() == GenericDialog.ButtonType.OK))
|
{
|
setEnabledOK(!error[0]);
|
}
|
errorPane.setVisible(error[0]);
|
add.setEnabled(attributes.getModel().getSize() > 0);
|
remove.setEnabled(sortOrder.getSelectedIndex() != -1);
|
if (repack[0])
|
{
|
packParentDialog();
|
if (relativeComponent != null)
|
{
|
Utilities.centerGoldenMean(
|
Utilities.getParentDialog(AbstractVLVIndexPanel.this),
|
relativeComponent);
|
}
|
}
|
}
|
});
|
return !error[0];
|
}
|
|
/**
|
* Returns <CODE>true</CODE> if the user accepts to continue creating the VLV
|
* index even if no indexes are created for the provided filter for the VLV
|
* index. Returns <CODE>false</CODE> if the user does not accept to create
|
* the index. Note that the confirmation dialog will only be displayed when
|
* the indexes are not defined, if the dialog is not displayed the method
|
* returns <CODE>true</CODE>.
|
* @return <CODE>true</CODE> if the user accepts to continue creating the VLV
|
* index even if no indexes are created for the provided filter for the VLV
|
* index. Returns <CODE>false</CODE> if the user does not accept to create
|
* the index.
|
*/
|
protected boolean checkIndexRequired()
|
{
|
boolean confirm = true;
|
String f = filter.getText().trim();
|
try
|
{
|
LDAPFilter ldapFilter = LDAPFilter.decode(f);
|
ArrayList<Message> msgs = new ArrayList<Message>();
|
updateIndexRequiredMessages(ldapFilter, msgs);
|
if (!msgs.isEmpty())
|
{
|
StringBuilder sb = new StringBuilder();
|
for (Message msg : msgs)
|
{
|
sb.append("<br>-"+msg);
|
}
|
confirm = displayConfirmationDialog(
|
INFO_CTRL_PANEL_VLV_INDEXES_NOT_DEFINED_CONFIRMATION_TITLE.get(),
|
INFO_CTRL_PANEL_VLV_INDEXES_NOT_DEFINED_CONFIRMATION_MSG.get(
|
getBackend().getBackendID(), sb.toString()));
|
}
|
}
|
catch (Throwable t)
|
{
|
// Bug
|
throw new RuntimeException("Unexpected error: "+t, t);
|
}
|
return confirm;
|
}
|
|
/**
|
* Updates the provided list of error messages by analyzing the provided
|
* filter. The idea is basically to analyze the filter and check if what
|
* appears on the filter is indexed or not. If it is not indexed it updates
|
* the error message list with a message explaining that.
|
* @param filter the filter to analyze.
|
* @param msgs the list of messages to be updated.
|
*/
|
private void updateIndexRequiredMessages(RawFilter filter,
|
Collection<Message> msgs)
|
{
|
switch (filter.getFilterType())
|
{
|
case AND:
|
case OR:
|
if (filter.getFilterComponents() != null)
|
{
|
for (RawFilter f : filter.getFilterComponents())
|
{
|
updateIndexRequiredMessages(f, msgs);
|
}
|
}
|
break;
|
case NOT:
|
updateIndexRequiredMessages(filter.getNOTComponent(), msgs);
|
break;
|
default:
|
FilterType[] filterTypes = {FilterType.EQUALITY, FilterType.SUBSTRING,
|
FilterType.GREATER_OR_EQUAL, FilterType.LESS_OR_EQUAL,
|
FilterType.PRESENT, FilterType.APPROXIMATE_MATCH,
|
FilterType.EXTENSIBLE_MATCH
|
};
|
IndexType[] indexTypes = {IndexType.EQUALITY, IndexType.SUBSTRING,
|
IndexType.ORDERING, IndexType.ORDERING, IndexType.PRESENCE,
|
IndexType.APPROXIMATE, null
|
};
|
Message[] indexTypeNames = {INFO_CTRL_PANEL_VLV_INDEX_EQUALITY_TYPE.get(),
|
INFO_CTRL_PANEL_VLV_INDEX_SUBSTRING_TYPE.get(),
|
INFO_CTRL_PANEL_VLV_INDEX_ORDERING_TYPE.get(),
|
INFO_CTRL_PANEL_VLV_INDEX_ORDERING_TYPE.get(),
|
INFO_CTRL_PANEL_VLV_INDEX_PRESENCE_TYPE.get(),
|
INFO_CTRL_PANEL_VLV_INDEX_APPROXIMATE_TYPE.get(),
|
null
|
};
|
for (int i=0; i<filterTypes.length; i++)
|
{
|
if (filterTypes[i] == filter.getFilterType())
|
{
|
IndexDescriptor index = getIndex(filter.getAttributeType());
|
if (index != null)
|
{
|
IndexType type = indexTypes[i];
|
if (type != null)
|
{
|
if (!index.getTypes().contains(type))
|
{
|
msgs.add(INFO_CTRL_PANEL_MUST_UPDATE_INDEX_DEFINITION_TYPE.get(
|
filter.getAttributeType(), indexTypeNames[i]));
|
}
|
}
|
}
|
else
|
{
|
Message type = indexTypeNames[i];
|
if (type != null)
|
{
|
msgs.add(INFO_CTRL_PANEL_MUST_DEFINE_INDEX_TYPE.get(
|
filter.getAttributeType(), type));
|
}
|
else
|
{
|
msgs.add(INFO_CTRL_PANEL_MUST_DEFINE_INDEX.get(
|
filter.getAttributeType()));
|
}
|
}
|
}
|
}
|
}
|
}
|
|
/**
|
* Returns the index descriptor for a given index name (<CODE>null</CODE> if
|
* no index descriptor is found for that name).
|
* @param indexName the name of the index.
|
* @return the index descriptor for a given index name.
|
*/
|
private IndexDescriptor getIndex(String indexName)
|
{
|
IndexDescriptor index = null;
|
BackendDescriptor backend = getBackend();
|
if (backend != null)
|
{
|
for (IndexDescriptor i : backend.getIndexes())
|
{
|
if (i.getName().equalsIgnoreCase(indexName))
|
{
|
index = i;
|
break;
|
}
|
}
|
}
|
return index;
|
}
|
|
/**
|
* Updates the base DN combo box with the provided backend.
|
* @param backend the backend to be used with the provided backend.
|
*/
|
protected void updateBaseDNCombo(BackendDescriptor backend)
|
{
|
ArrayList<Object> newElements =
|
new ArrayList<Object>();
|
for (BaseDNDescriptor baseDN : backend.getBaseDns())
|
{
|
String dn = null;
|
try
|
{
|
dn = Utilities.unescapeUtf8(baseDN.getDn().toString());
|
}
|
catch (Throwable t)
|
{
|
throw new RuntimeException("Unexpected error: "+t, t);
|
}
|
newElements.add(dn);
|
}
|
newElements.add(COMBO_SEPARATOR);
|
newElements.add(OTHER_BASE_DN);
|
updateComboBoxModel(newElements, (DefaultComboBoxModel)baseDNs.getModel());
|
}
|
|
/**
|
* Updates a list of errors with the errors found in the panel.
|
* @param checkName whether the name of the VLV index must be checked or not.
|
* @return a list containing the error messages found.
|
*/
|
protected List<Message> checkErrors(boolean checkName)
|
{
|
for (JLabel l : labels)
|
{
|
setPrimaryValid(l);
|
}
|
|
BackendDescriptor backend = getBackend();
|
|
ArrayList<Message> errors = new ArrayList<Message>();
|
if (checkName)
|
{
|
String n = name.getText();
|
if (n.trim().length() == 0)
|
{
|
errors.add(ERR_CTRL_PANEL_NO_VLV_INDEX_NAME_PROVIDED.get());
|
setPrimaryInvalid(lName);
|
}
|
else if (backend != null)
|
{
|
// Check that there is no other VLV index with same name
|
for (VLVIndexDescriptor index : backend.getVLVIndexes())
|
{
|
if (index.getName().equalsIgnoreCase(n))
|
{
|
errors.add(ERR_CTRL_PANEL_VLV_INDEX_ALREADY_DEFINED.get(n,
|
backendName.getText()));
|
setPrimaryInvalid(lName);
|
break;
|
}
|
}
|
}
|
}
|
|
String baseDN = getBaseDN();
|
if ((baseDN == null) || (baseDN.length() == 0))
|
{
|
errors.add(ERR_CTRL_PANEL_NO_BASE_DN_FOR_VLV_PROVIDED.get());
|
setPrimaryInvalid(lBaseDN);
|
}
|
else
|
{
|
try
|
{
|
DN.decode(baseDN);
|
}
|
catch (OpenDsException oe)
|
{
|
errors.add(ERR_CTRL_PANEL_INVALID_BASE_DN_FOR_VLV_PROVIDED.get(
|
oe.getMessageObject().toString()));
|
setPrimaryInvalid(lBaseDN);
|
}
|
}
|
|
String f = filter.getText().trim();
|
if (f.equals(""))
|
{
|
errors.add(ERR_CTRL_PANEL_NO_FILTER_FOR_VLV_PROVIDED.get());
|
setPrimaryInvalid(lFilter);
|
}
|
else
|
{
|
try
|
{
|
LDAPFilter.decode(f);
|
}
|
catch (LDAPException le)
|
{
|
errors.add(ERR_CTRL_PANEL_INVALID_FILTER_FOR_VLV_PROVIDED.get(
|
le.getMessageObject().toString()));
|
setPrimaryInvalid(lFilter);
|
}
|
}
|
|
if (sortOrder.getModel().getSize() == 0)
|
{
|
errors.add(ERR_CTRL_PANEL_NO_ATTRIBUTE_FOR_VLV_PROVIDED.get());
|
setPrimaryInvalid(lSortOrder);
|
}
|
String v = maxBlockSize.getText();
|
try
|
{
|
int n = Integer.parseInt(v);
|
if ((n < MIN_MAX_BLOCK_SIZE) || (n > MAX_MAX_BLOCK_SIZE))
|
{
|
errors.add(ERR_CTRL_PANEL_INVALID_MAX_BLOCK_SIZE_FOR_VLV_PROVIDED.get(
|
MIN_MAX_BLOCK_SIZE, MAX_MAX_BLOCK_SIZE));
|
setPrimaryInvalid(lMaxBlockSize);
|
}
|
}
|
catch (Throwable t)
|
{
|
errors.add(ERR_CTRL_PANEL_INVALID_MAX_BLOCK_SIZE_FOR_VLV_PROVIDED.get(
|
MIN_MAX_BLOCK_SIZE, MAX_MAX_BLOCK_SIZE));
|
setPrimaryInvalid(lMaxBlockSize);
|
}
|
return errors;
|
}
|
|
/**
|
* Returns the backend for the index.
|
* @return the backend for the index.
|
*/
|
protected BackendDescriptor getBackend()
|
{
|
// Check that the index does not exist
|
BackendDescriptor backend = null;
|
for (BackendDescriptor b : getInfo().getServerDescriptor().getBackends())
|
{
|
if (b.getBackendID().equalsIgnoreCase(backendName.getText()))
|
{
|
backend = b;
|
break;
|
}
|
}
|
return backend;
|
}
|
|
/**
|
* Returns the base DN for the VLV index.
|
* @return the base DN for the VLV index.
|
*/
|
protected String getBaseDN()
|
{
|
Object selectedItem = baseDNs.getSelectedItem();
|
if (OTHER_BASE_DN.equals(selectedItem))
|
{
|
selectedItem = baseDN.getText().trim();
|
}
|
if (selectedItem != null)
|
{
|
return selectedItem.toString();
|
}
|
else
|
{
|
return null;
|
}
|
}
|
|
|
/**
|
* Returns the selected attribute.
|
* @return the selected attribute.
|
*/
|
protected String getSelectedAttribute()
|
{
|
String attrName;
|
CategorizedComboBoxElement o =
|
(CategorizedComboBoxElement)attributes.getSelectedItem();
|
if (o != null)
|
{
|
attrName = o.getValue().toString();
|
}
|
else
|
{
|
attrName = null;
|
}
|
return attrName;
|
}
|
|
|
/**
|
* Creates the basic layout of the panel.
|
* @param c the container of the layout.
|
* @param gbc the grid bag constraints to be used.
|
* @param nameReadOnly whether the panel is read-only or not.
|
*/
|
protected void createBasicLayout(Container c, GridBagConstraints gbc,
|
boolean nameReadOnly)
|
{
|
gbc.gridx = 0;
|
gbc.gridy = 0;
|
gbc.gridwidth = 3;
|
addErrorPane(c, gbc);
|
|
if (nameReadOnly)
|
{
|
gbc.gridy ++;
|
titlePanel.setTitle(INFO_CTRL_PANEL_VLV_INDEX_DETAILS_LABEL.get());
|
gbc.fill = GridBagConstraints.NONE;
|
gbc.anchor = GridBagConstraints.WEST;
|
gbc.insets.top = 10;
|
c.add(titlePanel, gbc);
|
}
|
|
gbc.gridy ++;
|
gbc.gridwidth = 1;
|
gbc.anchor = GridBagConstraints.WEST;
|
gbc.fill = GridBagConstraints.HORIZONTAL;
|
gbc.insets.left = 0;
|
gbc.gridx = 0;
|
c.add(lName, gbc);
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
gbc.gridwidth = 2;
|
|
if (nameReadOnly)
|
{
|
c.add(readOnlyName, gbc);
|
}
|
else
|
{
|
JPanel p = new JPanel(new GridBagLayout());
|
p.setOpaque(false);
|
c.add(p, gbc);
|
GridBagConstraints gbc2 = new GridBagConstraints();
|
gbc2.weightx = 0.3;
|
gbc2.fill = GridBagConstraints.HORIZONTAL;
|
gbc2.gridwidth = GridBagConstraints.RELATIVE;
|
p.add(name, gbc2);
|
gbc2.gridwidth = GridBagConstraints.REMAINDER;
|
gbc2.weightx = 0.7;
|
p.add(Box.createHorizontalGlue(), gbc2);
|
}
|
gbc.gridy ++;
|
|
gbc.insets.left = 0;
|
gbc.insets.top = 10;
|
gbc.gridx = 0;
|
c.add(lBackend, gbc);
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
gbc.gridwidth = 2;
|
c.add(backendName, gbc);
|
gbc.gridy ++;
|
|
gbc.insets.left = 0;
|
gbc.gridx = 0;
|
gbc.gridwidth = 1;
|
gbc.anchor = GridBagConstraints.NORTHWEST;
|
c.add(lBaseDN, gbc);
|
gbc.anchor = GridBagConstraints.WEST;
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
gbc.gridwidth = 2;
|
JPanel p = new JPanel(new GridBagLayout());
|
p.setOpaque(false);
|
c.add(p, gbc);
|
gbc.gridy ++;
|
|
DefaultComboBoxModel model = new DefaultComboBoxModel(
|
new Object[]{COMBO_SEPARATOR, OTHER_BASE_DN});
|
baseDNs.setModel(model);
|
baseDNs.setRenderer(new CustomListCellRenderer(baseDNs));
|
ItemListener listener = new IgnoreItemListener(baseDNs);
|
baseDNs.addItemListener(listener);
|
baseDNs.addItemListener(new ItemListener()
|
{
|
public void itemStateChanged(ItemEvent ev)
|
{
|
baseDN.setEnabled(OTHER_BASE_DN.equals(baseDNs.getSelectedItem()));
|
}
|
});
|
listener.itemStateChanged(null);
|
GridBagConstraints gbc2 = new GridBagConstraints();
|
gbc2.fill = GridBagConstraints.HORIZONTAL;
|
p.add(baseDNs, gbc2);
|
gbc2.gridwidth = GridBagConstraints.REMAINDER;
|
gbc2.weightx = 1.0;
|
gbc2.insets.left = 5;
|
p.add(baseDN, gbc2);
|
gbc2.insets.top = 3;
|
JLabel inlineHelp = Utilities.createInlineHelpLabel(
|
INFO_CTRL_PANEL_SUBTREE_INLINE_HELP_LABEL.get());
|
p.add(inlineHelp, gbc2);
|
|
|
gbc.insets.left = 0;
|
gbc.gridx = 0;
|
gbc.gridwidth = 1;
|
c.add(lSearchScope, gbc);
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
gbc.gridwidth = 2;
|
JRadioButton [] radios = {baseObject, singleLevel, subordinateSubtree,
|
wholeSubtree
|
};
|
singleLevel.setSelected(true);
|
ButtonGroup group = new ButtonGroup();
|
for (JRadioButton radio : radios)
|
{
|
c.add(radio, gbc);
|
group.add(radio);
|
gbc.insets.top = 5;
|
gbc.gridy ++;
|
}
|
|
gbc.insets.top = 10;
|
gbc.insets.left = 0;
|
gbc.gridx = 0;
|
gbc.gridwidth = 1;
|
c.add(lFilter, gbc);
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
gbc.gridwidth = 2;
|
c.add(filter, gbc);
|
gbc.gridy ++;
|
gbc.insets.top = 3;
|
inlineHelp = Utilities.createInlineHelpLabel(
|
INFO_CTRL_PANEL_FILTER_INLINE_HELP_LABEL.get());
|
c.add(inlineHelp, gbc);
|
gbc.gridy ++;
|
|
gbc.insets.top = 10;
|
gbc.insets.left = 0;
|
gbc.gridx = 0;
|
gbc.gridwidth = 1;
|
c.add(lMaxBlockSize, gbc);
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
gbc.fill = GridBagConstraints.HORIZONTAL;
|
c.add(maxBlockSize, gbc);
|
maxBlockSize.setText(String.valueOf(DEFAULT_MAX_BLOCK_SIZE));
|
gbc.gridy ++;
|
|
gbc.insets.top = 10;
|
gbc.insets.left = 0;
|
gbc.gridx = 0;
|
gbc.gridwidth = 1;
|
c.add(lSortOrder, gbc);
|
gbc.fill = GridBagConstraints.HORIZONTAL;
|
gbc.insets.left = 10;
|
gbc.gridx = 1;
|
attributes.addItemListener(new IgnoreItemListener(attributes));
|
attributes.setRenderer(new IndexComboBoxCellRenderer(attributes));
|
c.add(attributes, gbc);
|
gbc.gridx ++;
|
|
ascendingOrder.setModel(new DefaultComboBoxModel(
|
new Object[] {ASCENDING, DESCENDING}));
|
c.add(ascendingOrder, gbc);
|
gbc.gridy ++;
|
|
final ListSelectionListener listListener = new ListSelectionListener()
|
{
|
public void valueChanged(ListSelectionEvent ev)
|
{
|
int[] indexes = sortOrder.getSelectedIndices();
|
if ((indexes != null) && (indexes.length > 0))
|
{
|
moveUp.setEnabled(indexes[0] != 0);
|
moveDown.setEnabled(indexes[indexes.length - 1] !=
|
sortOrder.getModel().getSize() - 1);
|
remove.setEnabled(true);
|
}
|
else
|
{
|
moveUp.setEnabled(false);
|
moveUp.setEnabled(false);
|
remove.setEnabled(false);
|
}
|
}
|
};
|
|
JButton[] buttons = {add, remove, moveUp, moveDown};
|
for (JButton button : buttons)
|
{
|
button.setOpaque(false);
|
button.setEnabled(false);
|
}
|
|
add.addActionListener(new ActionListener()
|
{
|
public void actionPerformed(ActionEvent ev)
|
{
|
String attr = getSelectedAttribute();
|
if (attr != null)
|
{
|
boolean isAscending = ASCENDING == ascendingOrder.getSelectedItem();
|
sortOrderModel.addElement(new VLVSortOrder(attr, isAscending));
|
DefaultComboBoxModel model =
|
(DefaultComboBoxModel)attributes.getModel();
|
int i = attributes.getSelectedIndex();
|
if (i > 0)
|
{
|
// To avoid issues, try to figure out first the new selection
|
int newIndex = -1;
|
for (int j= i -1; j>0 && (newIndex == -1); j--)
|
{
|
CategorizedComboBoxElement o = (CategorizedComboBoxElement)
|
model.getElementAt(j);
|
if (o.getType() == CategorizedComboBoxElement.Type.REGULAR)
|
{
|
newIndex = j;
|
}
|
}
|
if (newIndex == -1)
|
{
|
for (int j= i + 1; j<model.getSize() && (newIndex == -1); j++)
|
{
|
CategorizedComboBoxElement o = (CategorizedComboBoxElement)
|
model.getElementAt(j);
|
if (o.getType() == CategorizedComboBoxElement.Type.REGULAR)
|
{
|
newIndex = j;
|
}
|
}
|
}
|
if (newIndex != -1)
|
{
|
attributes.setSelectedIndex(newIndex);
|
}
|
model.removeElementAt(i);
|
}
|
}
|
listListener.valueChanged(null);
|
}
|
});
|
moveUp.addActionListener(new ActionListener()
|
{
|
public void actionPerformed(ActionEvent ev)
|
{
|
int[] indexes = sortOrder.getSelectedIndices();
|
for (int i=0; i<indexes.length; i++)
|
{
|
Object o1 = sortOrderModel.elementAt(indexes[i] - 1);
|
Object o2 = sortOrderModel.elementAt(indexes[i]);
|
sortOrderModel.set(indexes[i] - 1, o2);
|
sortOrderModel.set(indexes[i], o1);
|
|
indexes[i] = indexes[i] - 1;
|
}
|
sortOrder.setSelectedIndices(indexes);
|
listListener.valueChanged(null);
|
}
|
});
|
moveDown.addActionListener(new ActionListener()
|
{
|
public void actionPerformed(ActionEvent ev)
|
{
|
int[] indexes = sortOrder.getSelectedIndices();
|
for (int i=0; i<indexes.length; i++)
|
{
|
Object o1 = sortOrderModel.elementAt(indexes[i] + 1);
|
Object o2 = sortOrderModel.elementAt(indexes[i]);
|
sortOrderModel.set(indexes[i] + 1, o2);
|
sortOrderModel.set(indexes[i], o1);
|
|
indexes[i] = indexes[i] + 1;
|
}
|
sortOrder.setSelectedIndices(indexes);
|
listListener.valueChanged(null);
|
}
|
});
|
remove.addActionListener(new ActionListener()
|
{
|
public void actionPerformed(ActionEvent ev)
|
{
|
int[] indexes = sortOrder.getSelectedIndices();
|
|
synchronized (standardAttrNames)
|
{
|
DefaultComboBoxModel model =
|
(DefaultComboBoxModel)attributes.getModel();
|
for (int i=0; i<indexes.length; i++)
|
{
|
VLVSortOrder sortOrder = (VLVSortOrder)sortOrderModel.getElementAt(
|
indexes[i]);
|
String attrName = sortOrder.getAttributeName();
|
boolean isCustom = customAttrNames.contains(attrName);
|
boolean dealingWithCustom = true;
|
for (int j = 0; j < model.getSize(); j++)
|
{
|
CategorizedComboBoxElement o = (CategorizedComboBoxElement)
|
model.getElementAt(j);
|
if (o.getType() == CategorizedComboBoxElement.Type.REGULAR)
|
{
|
if (dealingWithCustom == isCustom)
|
{
|
if (attrName.compareTo(o.getValue().toString()) < 0)
|
{
|
model.insertElementAt(new CategorizedComboBoxElement(
|
attrName,
|
CategorizedComboBoxElement.Type.REGULAR), j);
|
break;
|
}
|
}
|
}
|
else if (!o.getValue().equals(CUSTOM_ATTRIBUTES))
|
{
|
dealingWithCustom = false;
|
if (isCustom)
|
{
|
model.insertElementAt(new CategorizedComboBoxElement(
|
attrName,
|
CategorizedComboBoxElement.Type.REGULAR), j);
|
break;
|
}
|
}
|
}
|
}
|
}
|
|
for (int i=indexes.length - 1; i >=0; i--)
|
{
|
sortOrderModel.remove(indexes[i]);
|
}
|
listListener.valueChanged(null);
|
}
|
});
|
|
gbc.insets.top = 5;
|
gbc.gridx = 1;
|
gbc.fill = GridBagConstraints.NONE;
|
gbc.anchor = GridBagConstraints.EAST;
|
c.add(add, gbc);
|
gbc.gridy ++;
|
|
gbc.insets.top = 10;
|
gbc.gridwidth = 1;
|
gbc.gridheight = 3;
|
gbc.gridx = 1;
|
gbc.fill = GridBagConstraints.BOTH;
|
gbc.anchor = GridBagConstraints.NORTH;
|
gbc.weightx = 1.0;
|
gbc.weighty = 1.0;
|
sortOrderModel = new DefaultListModel();
|
sortOrder.setModel(sortOrderModel);
|
sortOrder.setCellRenderer(new VLVSortOrderRenderer(sortOrder));
|
sortOrder.setVisibleRowCount(6);
|
sortOrder.setPrototypeCellValue("AjA");
|
c.add(Utilities.createScrollPane(sortOrder), gbc);
|
sortOrder.addListSelectionListener(listListener);
|
|
gbc.gridx = 2;
|
gbc.weighty = 0.0;
|
gbc.weightx = 0.0;
|
gbc.gridheight = 1;
|
gbc.insets.left = 5;
|
gbc.fill = GridBagConstraints.HORIZONTAL;
|
gbc.anchor = GridBagConstraints.WEST;
|
c.add(moveUp, gbc);
|
gbc.gridy ++;
|
gbc.insets.top = 5;
|
c.add(moveDown, gbc);
|
gbc.insets.top = 0;
|
gbc.gridy ++;
|
gbc.weighty = 1.0;
|
|
Dimension d = new Dimension(
|
Math.max(moveUp.getPreferredSize().width,
|
moveDown.getPreferredSize().width),
|
Math.max(moveUp.getPreferredSize().height,
|
moveDown.getPreferredSize().height));
|
moveUp.setPreferredSize(d);
|
moveDown.setPreferredSize(d);
|
|
c.add(Box.createVerticalGlue(), gbc);
|
|
gbc.gridx = 1;
|
gbc.gridy ++;
|
gbc.weighty = 0.0;
|
gbc.anchor = GridBagConstraints.NORTHEAST;
|
gbc.fill = GridBagConstraints.NONE;
|
c.add(remove, gbc);
|
}
|
}
|