Merge "BUG-632: add missing module"
[bgpcep.git] / pcep / spi / src / main / java / org / opendaylight / protocol / pcep / spi / AbstractMessageParser.java
index bb431c72498ecc79be859df24f3357c2c11409ee..60914d7d759940d85fbc7d2a2b134e4224884f72 100644 (file)
@@ -1,16 +1,28 @@
+/*
+ * Copyright (c) 2013 Cisco Systems, Inc. and others.  All rights reserved.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License v1.0 which accompanies this distribution,
+ * and is available at http://www.eclipse.org/legal/epl-v10.html
+ */
 package org.opendaylight.protocol.pcep.spi;
 
 import java.util.Arrays;
 import java.util.BitSet;
 import java.util.List;
 
-import org.opendaylight.protocol.pcep.PCEPDeserializerException;
-import org.opendaylight.protocol.pcep.PCEPDocumentedException;
-import org.opendaylight.protocol.pcep.PCEPErrors;
-import org.opendaylight.protocol.pcep.UnknownObject;
 import org.opendaylight.protocol.util.ByteArray;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.PcerrBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.Message;
 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.Object;
 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.ObjectHeader;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcep.error.object.ErrorObjectBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.PcerrMessageBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.ErrorsBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.error.type.RequestCaseBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.error.type.request._case.RequestBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.error.type.request._case.request.RpsBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.rp.object.Rp;
 
 import com.google.common.base.Preconditions;
 import com.google.common.collect.Lists;
@@ -18,70 +30,39 @@ import com.google.common.primitives.UnsignedBytes;
 
 public abstract class AbstractMessageParser implements MessageParser, MessageSerializer {
 
-       private final static int COMMON_OBJECT_HEADER_LENGTH = 4;
+       private static final int COMMON_OBJECT_HEADER_LENGTH = 4;
 
-       private final static int OC_F_LENGTH = 1;
-       private final static int OT_FLAGS_MF_LENGTH = 1; // multi-field
-       private final static int OBJ_LENGTH_F_LENGTH = 2;
-
-       private final static int OC_F_OFFSET = 0;
-       private final static int OT_FLAGS_MF_OFFSET = OC_F_OFFSET + OC_F_LENGTH;
-       private final static int OBJ_LENGTH_F_OFFSET = OT_FLAGS_MF_OFFSET + OT_FLAGS_MF_LENGTH;
-
-       private final static int OT_SF_LENGTH = 4;
-       private final static int FLAGS_SF_LENGTH = 4;
+       private static final int OC_F_LENGTH = 1;
+       private static final int OT_FLAGS_MF_LENGTH = 1;
+       private static final int OBJ_LENGTH_F_LENGTH = 2;
 
+       private static final int OT_SF_LENGTH = 4;
+       private static final int FLAGS_SF_LENGTH = 4;
        /*
         * offsets of fields inside of multi-field in bits
         */
-       private final static int OT_SF_OFFSET = 0;
-       private final static int FLAGS_SF_OFFSET = OT_SF_OFFSET + OT_SF_LENGTH;
-
+       private static final int OT_SF_OFFSET = 0;
+       private static final int FLAGS_SF_OFFSET = OT_SF_OFFSET + OT_SF_LENGTH;
        /*
         * flags offsets inside multi-filed
         */
-       private final static int P_FLAG_OFFSET = 6;
-       private final static int I_FLAG_OFFSET = 7;
+       private static final int P_FLAG_OFFSET = 6;
+       private static final int I_FLAG_OFFSET = 7;
 
-       private final ObjectHandlerRegistry registry;
+       private final ObjectRegistry registry;
 
-       protected AbstractMessageParser(final ObjectHandlerRegistry registry) {
+       protected AbstractMessageParser(final ObjectRegistry registry) {
                this.registry = Preconditions.checkNotNull(registry);
        }
 
        protected byte[] serializeObject(final Object object) {
                if (object == null) {
-                       throw new IllegalArgumentException("Null object passed.");
-               }
-
-               final ObjectSerializer serializer = this.registry.getObjectSerializer(object);
-
-               final byte[] valueBytes = serializer.serializeObject(object);
-
-               final byte[] retBytes = new byte[COMMON_OBJECT_HEADER_LENGTH + valueBytes.length];
-
-               // objClass
-               retBytes[OC_F_OFFSET] = (byte) serializer.getObjectClass();
-
-               // objType_flags multi-field
-               retBytes[OT_FLAGS_MF_OFFSET] = (byte) (serializer.getObjectType() << (Byte.SIZE - OT_SF_LENGTH));
-               if (object.isProcessingRule()) {
-                       retBytes[OT_FLAGS_MF_OFFSET] |= 1 << Byte.SIZE - (P_FLAG_OFFSET) - 1;
+                       return new byte[] {};
                }
-               if (object.isIgnore()) {
-                       retBytes[OT_FLAGS_MF_OFFSET] |= 1 << Byte.SIZE - (I_FLAG_OFFSET) - 1;
-               }
-
-               // objLength
-               System.arraycopy(ByteArray.intToBytes(valueBytes.length), Integer.SIZE / Byte.SIZE - OBJ_LENGTH_F_LENGTH, retBytes,
-                               OBJ_LENGTH_F_OFFSET, OBJ_LENGTH_F_LENGTH);
-
-               System.arraycopy(valueBytes, 0, retBytes, COMMON_OBJECT_HEADER_LENGTH, valueBytes.length);
-
-               return retBytes;
+               return this.registry.serializeObject(object);
        }
 
-       protected List<Object> parseObjects(final byte[] bytes) throws PCEPDeserializerException, PCEPDocumentedException {
+       private List<Object> parseObjects(final byte[] bytes) throws PCEPDeserializerException {
                int offset = 0;
                final List<Object> objs = Lists.newArrayList();
                while (bytes.length - offset > 0) {
@@ -90,43 +71,74 @@ public abstract class AbstractMessageParser implements MessageParser, MessageSer
                                                + COMMON_OBJECT_HEADER_LENGTH + ".");
                        }
 
-                       final int objClass = ByteArray.bytesToInt(Arrays.copyOfRange(bytes, OC_F_OFFSET, OC_F_OFFSET + OC_F_LENGTH));
+                       final int objClass = UnsignedBytes.toInt(bytes[offset]);
 
-                       final int objType = UnsignedBytes.toInt(ByteArray.copyBitsRange(bytes[OT_FLAGS_MF_OFFSET], OT_SF_OFFSET, OT_SF_LENGTH));
+                       offset += OC_F_LENGTH;
 
-                       final int objLength = ByteArray.bytesToInt(Arrays.copyOfRange(bytes, OBJ_LENGTH_F_OFFSET, OBJ_LENGTH_F_OFFSET
-                                       + OBJ_LENGTH_F_LENGTH));
+                       final int objType = UnsignedBytes.toInt(ByteArray.copyBitsRange(bytes[offset], OT_SF_OFFSET, OT_SF_LENGTH));
 
-                       final byte[] flagsBytes = { ByteArray.copyBitsRange(bytes[OT_FLAGS_MF_OFFSET], FLAGS_SF_OFFSET, FLAGS_SF_LENGTH) };
+                       final byte[] flagsBytes = { ByteArray.copyBitsRange(bytes[offset], FLAGS_SF_OFFSET, FLAGS_SF_LENGTH) };
 
                        final BitSet flags = ByteArray.bytesToBitSet(flagsBytes);
 
-                       if (bytes.length - offset < objLength) {
+                       offset += OT_FLAGS_MF_LENGTH;
+
+                       final int objLength = ByteArray.bytesToInt(ByteArray.subByte(bytes, offset, OBJ_LENGTH_F_LENGTH));
+
+                       if (bytes.length - offset < objLength - COMMON_OBJECT_HEADER_LENGTH) {
                                throw new PCEPDeserializerException("Too few bytes in passed array. Passed: " + (bytes.length - offset) + " Expected: >= "
                                                + objLength + ".");
                        }
 
-                       // copy bytes for deeper parsing
-                       final byte[] bytesToPass = ByteArray.subByte(bytes, offset + COMMON_OBJECT_HEADER_LENGTH, objLength
-                                       - COMMON_OBJECT_HEADER_LENGTH);
+                       offset += OBJ_LENGTH_F_LENGTH;
 
-                       offset += objLength;
+                       // copy bytes for deeper parsing
+                       final byte[] bytesToPass = ByteArray.subByte(bytes, offset, objLength - COMMON_OBJECT_HEADER_LENGTH);
 
-                       final ObjectParser parser = this.registry.getObjectParser(objClass, objType);
+                       offset += objLength - COMMON_OBJECT_HEADER_LENGTH;
 
                        final ObjectHeader header = new ObjectHeaderImpl(flags.get(P_FLAG_OFFSET), flags.get(I_FLAG_OFFSET));
 
-                       try {
-                               objs.add(parser.parseObject(header, bytesToPass));
-                       } catch (final PCEPDocumentedException e) {
-                               if (e.getError() == PCEPErrors.UNRECOGNIZED_OBJ_CLASS | e.getError() == PCEPErrors.UNRECOGNIZED_OBJ_TYPE
-                                               | e.getError() == PCEPErrors.NOT_SUPPORTED_OBJ_CLASS | e.getError() == PCEPErrors.NOT_SUPPORTED_OBJ_TYPE) {
-                                       objs.add(new UnknownObject(e.getError()));
-                               } else {
-                                       throw e;
-                               }
+                       // parseObject is required to return null for P=0 errored objects
+                       final Object o = this.registry.parseObject(objClass, objType, header, bytesToPass);
+                       if (o != null) {
+                               objs.add(o);
                        }
                }
+
                return objs;
        }
+
+       public static Message createErrorMsg(final PCEPErrors e) {
+               final PCEPErrorMapping maping = PCEPErrorMapping.getInstance();
+               return new PcerrBuilder().setPcerrMessage(
+                               new PcerrMessageBuilder().setErrors(
+                                               Arrays.asList(new ErrorsBuilder().setErrorObject(
+                                                               new ErrorObjectBuilder().setType(maping.getFromErrorsEnum(e).getType()).setValue(
+                                                                               maping.getFromErrorsEnum(e).getValue()).build()).build())).build()).build();
+       }
+
+       public static Message createErrorMsg(final PCEPErrors e, final Rp rp) {
+               final PCEPErrorMapping maping = PCEPErrorMapping.getInstance();
+               return new PcerrBuilder().setPcerrMessage(
+                               new PcerrMessageBuilder().setErrorType(
+                                               new RequestCaseBuilder().setRequest(
+                                                               new RequestBuilder().setRps(Lists.newArrayList(new RpsBuilder().setRp(rp).build())).build()).build()).setErrors(
+                                                                               Arrays.asList(new ErrorsBuilder().setErrorObject(
+                                                                                               new ErrorObjectBuilder().setType(maping.getFromErrorsEnum(e).getType()).setValue(
+                                                                                                               maping.getFromErrorsEnum(e).getValue()).build()).build())).build()).build();
+       }
+
+       protected abstract Message validate(final List<Object> objects, final List<Message> errors) throws PCEPDeserializerException;
+
+       @Override
+       public final Message parseMessage(final byte[] buffer, final List<Message> errors) throws PCEPDeserializerException {
+               Preconditions.checkNotNull(buffer, "Buffer may not be null");
+
+               // Parse objects first
+               final List<Object> objs = parseObjects(buffer);
+
+               // Run validation
+               return validate(objs, errors);
+       }
 }