/*
|
* 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 2010 Sun Microsystems, Inc.
|
* Portions copyright 2011-2013 ForgeRock AS
|
*/
|
|
package org.forgerock.opendj.ldap;
|
|
import static org.forgerock.opendj.ldap.AttributeDescription.objectClass;
|
import static org.forgerock.opendj.ldap.CoreMessages.ERR_ENTRY_DUPLICATE_VALUES;
|
import static org.forgerock.opendj.ldap.CoreMessages.ERR_ENTRY_INCREMENT_CANNOT_PARSE_AS_INT;
|
import static org.forgerock.opendj.ldap.CoreMessages.ERR_ENTRY_INCREMENT_INVALID_VALUE_COUNT;
|
import static org.forgerock.opendj.ldap.CoreMessages.ERR_ENTRY_INCREMENT_NO_SUCH_ATTRIBUTE;
|
import static org.forgerock.opendj.ldap.CoreMessages.ERR_ENTRY_NO_SUCH_VALUE;
|
import static org.forgerock.opendj.ldap.CoreMessages.ERR_ENTRY_UNKNOWN_MODIFICATION_TYPE;
|
import static org.forgerock.opendj.ldap.ErrorResultException.newErrorResult;
|
|
import java.util.ArrayList;
|
import java.util.Collection;
|
import java.util.Collections;
|
import java.util.Comparator;
|
import java.util.HashSet;
|
import java.util.Iterator;
|
import java.util.Set;
|
|
import org.forgerock.i18n.LocalizableMessage;
|
import org.forgerock.opendj.ldap.controls.PermissiveModifyRequestControl;
|
import org.forgerock.opendj.ldap.requests.ModifyRequest;
|
import org.forgerock.opendj.ldap.requests.Requests;
|
import org.forgerock.opendj.ldap.schema.ObjectClass;
|
import org.forgerock.opendj.ldap.schema.ObjectClassType;
|
import org.forgerock.opendj.ldap.schema.Schema;
|
import org.forgerock.opendj.ldap.schema.SchemaValidationPolicy;
|
import org.forgerock.opendj.ldap.schema.UnknownSchemaElementException;
|
|
import com.forgerock.opendj.util.Iterables;
|
import com.forgerock.opendj.util.Validator;
|
|
/**
|
* This class contains methods for creating and manipulating entries.
|
*
|
* @see Entry
|
*/
|
public final class Entries {
|
|
private static final class UnmodifiableEntry implements Entry {
|
private final Entry entry;
|
|
private UnmodifiableEntry(final Entry entry) {
|
this.entry = entry;
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public boolean addAttribute(final Attribute attribute) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public boolean addAttribute(final Attribute attribute,
|
final Collection<? super ByteString> duplicateValues) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public Entry addAttribute(final String attributeDescription, final Object... values) {
|
throw new UnsupportedOperationException();
|
}
|
|
@Override
|
public Entry clearAttributes() {
|
throw new UnsupportedOperationException();
|
}
|
|
@Override
|
public boolean containsAttribute(final Attribute attribute,
|
final Collection<? super ByteString> missingValues) {
|
return entry.containsAttribute(attribute, missingValues);
|
}
|
|
@Override
|
public boolean containsAttribute(final String attributeDescription, final Object... values) {
|
return entry.containsAttribute(attributeDescription, values);
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public boolean equals(final Object object) {
|
return (object == this || entry.equals(object));
|
}
|
|
@Override
|
public Iterable<Attribute> getAllAttributes() {
|
return Iterables.unmodifiableIterable(Iterables.transformedIterable(entry
|
.getAllAttributes(), UNMODIFIABLE_ATTRIBUTE_FUNCTION));
|
}
|
|
@Override
|
public Iterable<Attribute> getAllAttributes(final AttributeDescription attributeDescription) {
|
return Iterables.unmodifiableIterable(Iterables.transformedIterable(entry
|
.getAllAttributes(attributeDescription), UNMODIFIABLE_ATTRIBUTE_FUNCTION));
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public Iterable<Attribute> getAllAttributes(final String attributeDescription) {
|
return Iterables.unmodifiableIterable(Iterables.transformedIterable(entry
|
.getAllAttributes(attributeDescription), UNMODIFIABLE_ATTRIBUTE_FUNCTION));
|
}
|
|
@Override
|
public Attribute getAttribute(final AttributeDescription attributeDescription) {
|
final Attribute attribute = entry.getAttribute(attributeDescription);
|
if (attribute != null) {
|
return Attributes.unmodifiableAttribute(attribute);
|
} else {
|
return null;
|
}
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public Attribute getAttribute(final String attributeDescription) {
|
final Attribute attribute = entry.getAttribute(attributeDescription);
|
if (attribute != null) {
|
return Attributes.unmodifiableAttribute(attribute);
|
} else {
|
return null;
|
}
|
}
|
|
@Override
|
public int getAttributeCount() {
|
return entry.getAttributeCount();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public DN getName() {
|
return entry.getName();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public int hashCode() {
|
return entry.hashCode();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public AttributeParser parseAttribute(final AttributeDescription attributeDescription) {
|
return entry.parseAttribute(attributeDescription);
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public AttributeParser parseAttribute(final String attributeDescription) {
|
return entry.parseAttribute(attributeDescription);
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public boolean removeAttribute(final Attribute attribute,
|
final Collection<? super ByteString> missingValues) {
|
throw new UnsupportedOperationException();
|
}
|
|
@Override
|
public boolean removeAttribute(final AttributeDescription attributeDescription) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public Entry removeAttribute(final String attributeDescription, final Object... values) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public boolean replaceAttribute(final Attribute attribute) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public Entry replaceAttribute(final String attributeDescription, final Object... values) {
|
throw new UnsupportedOperationException();
|
}
|
|
@Override
|
public Entry setName(final DN dn) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public Entry setName(final String dn) {
|
throw new UnsupportedOperationException();
|
}
|
|
/**
|
* {@inheritDoc}
|
*/
|
@Override
|
public String toString() {
|
return entry.toString();
|
}
|
|
}
|
|
private static final Comparator<Entry> COMPARATOR = new Comparator<Entry>() {
|
@Override
|
public int compare(final Entry o1, final Entry o2) {
|
return o1.getName().compareTo(o2.getName());
|
}
|
};
|
|
private static final Function<Attribute, Attribute, Void> UNMODIFIABLE_ATTRIBUTE_FUNCTION =
|
new Function<Attribute, Attribute, Void>() {
|
|
@Override
|
public Attribute apply(final Attribute value, final Void p) {
|
return Attributes.unmodifiableAttribute(value);
|
}
|
|
};
|
|
/**
|
* Returns a {@code Comparator} which can be used to compare entries by name
|
* using the natural order for DN comparisons (parent before children).
|
* <p>
|
* In order to sort entries in reverse order (children first) use the
|
* following code:
|
*
|
* <pre>
|
* Collections.reverseOrder(Entries.compareByName());
|
* </pre>
|
*
|
* For more complex sort orders involving one or more attributes refer to
|
* the {@link SortKey} class.
|
*
|
* @return The {@code Comparator}.
|
*/
|
public static Comparator<Entry> compareByName() {
|
return COMPARATOR;
|
}
|
|
/**
|
* Returns {@code true} if the provided entry is valid according to the
|
* specified schema and schema validation policy.
|
* <p>
|
* If attribute value validation is enabled then following checks will be
|
* performed:
|
* <ul>
|
* <li>checking that there is at least one value
|
* <li>checking that single-valued attributes contain only a single value
|
* </ul>
|
* In particular, attribute values will not be checked for conformance to
|
* their syntax since this is expected to have already been performed while
|
* adding the values to the entry.
|
*
|
* @param entry
|
* The entry to be validated.
|
* @param schema
|
* The schema against which the entry will be validated.
|
* @param policy
|
* The schema validation policy.
|
* @param errorMessages
|
* A collection into which any schema validation warnings or
|
* error messages can be placed, or {@code null} if they should
|
* not be saved.
|
* @return {@code true} if the provided entry is valid according to the
|
* specified schema and schema validation policy.
|
* @see Schema#validateEntry(Entry, SchemaValidationPolicy, Collection)
|
*/
|
public static boolean conformsToSchema(final Entry entry, final Schema schema,
|
final SchemaValidationPolicy policy, final Collection<LocalizableMessage> errorMessages) {
|
return schema.validateEntry(entry, policy, errorMessages);
|
}
|
|
/**
|
* Returns {@code true} if the provided entry is valid according to the
|
* default schema and schema validation policy.
|
* <p>
|
* If attribute value validation is enabled then following checks will be
|
* performed:
|
* <ul>
|
* <li>checking that there is at least one value
|
* <li>checking that single-valued attributes contain only a single value
|
* </ul>
|
* In particular, attribute values will not be checked for conformance to
|
* their syntax since this is expected to have already been performed while
|
* adding the values to the entry.
|
*
|
* @param entry
|
* The entry to be validated.
|
* @param policy
|
* The schema validation policy.
|
* @param errorMessages
|
* A collection into which any schema validation warnings or
|
* error messages can be placed, or {@code null} if they should
|
* not be saved.
|
* @return {@code true} if the provided entry is valid according to the
|
* default schema and schema validation policy.
|
* @see Schema#validateEntry(Entry, SchemaValidationPolicy, Collection)
|
*/
|
public static boolean conformsToSchema(final Entry entry, final SchemaValidationPolicy policy,
|
final Collection<LocalizableMessage> errorMessages) {
|
return conformsToSchema(entry, Schema.getDefaultSchema(), policy, errorMessages);
|
}
|
|
/**
|
* Creates a new modify request containing a list of modifications which can
|
* be used to transform {@code fromEntry} into entry {@code toEntry}.
|
* <p>
|
* The modify request is reversible: it will contain only modifications of
|
* type {@link ModificationType#ADD ADD} and {@link ModificationType#DELETE
|
* DELETE}.
|
* <p>
|
* Finally, the modify request will use the distinguished name taken from
|
* {@code fromEntry}. Moreover, this method will not check to see if both
|
* {@code fromEntry} and {@code toEntry} have the same distinguished name.
|
* <p>
|
* This method is equivalent to:
|
*
|
* <pre>
|
* ModifyRequest request = Requests.newModifyRequest(fromEntry, toEntry);
|
* </pre>
|
*
|
* @param fromEntry
|
* The source entry.
|
* @param toEntry
|
* The destination entry.
|
* @return A modify request containing a list of modifications which can be
|
* used to transform {@code fromEntry} into entry {@code toEntry}.
|
* @throws NullPointerException
|
* If {@code fromEntry} or {@code toEntry} were {@code null}.
|
* @see Requests#newModifyRequest(Entry, Entry)
|
*/
|
public static ModifyRequest diffEntries(final Entry fromEntry, final Entry toEntry) {
|
Validator.ensureNotNull(fromEntry, toEntry);
|
|
final ModifyRequest request = Requests.newModifyRequest(fromEntry.getName());
|
|
TreeMapEntry tfrom;
|
if (fromEntry instanceof TreeMapEntry) {
|
tfrom = (TreeMapEntry) fromEntry;
|
} else {
|
tfrom = new TreeMapEntry(fromEntry);
|
}
|
|
TreeMapEntry tto;
|
if (toEntry instanceof TreeMapEntry) {
|
tto = (TreeMapEntry) toEntry;
|
} else {
|
tto = new TreeMapEntry(toEntry);
|
}
|
|
final Iterator<Attribute> ifrom = tfrom.getAllAttributes().iterator();
|
final Iterator<Attribute> ito = tto.getAllAttributes().iterator();
|
|
Attribute afrom = ifrom.hasNext() ? ifrom.next() : null;
|
Attribute ato = ito.hasNext() ? ito.next() : null;
|
|
while (afrom != null && ato != null) {
|
final AttributeDescription adfrom = afrom.getAttributeDescription();
|
final AttributeDescription adto = ato.getAttributeDescription();
|
|
final int cmp = adfrom.compareTo(adto);
|
if (cmp == 0) {
|
// Attribute is in both entries. Compute the set of values to be
|
// added
|
// and removed. We won't replace the attribute because this is
|
// not
|
// reversible.
|
final Attribute addedValues = new LinkedAttribute(ato);
|
addedValues.removeAll(afrom);
|
if (!addedValues.isEmpty()) {
|
request.addModification(new Modification(ModificationType.ADD, addedValues));
|
}
|
|
final Attribute deletedValues = new LinkedAttribute(afrom);
|
deletedValues.removeAll(ato);
|
if (!deletedValues.isEmpty()) {
|
request.addModification(new Modification(ModificationType.DELETE, deletedValues));
|
}
|
|
afrom = ifrom.hasNext() ? ifrom.next() : null;
|
ato = ito.hasNext() ? ito.next() : null;
|
} else if (cmp < 0) {
|
// afrom in source, but not destination.
|
request.addModification(new Modification(ModificationType.DELETE, afrom));
|
afrom = ifrom.hasNext() ? ifrom.next() : null;
|
} else {
|
// ato in destination, but not in source.
|
request.addModification(new Modification(ModificationType.ADD, ato));
|
ato = ito.hasNext() ? ito.next() : null;
|
}
|
}
|
|
// Additional attributes in source entry: these must be deleted.
|
if (afrom != null) {
|
request.addModification(new Modification(ModificationType.DELETE, afrom));
|
}
|
|
while (ifrom.hasNext()) {
|
final Attribute a = ifrom.next();
|
request.addModification(new Modification(ModificationType.DELETE, a));
|
}
|
|
// Additional attributes in destination entry: these must be added.
|
if (ato != null) {
|
request.addModification(new Modification(ModificationType.ADD, ato));
|
}
|
|
while (ito.hasNext()) {
|
final Attribute a = ito.next();
|
request.addModification(new Modification(ModificationType.ADD, a));
|
}
|
|
return request;
|
}
|
|
/**
|
* Returns an unmodifiable set containing the object classes associated with
|
* the provided entry. This method will ignore unrecognized object classes.
|
* <p>
|
* This method uses the default schema for decoding the object class
|
* attribute values.
|
*
|
* @param entry
|
* The entry whose object classes are required.
|
* @return An unmodifiable set containing the object classes associated with
|
* the provided entry.
|
*/
|
public static Set<ObjectClass> getObjectClasses(final Entry entry) {
|
return getObjectClasses(entry, Schema.getDefaultSchema());
|
}
|
|
/**
|
* Returns an unmodifiable set containing the object classes associated with
|
* the provided entry. This method will ignore unrecognized object classes.
|
*
|
* @param entry
|
* The entry whose object classes are required.
|
* @param schema
|
* The schema which should be used for decoding the object class
|
* attribute values.
|
* @return An unmodifiable set containing the object classes associated with
|
* the provided entry.
|
*/
|
public static Set<ObjectClass> getObjectClasses(final Entry entry, final Schema schema) {
|
final Attribute objectClassAttribute =
|
entry.getAttribute(AttributeDescription.objectClass());
|
if (objectClassAttribute == null) {
|
return Collections.emptySet();
|
} else {
|
final Set<ObjectClass> objectClasses =
|
new HashSet<ObjectClass>(objectClassAttribute.size());
|
for (final ByteString v : objectClassAttribute) {
|
final String objectClassName = v.toString();
|
final ObjectClass objectClass;
|
try {
|
objectClass = schema.getObjectClass(objectClassName);
|
objectClasses.add(objectClass);
|
} catch (final UnknownSchemaElementException e) {
|
// Ignore.
|
continue;
|
}
|
}
|
return Collections.unmodifiableSet(objectClasses);
|
}
|
}
|
|
/**
|
* Returns the structural object class associated with the provided entry,
|
* or {@code null} if none was found. If the entry contains multiple
|
* structural object classes then the first will be returned. This method
|
* will ignore unrecognized object classes.
|
* <p>
|
* This method uses the default schema for decoding the object class
|
* attribute values.
|
*
|
* @param entry
|
* The entry whose structural object class is required.
|
* @return The structural object class associated with the provided entry,
|
* or {@code null} if none was found.
|
*/
|
public static ObjectClass getStructuralObjectClass(final Entry entry) {
|
return getStructuralObjectClass(entry, Schema.getDefaultSchema());
|
}
|
|
/**
|
* Returns the structural object class associated with the provided entry,
|
* or {@code null} if none was found. If the entry contains multiple
|
* structural object classes then the first will be returned. This method
|
* will ignore unrecognized object classes.
|
*
|
* @param entry
|
* The entry whose structural object class is required.
|
* @param schema
|
* The schema which should be used for decoding the object class
|
* attribute values.
|
* @return The structural object class associated with the provided entry,
|
* or {@code null} if none was found.
|
*/
|
public static ObjectClass getStructuralObjectClass(final Entry entry, final Schema schema) {
|
ObjectClass structuralObjectClass = null;
|
final Attribute objectClassAttribute = entry.getAttribute(objectClass());
|
|
if (objectClassAttribute == null) {
|
return null;
|
}
|
|
for (final ByteString v : objectClassAttribute) {
|
final String objectClassName = v.toString();
|
final ObjectClass objectClass;
|
try {
|
objectClass = schema.getObjectClass(objectClassName);
|
} catch (final UnknownSchemaElementException e) {
|
// Ignore.
|
continue;
|
}
|
|
if (objectClass.getObjectClassType() == ObjectClassType.STRUCTURAL) {
|
if (structuralObjectClass == null
|
|| objectClass.isDescendantOf(structuralObjectClass)) {
|
structuralObjectClass = objectClass;
|
}
|
}
|
}
|
|
return structuralObjectClass;
|
}
|
|
/**
|
* Applies the provided modification to an entry. This method implements
|
* "permissive" modify semantics, ignoring attempts to add duplicate values
|
* or attempts to remove values which do not exist.
|
*
|
* @param entry
|
* The entry to be modified.
|
* @param change
|
* The modification to be applied to the entry.
|
* @return A reference to the updated entry.
|
* @throws ErrorResultException
|
* If an error occurred while performing the change such as an
|
* attempt to increment a value which is not a number. The entry
|
* will not have been modified.
|
*/
|
public static Entry modifyEntry(final Entry entry, final Modification change)
|
throws ErrorResultException {
|
return modifyEntry(entry, change, null);
|
}
|
|
/**
|
* Applies the provided modification to an entry. This method implements
|
* "permissive" modify semantics, recording attempts to add duplicate values
|
* or attempts to remove values which do not exist in the provided
|
* collection if provided.
|
*
|
* @param entry
|
* The entry to be modified.
|
* @param change
|
* The modification to be applied to the entry.
|
* @param conflictingValues
|
* A collection into which duplicate or missing values will be
|
* added, or {@code null} if conflicting values should not be
|
* saved.
|
* @return A reference to the updated entry.
|
* @throws ErrorResultException
|
* If an error occurred while performing the change such as an
|
* attempt to increment a value which is not a number. The entry
|
* will not have been modified.
|
*/
|
public static Entry modifyEntry(final Entry entry, final Modification change,
|
final Collection<? super ByteString> conflictingValues) throws ErrorResultException {
|
final ModificationType modType = change.getModificationType();
|
if (modType.equals(ModificationType.ADD)) {
|
entry.addAttribute(change.getAttribute(), conflictingValues);
|
} else if (modType.equals(ModificationType.DELETE)) {
|
entry.removeAttribute(change.getAttribute(), conflictingValues);
|
} else if (modType.equals(ModificationType.REPLACE)) {
|
entry.replaceAttribute(change.getAttribute());
|
} else if (modType.equals(ModificationType.INCREMENT)) {
|
incrementAttribute(entry, change.getAttribute());
|
} else {
|
throw newErrorResult(ResultCode.UNWILLING_TO_PERFORM,
|
ERR_ENTRY_UNKNOWN_MODIFICATION_TYPE.get(String.valueOf(modType)).toString());
|
}
|
return entry;
|
}
|
|
/**
|
* Applies the provided modification request to an entry. This method will
|
* utilize "permissive" modify semantics if the request contains the
|
* {@link PermissiveModifyRequestControl}.
|
*
|
* @param entry
|
* The entry to be modified.
|
* @param changes
|
* The modification request to be applied to the entry.
|
* @return A reference to the updated entry.
|
* @throws ErrorResultException
|
* If an error occurred while performing the changes such as an
|
* attempt to add duplicate values, remove values which do not
|
* exist, or increment a value which is not a number. The entry
|
* may have been modified.
|
*/
|
public static Entry modifyEntry(final Entry entry, final ModifyRequest changes)
|
throws ErrorResultException {
|
final boolean isPermissive = changes.containsControl(PermissiveModifyRequestControl.OID);
|
return modifyEntry0(entry, changes.getModifications(), isPermissive);
|
}
|
|
/**
|
* Applies the provided modifications to an entry using "permissive" modify
|
* semantics.
|
*
|
* @param entry
|
* The entry to be modified.
|
* @param changes
|
* The modification request to be applied to the entry.
|
* @return A reference to the updated entry.
|
* @throws ErrorResultException
|
* If an error occurred while performing the changes such as an
|
* attempt to increment a value which is not a number. The entry
|
* may have been modified.
|
*/
|
public static Entry modifyEntryPermissive(final Entry entry,
|
final Collection<Modification> changes) throws ErrorResultException {
|
return modifyEntry0(entry, changes, true);
|
}
|
|
/**
|
* Applies the provided modifications to an entry using "strict" modify
|
* semantics. Attempts to add duplicate values or attempts to remove values
|
* which do not exist will cause the update to fail.
|
*
|
* @param entry
|
* The entry to be modified.
|
* @param changes
|
* The modification request to be applied to the entry.
|
* @return A reference to the updated entry.
|
* @throws ErrorResultException
|
* If an error occurred while performing the changes such as an
|
* attempt to add duplicate values, remove values which do not
|
* exist, or increment a value which is not a number. The entry
|
* may have been modified.
|
*/
|
public static Entry modifyEntryStrict(final Entry entry, final Collection<Modification> changes)
|
throws ErrorResultException {
|
return modifyEntry0(entry, changes, false);
|
}
|
|
/**
|
* Returns a read-only view of {@code entry} and its attributes. Query
|
* operations on the returned entry and its attributes "read-through" to the
|
* underlying entry or attribute, and attempts to modify the returned entry
|
* and its attributes either directly or indirectly via an iterator result
|
* in an {@code UnsupportedOperationException}.
|
*
|
* @param entry
|
* The entry for which a read-only view is to be returned.
|
* @return A read-only view of {@code entry}.
|
* @throws NullPointerException
|
* If {@code entry} was {@code null}.
|
*/
|
public static Entry unmodifiableEntry(final Entry entry) {
|
if (entry instanceof UnmodifiableEntry) {
|
return entry;
|
} else {
|
return new UnmodifiableEntry(entry);
|
}
|
}
|
|
private static void incrementAttribute(final Entry entry, final Attribute change)
|
throws ErrorResultException {
|
// First parse the change.
|
final AttributeDescription deltaAd = change.getAttributeDescription();
|
if (change.size() != 1) {
|
throw newErrorResult(ResultCode.CONSTRAINT_VIOLATION,
|
ERR_ENTRY_INCREMENT_INVALID_VALUE_COUNT.get(deltaAd.toString()).toString());
|
}
|
final long delta;
|
try {
|
delta = change.parse().asLong();
|
} catch (final Exception e) {
|
throw newErrorResult(ResultCode.CONSTRAINT_VIOLATION,
|
ERR_ENTRY_INCREMENT_CANNOT_PARSE_AS_INT.get(deltaAd.toString()).toString());
|
}
|
|
// Now apply the increment to the attribute.
|
final Attribute oldAttribute = entry.getAttribute(deltaAd);
|
if (oldAttribute == null) {
|
throw newErrorResult(ResultCode.NO_SUCH_ATTRIBUTE,
|
ERR_ENTRY_INCREMENT_NO_SUCH_ATTRIBUTE.get(deltaAd.toString()).toString());
|
}
|
|
// Re-use existing attribute description in case it differs in case, etc.
|
final Attribute newAttribute = new LinkedAttribute(oldAttribute.getAttributeDescription());
|
try {
|
for (final Long value : oldAttribute.parse().asSetOfLong()) {
|
newAttribute.add(value + delta);
|
}
|
} catch (final Exception e) {
|
throw newErrorResult(ResultCode.CONSTRAINT_VIOLATION,
|
ERR_ENTRY_INCREMENT_CANNOT_PARSE_AS_INT.get(deltaAd.toString()).toString());
|
}
|
entry.replaceAttribute(newAttribute);
|
}
|
|
private static Entry modifyEntry0(final Entry entry, final Collection<Modification> changes,
|
final boolean isPermissive) throws ErrorResultException {
|
final Collection<ByteString> conflictingValues =
|
isPermissive ? null : new ArrayList<ByteString>(0);
|
for (final Modification change : changes) {
|
modifyEntry(entry, change, conflictingValues);
|
if (!isPermissive && !conflictingValues.isEmpty()) {
|
if (change.getModificationType().equals(ModificationType.ADD)) {
|
// Duplicate values.
|
throw newErrorResult(ResultCode.ATTRIBUTE_OR_VALUE_EXISTS,
|
ERR_ENTRY_DUPLICATE_VALUES.get(
|
change.getAttribute().getAttributeDescriptionAsString())
|
.toString());
|
} else {
|
// Missing values.
|
throw newErrorResult(ResultCode.NO_SUCH_ATTRIBUTE, ERR_ENTRY_NO_SUCH_VALUE.get(
|
change.getAttribute().getAttributeDescriptionAsString()).toString());
|
}
|
}
|
}
|
return entry;
|
}
|
|
// Prevent instantiation.
|
private Entries() {
|
// Nothing to do.
|
}
|
}
|