/*
* 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
*
*
* Portions Copyright 2007 Sun Microsystems, Inc.
*/
package org.opends.server.tools.dsconfig;
import static org.opends.server.messages.MessageHandler.*;
import static org.opends.server.messages.ToolMessages.*;
import java.io.PrintStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeSet;
import org.opends.server.admin.DefinitionDecodingException;
import org.opends.server.admin.IllegalPropertyValueStringException;
import org.opends.server.admin.InstantiableRelationDefinition;
import org.opends.server.admin.ManagedObjectAlreadyExistsException;
import org.opends.server.admin.ManagedObjectDefinition;
import org.opends.server.admin.ManagedObjectNotFoundException;
import org.opends.server.admin.ManagedObjectPath;
import org.opends.server.admin.OptionalRelationDefinition;
import org.opends.server.admin.PropertyDefinition;
import org.opends.server.admin.PropertyException;
import org.opends.server.admin.PropertyOption;
import org.opends.server.admin.RelationDefinition;
import org.opends.server.admin.SingletonRelationDefinition;
import org.opends.server.admin.UndefinedDefaultBehaviorProvider;
import org.opends.server.admin.client.AuthorizationException;
import org.opends.server.admin.client.CommunicationException;
import org.opends.server.admin.client.ConcurrentModificationException;
import org.opends.server.admin.client.ManagedObject;
import org.opends.server.admin.client.ManagedObjectDecodingException;
import org.opends.server.admin.client.ManagementContext;
import org.opends.server.admin.client.MissingMandatoryPropertiesException;
import org.opends.server.admin.client.OperationRejectedException;
import org.opends.server.protocols.ldap.LDAPResultCode;
import org.opends.server.tools.ClientException;
import org.opends.server.util.args.ArgumentException;
import org.opends.server.util.args.StringArgument;
import org.opends.server.util.args.SubCommand;
import org.opends.server.util.args.SubCommandArgumentParser;
/**
* A sub-command handler which is used to modify the properties of a
* managed object.
*
* This sub-command implements the various set-xxx-prop sub-commands.
*/
final class SetPropSubCommandHandler extends SubCommandHandler {
/**
* Type of modication being performed.
*/
private static enum ModificationType {
/**
* Append a single value to the property.
*/
ADD,
/**
* Remove a single value from the property.
*/
REMOVE,
/**
* Append a single value to the property (first invocation removes
* existing values).
*/
SET;
}
/**
* The value for the long option add.
*/
private static final String OPTION_DSCFG_LONG_ADD = "add";
/**
* The value for the long option remove.
*/
private static final String OPTION_DSCFG_LONG_REMOVE = "remove";
/**
* The value for the long option reset.
*/
private static final String OPTION_DSCFG_LONG_RESET = "reset";
/**
* The value for the long option set.
*/
private static final String OPTION_DSCFG_LONG_SET = "set";
/**
* The value for the short option add.
*/
private static final Character OPTION_DSCFG_SHORT_ADD = null;
/**
* The value for the short option remove.
*/
private static final Character OPTION_DSCFG_SHORT_REMOVE = null;
/**
* The value for the short option reset.
*/
private static final Character OPTION_DSCFG_SHORT_RESET = null;
/**
* The value for the short option set.
*/
private static final Character OPTION_DSCFG_SHORT_SET = null;
/**
* Creates a new set-xxx-prop sub-command for an instantiable
* relation.
*
* @param parser
* The sub-command argument parser.
* @param path
* The parent managed object path.
* @param r
* The instantiable relation.
* @return Returns the new set-xxx-prop sub-command.
* @throws ArgumentException
* If the sub-command could not be created successfully.
*/
public static SetPropSubCommandHandler create(
SubCommandArgumentParser parser, ManagedObjectPath, ?> path,
InstantiableRelationDefinition, ?> r) throws ArgumentException {
return new SetPropSubCommandHandler(parser, path.child(r, "DUMMY"), r);
}
/**
* Creates a new set-xxx-prop sub-command for an optional relation.
*
* @param parser
* The sub-command argument parser.
* @param path
* The parent managed object path.
* @param r
* The optional relation.
* @return Returns the new set-xxx-prop sub-command.
* @throws ArgumentException
* If the sub-command could not be created successfully.
*/
public static SetPropSubCommandHandler create(
SubCommandArgumentParser parser, ManagedObjectPath, ?> path,
OptionalRelationDefinition, ?> r) throws ArgumentException {
return new SetPropSubCommandHandler(parser, path.child(r), r);
}
/**
* Creates a new set-xxx-prop sub-command for a singleton relation.
*
* @param parser
* The sub-command argument parser.
* @param path
* The parent managed object path.
* @param r
* The singleton relation.
* @return Returns the new set-xxx-prop sub-command.
* @throws ArgumentException
* If the sub-command could not be created successfully.
*/
public static SetPropSubCommandHandler create(
SubCommandArgumentParser parser, ManagedObjectPath, ?> path,
SingletonRelationDefinition, ?> r) throws ArgumentException {
return new SetPropSubCommandHandler(parser, path.child(r), r);
}
// The sub-commands naming arguments.
private final List namingArgs;
// The path of the managed object.
private final ManagedObjectPath, ?> path;
// The argument which should be used to specify zero or more
// property value adds.
private final StringArgument propertyAddArgument;
// The argument which should be used to specify zero or more
// property value removes.
private final StringArgument propertyRemoveArgument;
// The argument which should be used to specify zero or more
// property value resets.
private final StringArgument propertyResetArgument;
// The argument which should be used to specify zero or more
// property value assignments.
private final StringArgument propertySetArgument;
// The sub-command associated with this handler.
private final SubCommand subCommand;
// Private constructor.
private SetPropSubCommandHandler(SubCommandArgumentParser parser,
ManagedObjectPath, ?> path, RelationDefinition, ?> r)
throws ArgumentException {
this.path = path;
// Create the sub-command.
String name = "set-" + r.getName() + "-prop";
int descriptionID = MSGID_DSCFG_DESCRIPTION_SUBCMD_SETPROP;
this.subCommand = new SubCommand(parser, name, false, 0, 0, null,
descriptionID, r.getChildDefinition().getUserFriendlyName());
// Create the naming arguments.
this.namingArgs = createNamingArgs(subCommand, path);
// Create the --set argument.
this.propertySetArgument = new StringArgument(OPTION_DSCFG_LONG_SET,
OPTION_DSCFG_SHORT_SET, OPTION_DSCFG_LONG_SET, false, true, true,
"{PROP:VAL}", null, null, MSGID_DSCFG_DESCRIPTION_PROP_VAL);
this.subCommand.addArgument(this.propertySetArgument);
// Create the --reset argument.
this.propertyResetArgument = new StringArgument(OPTION_DSCFG_LONG_RESET,
OPTION_DSCFG_SHORT_RESET, OPTION_DSCFG_LONG_RESET, false, true, true,
"{PROP}", null, null, MSGID_DSCFG_DESCRIPTION_RESET_PROP);
this.subCommand.addArgument(this.propertyResetArgument);
// Create the --add argument.
this.propertyAddArgument = new StringArgument(OPTION_DSCFG_LONG_ADD,
OPTION_DSCFG_SHORT_ADD, OPTION_DSCFG_LONG_ADD, false, true, true,
"{PROP:VAL}", null, null, MSGID_DSCFG_DESCRIPTION_ADD_PROP_VAL);
this.subCommand.addArgument(this.propertyAddArgument);
// Create the --remove argument.
this.propertyRemoveArgument = new StringArgument(OPTION_DSCFG_LONG_REMOVE,
OPTION_DSCFG_SHORT_REMOVE, OPTION_DSCFG_LONG_REMOVE, false, true, true,
"{PROP:VAL}", null, null, MSGID_DSCFG_DESCRIPTION_REMOVE_PROP_VAL);
this.subCommand.addArgument(this.propertyRemoveArgument);
// Register the tags associated with the child managed objects.
addTags(path.getManagedObjectDefinition().getAllTags());
}
/**
* {@inheritDoc}
*/
@Override
public SubCommand getSubCommand() {
return subCommand;
}
/**
* {@inheritDoc}
*/
@SuppressWarnings("unchecked")
@Override
public int run(DSConfig app, PrintStream out, PrintStream err)
throws ArgumentException, ClientException {
// Get the naming argument values.
List names = getNamingArgValues(namingArgs);
ManagementContext context = app.getManagementContext();
ManagedObject> child;
try {
child = getManagedObject(context, path, names);
} catch (AuthorizationException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_AUTHZ;
String ufn = path.getManagedObjectDefinition().getUserFriendlyName();
String msg = getMessage(msgID, ufn);
throw new ClientException(LDAPResultCode.INSUFFICIENT_ACCESS_RIGHTS,
msgID, msg);
} catch (DefinitionDecodingException e) {
int msgID = MSGID_DSCFG_ERROR_GET_CHILD_DDE;
String ufn = path.getManagedObjectDefinition().getUserFriendlyName();
String msg = getMessage(msgID, ufn, ufn, ufn);
throw new ClientException(LDAPResultCode.OPERATIONS_ERROR, msgID, msg);
} catch (ManagedObjectDecodingException e) {
// FIXME: should not abort here. Instead, display the errors (if
// verbose) and apply the changes to the partial managed object.
int msgID = MSGID_DSCFG_ERROR_GET_CHILD_MODE;
String ufn = path.getManagedObjectDefinition().getUserFriendlyName();
String msg = getMessage(msgID, ufn);
throw new ClientException(LDAPResultCode.OPERATIONS_ERROR, msgID, msg);
} catch (CommunicationException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_CE;
String ufn = path.getManagedObjectDefinition().getUserFriendlyName();
String msg = getMessage(msgID, ufn, e.getMessage());
throw new ClientException(LDAPResultCode.OPERATIONS_ERROR, msgID, msg);
} catch (ConcurrentModificationException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_CME;
String ufn = path.getManagedObjectDefinition().getUserFriendlyName();
String msg = getMessage(msgID, ufn);
throw new ClientException(LDAPResultCode.CONSTRAINT_VIOLATION,
msgID, msg);
} catch (ManagedObjectNotFoundException e) {
int msgID = MSGID_DSCFG_ERROR_GET_CHILD_MONFE;
String ufn = path.getManagedObjectDefinition().getUserFriendlyName();
String msg = getMessage(msgID, ufn);
throw new ClientException(LDAPResultCode.NO_SUCH_OBJECT, msgID, msg);
}
ManagedObjectDefinition, ?> d = child.getManagedObjectDefinition();
Map lastModTypes =
new HashMap();
Map changes =
new HashMap();
// Reset properties.
for (String m : propertyResetArgument.getValues()) {
// Check the property definition.
PropertyDefinition> pd;
try {
pd = d.getPropertyDefinition(m);
} catch (IllegalArgumentException e) {
throw ArgumentExceptionFactory.unknownProperty(d, m);
}
// Mandatory properties which have no defined defaults cannot be
// reset.
if (pd.hasOption(PropertyOption.MANDATORY)) {
if (pd.getDefaultBehaviorProvider()
instanceof UndefinedDefaultBehaviorProvider) {
throw ArgumentExceptionFactory.unableToResetMandatoryProperty(d, m,
OPTION_DSCFG_LONG_SET);
}
}
// Save the modification type.
lastModTypes.put(m, ModificationType.SET);
// Apply the modification.
modifyPropertyValues(child, pd, changes, ModificationType.SET, null);
}
// Set properties.
for (String m : propertySetArgument.getValues()) {
// Parse the property "property:value".
int sep = m.indexOf(':');
if (sep < 0) {
throw ArgumentExceptionFactory.missingSeparatorInPropertyArgument(m);
}
if (sep == 0) {
throw ArgumentExceptionFactory.missingNameInPropertyArgument(m);
}
String propertyName = m.substring(0, sep);
String value = m.substring(sep + 1, m.length());
if (value.length() == 0) {
throw ArgumentExceptionFactory.missingValueInPropertyArgument(m);
}
// Check the property definition.
PropertyDefinition> pd;
try {
pd = d.getPropertyDefinition(propertyName);
} catch (IllegalArgumentException e) {
throw ArgumentExceptionFactory.unknownProperty(d, propertyName);
}
// Apply the modification.
if (lastModTypes.containsKey(propertyName)) {
modifyPropertyValues(child, pd, changes, ModificationType.ADD, value);
} else {
lastModTypes.put(propertyName, ModificationType.SET);
modifyPropertyValues(child, pd, changes, ModificationType.SET, value);
}
}
// Remove properties.
for (String m : propertyRemoveArgument.getValues()) {
// Parse the property "property:value".
int sep = m.indexOf(':');
if (sep < 0) {
throw ArgumentExceptionFactory.missingSeparatorInPropertyArgument(m);
}
if (sep == 0) {
throw ArgumentExceptionFactory.missingNameInPropertyArgument(m);
}
String propertyName = m.substring(0, sep);
String value = m.substring(sep + 1, m.length());
if (value.length() == 0) {
throw ArgumentExceptionFactory.missingValueInPropertyArgument(m);
}
// Check the property definition.
PropertyDefinition> pd;
try {
pd = d.getPropertyDefinition(propertyName);
} catch (IllegalArgumentException e) {
throw ArgumentExceptionFactory.unknownProperty(d, propertyName);
}
// Apply the modification.
if (lastModTypes.containsKey(propertyName)) {
if (lastModTypes.get(propertyName) == ModificationType.SET) {
throw ArgumentExceptionFactory.incompatiblePropertyModification(m);
}
} else {
lastModTypes.put(propertyName, ModificationType.REMOVE);
modifyPropertyValues(child, pd, changes,
ModificationType.REMOVE, value);
}
}
// Add properties.
for (String m : propertyAddArgument.getValues()) {
// Parse the property "property:value".
int sep = m.indexOf(':');
if (sep < 0) {
throw ArgumentExceptionFactory.missingSeparatorInPropertyArgument(m);
}
if (sep == 0) {
throw ArgumentExceptionFactory.missingNameInPropertyArgument(m);
}
String propertyName = m.substring(0, sep);
String value = m.substring(sep + 1, m.length());
if (value.length() == 0) {
throw ArgumentExceptionFactory.missingValueInPropertyArgument(m);
}
// Check the property definition.
PropertyDefinition> pd;
try {
pd = d.getPropertyDefinition(propertyName);
} catch (IllegalArgumentException e) {
throw ArgumentExceptionFactory.unknownProperty(d, propertyName);
}
// Apply the modification.
if (lastModTypes.containsKey(propertyName)) {
if (lastModTypes.get(propertyName) == ModificationType.SET) {
throw ArgumentExceptionFactory.incompatiblePropertyModification(m);
}
} else {
lastModTypes.put(propertyName, ModificationType.ADD);
modifyPropertyValues(child, pd, changes, ModificationType.ADD, value);
}
}
// Commit the changes.
for (PropertyDefinition> pd : changes.keySet()) {
try {
child.setPropertyValues(pd, changes.get(pd));
} catch (PropertyException e) {
throw ArgumentExceptionFactory.adaptPropertyException(e, d);
}
}
try {
// Confirm commit.
String prompt = String.format(Messages.getString("modify.confirm"), d
.getUserFriendlyName());
if (!app.confirmAction(prompt)) {
// Output failure message.
String msg = String.format(Messages.getString("modify.failed"), d
.getUserFriendlyName());
app.displayVerboseMessage(msg);
return 1;
}
child.commit();
// Output success message.
String msg = String.format(Messages.getString("modify.done"), d
.getUserFriendlyName());
app.displayVerboseMessage(msg);
} catch (MissingMandatoryPropertiesException e) {
throw ArgumentExceptionFactory.adaptMissingMandatoryPropertiesException(
e, d);
} catch (AuthorizationException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_AUTHZ;
String msg = getMessage(msgID, d.getUserFriendlyName());
throw new ClientException(LDAPResultCode.INSUFFICIENT_ACCESS_RIGHTS,
msgID, msg);
} catch (ConcurrentModificationException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_CME;
String msg = getMessage(msgID, d.getUserFriendlyName());
throw new ClientException(LDAPResultCode.CONSTRAINT_VIOLATION,
msgID, msg);
} catch (OperationRejectedException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_ORE;
String msg = getMessage(msgID, d.getUserFriendlyName(), e.getMessage());
throw new ClientException(LDAPResultCode.CONSTRAINT_VIOLATION,
msgID, msg);
} catch (CommunicationException e) {
int msgID = MSGID_DSCFG_ERROR_MODIFY_CE;
String msg = getMessage(msgID, d.getUserFriendlyName(), e.getMessage());
throw new ClientException(LDAPResultCode.OPERATIONS_ERROR, msgID, msg);
} catch (ManagedObjectAlreadyExistsException e) {
// Should never happen.
throw new IllegalStateException(e);
}
return 0;
}
// Apply a single modification to the current change-set.
@SuppressWarnings("unchecked")
private void modifyPropertyValues(ManagedObject> mo,
PropertyDefinition pd, Map changes,
ModificationType modType, String s) throws ArgumentException {
Set values = changes.get(pd);
if (values == null) {
values = mo.getPropertyValues(pd);
}
if (s == null || s.length() == 0) {
// Reset back to defaults.
values.clear();
} else {
T value;
try {
value = pd.decodeValue(s);
} catch (IllegalPropertyValueStringException e) {
throw ArgumentExceptionFactory.adaptPropertyException(e, mo
.getManagedObjectDefinition());
}
switch (modType) {
case ADD:
values.add(value);
break;
case REMOVE:
values.remove(value);
break;
case SET:
values = new TreeSet(pd);
values.add(value);
break;
}
}
changes.put(pd, values);
}
}