/* * 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 2006-2009 Sun Microsystems, Inc. */ package org.opends.server.replication.protocol; import org.opends.server.core.AddOperationBasis; import org.opends.server.core.DirectoryServer; import java.io.UnsupportedEncodingException; import java.util.ArrayList; import java.util.Collection; import java.util.List; import java.util.Map; import java.util.zip.DataFormatException; import org.opends.server.protocols.internal.InternalClientConnection; import org.opends.server.protocols.ldap.LDAPAttribute; import org.opends.server.protocols.asn1.ASN1Writer; import org.opends.server.protocols.asn1.ASN1; import org.opends.server.protocols.asn1.ASN1Exception; import org.opends.server.protocols.asn1.ASN1Reader; import org.opends.server.replication.common.ChangeNumber; import org.opends.server.types.*; import org.opends.server.types.operation.PostOperationAddOperation; import static org.opends.server.replication.protocol.OperationContext.*; import static org.opends.server.util.StaticUtils.toLowerCase; /** * This class is used to exchange Add operation between LDAP servers * and replication servers. */ public class AddMsg extends LDAPUpdateMsg { private byte[] encodedAttributes; private String parentUniqueId; /** * Creates a new AddMessage. * @param op the operation to use when creating the message */ public AddMsg(PostOperationAddOperation op) { super((AddContext) op.getAttachment(SYNCHROCONTEXT), op.getRawEntryDN().toString()); AddContext ctx = (AddContext) op.getAttachment(SYNCHROCONTEXT); this.parentUniqueId = ctx.getParentUid(); this.encodedAttributes = encodeAttributes(op.getObjectClasses(), op.getUserAttributes(), op.getOperationalAttributes()); } private byte[] encodeAttributes( Map objectClasses, Map> userAttributes, Map> operationalAttributes) { ByteStringBuilder byteBuilder = new ByteStringBuilder(); ASN1Writer writer = ASN1.getWriter(byteBuilder); try { // Encode the object classes (SET OF LDAPString). AttributeBuilder builder = new AttributeBuilder( DirectoryServer.getObjectClassAttributeType()); builder.setInitialCapacity(objectClasses.size()); for (String s : objectClasses.values()) { builder.add(AttributeValues.create(ByteString.valueOf(s), ByteString.valueOf(toLowerCase(s)))); } Attribute attr = builder.toAttribute(); new LDAPAttribute(attr).write(writer); // Encode the user attributes (AttributeList). for (List list : userAttributes.values()) { for (Attribute a : list) { if (!a.isVirtual()) new LDAPAttribute(a).write(writer); } } // Encode the operational attributes (AttributeList). for (List list : operationalAttributes.values()) { for (Attribute a : list) { if (!a.isVirtual()) new LDAPAttribute(a).write(writer); } } } catch(Exception e) { // TODO: DO something } // Encode the sequence. return byteBuilder.toByteArray(); } /** * Creates a new AddMessage. * * @param cn ChangeNumber of the add. * @param dn DN of the added entry. * @param uniqueId The Unique identifier of the added entry. * @param parentId The unique Id of the parent of the added * entry. * @param objectClasses objectclass of the added entry. * @param userAttributes user attributes of the added entry. * @param operationalAttributes operational attributes of the added entry. */ public AddMsg(ChangeNumber cn, String dn, String uniqueId, String parentId, Map objectClasses, Map> userAttributes, Map> operationalAttributes) { super (cn, uniqueId, dn); this.parentUniqueId = parentId; encodedAttributes = encodeAttributes(objectClasses, userAttributes, operationalAttributes); } /** * Creates a new AddMessage. * * @param cn ChangeNumber of the add. * @param dn DN of the added entry. * @param uniqueId The Unique identifier of the added entry. * @param parentId The unique Id of the parent of the added entry. * @param objectClass objectclass of the added entry. * @param userAttributes user attributes of the added entry. * @param operationalAttributes operational attributes of the added entry. */ public AddMsg(ChangeNumber cn, String dn, String uniqueId, String parentId, Attribute objectClass, Collection userAttributes, Collection operationalAttributes) { super (cn, uniqueId, dn); this.parentUniqueId = parentId; ByteStringBuilder byteBuilder = new ByteStringBuilder(); ASN1Writer writer = ASN1.getWriter(byteBuilder); try { new LDAPAttribute(objectClass).write(writer); for (Attribute a : userAttributes) new LDAPAttribute(a).write(writer); if (operationalAttributes != null) for (Attribute a : operationalAttributes) new LDAPAttribute(a).write(writer); } catch(Exception e) { // Do something } encodedAttributes = byteBuilder.toByteArray(); } /** * Creates a new Add message from a byte[]. * * @param in The byte[] from which the operation must be read. * @throws DataFormatException The input byte[] is not a valid AddMsg * @throws UnsupportedEncodingException If UTF8 is not supported by the jvm */ public AddMsg(byte[] in) throws DataFormatException, UnsupportedEncodingException { byte[] allowedPduTypes = new byte[2]; allowedPduTypes[0] = MSG_TYPE_ADD; allowedPduTypes[1] = MSG_TYPE_ADD_V1; int pos = decodeHeader(allowedPduTypes, in); // read the parent unique Id int length = getNextLength(in, pos); if (length != 0) { parentUniqueId = new String(in, pos, length, "UTF-8"); pos += length + 1; } else { parentUniqueId = null; pos += 1; } // Read the attributes : all the remaining bytes encodedAttributes = new byte[in.length-pos]; int i =0; while (pos attr = new ArrayList(); while(asn1Reader.hasNextElement()) { attr.add(LDAPAttribute.decode(asn1Reader)); } AddOperationBasis add = new AddOperationBasis(connection, InternalClientConnection.nextOperationID(), InternalClientConnection.nextMessageID(), null, ByteString.valueOf(newDn), attr); AddContext ctx = new AddContext(getChangeNumber(), getUniqueId(), parentUniqueId); add.setAttachment(SYNCHROCONTEXT, ctx); return add; } /** * {@inheritDoc} */ @Override public byte[] getBytes() throws UnsupportedEncodingException { if (bytes == null) { int length = encodedAttributes.length; byte[] byteParentId = null; if (parentUniqueId != null) { byteParentId = parentUniqueId.getBytes("UTF-8"); length += byteParentId.length + 1; } else { length += 1; } /* encode the header in a byte[] large enough to also contain the mods */ byte [] resultByteArray = encodeHeader(MSG_TYPE_ADD, length); int pos = resultByteArray.length - length; if (byteParentId != null) pos = addByteArray(byteParentId, resultByteArray, pos); else resultByteArray[pos++] = 0; /* put the attributes */ for (int i=0; i