import java.util.BitSet;
import org.opendaylight.protocol.pcep.PCEPDeserializerException;
-import org.opendaylight.protocol.pcep.impl.Util.BiParsersMap;
-import org.opendaylight.protocol.pcep.subobject.EROLabelSubobject;
+import org.opendaylight.protocol.pcep.spi.EROSubobjectParser;
+import org.opendaylight.protocol.pcep.spi.EROSubobjectSerializer;
+import org.opendaylight.protocol.pcep.spi.LabelHandlerRegistry;
+import org.opendaylight.protocol.pcep.spi.LabelParser;
+import org.opendaylight.protocol.pcep.spi.LabelSerializer;
import org.opendaylight.protocol.util.ByteArray;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.explicit.route.object.Subobjects;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.explicit.route.object.SubobjectsBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.rsvp.rev130820.CLabel;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.rsvp.rev130820.LabelSubobject;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.rsvp.rev130820.basic.explicit.route.subobjects.subobject.type.LabelBuilder;
-public class EROLabelSubobjectParser {
+import com.google.common.base.Preconditions;
- public static final int RES_F_LENGTH = 1;
+public class EROLabelSubobjectParser implements EROSubobjectParser, EROSubobjectSerializer {
- public static final int C_TYPE_F_LENGTH = 1;
+ public static final int TYPE = 3;
- public static final int RES_F_OFFSET = 0;
+ private static final int RES_F_LENGTH = 1;
- public static final int C_TYPE_F_OFFSET = RES_F_OFFSET + RES_F_LENGTH;
+ private static final int C_TYPE_F_LENGTH = 1;
- public static final int HEADER_LENGTH = C_TYPE_F_OFFSET + C_TYPE_F_LENGTH;
+ private static final int RES_F_OFFSET = 0;
- public static final int U_FLAG_OFFSET = 0;
+ private static final int C_TYPE_F_OFFSET = RES_F_OFFSET + RES_F_LENGTH;
- private static class MapOfParsers extends BiParsersMap<Class<? extends EROLabelSubobject>, Integer, EROLabelParser> {
- private final static MapOfParsers instance = new MapOfParsers();
+ private static final int HEADER_LENGTH = C_TYPE_F_OFFSET + C_TYPE_F_LENGTH;
- private MapOfParsers() {
- this.fillInMap();
- }
-
- private void fillInMap() {
+ private static final int U_FLAG_OFFSET = 0;
- }
+ private final LabelHandlerRegistry registry;
- public static MapOfParsers getInstance() {
- return instance;
- }
+ public EROLabelSubobjectParser(final LabelHandlerRegistry labelReg) {
+ this.registry = Preconditions.checkNotNull(labelReg);
}
- public static EROLabelSubobject parse(final byte[] soContentsBytes, final boolean loose) throws PCEPDeserializerException {
- if (soContentsBytes == null || soContentsBytes.length == 0)
+ @Override
+ public Subobjects parseSubobject(final byte[] buffer, final boolean loose) throws PCEPDeserializerException {
+ if (buffer == null || buffer.length == 0)
throw new IllegalArgumentException("Array of bytes is mandatory. Can't be null or empty.");
- if (soContentsBytes.length < HEADER_LENGTH)
- throw new PCEPDeserializerException("Wrong length of array of bytes. Passed: " + soContentsBytes.length + "; Expected: >"
+ if (buffer.length < HEADER_LENGTH)
+ throw new PCEPDeserializerException("Wrong length of array of bytes. Passed: " + buffer.length + "; Expected: >"
+ HEADER_LENGTH + ".");
- final BitSet reserved = ByteArray.bytesToBitSet(Arrays.copyOfRange(soContentsBytes, RES_F_OFFSET, RES_F_LENGTH));
+ final BitSet reserved = ByteArray.bytesToBitSet(Arrays.copyOfRange(buffer, RES_F_OFFSET, RES_F_LENGTH));
- final int c_type = soContentsBytes[C_TYPE_F_OFFSET] & 0xFF;
+ final short c_type = (short) (buffer[C_TYPE_F_OFFSET] & 0xFF);
- final EROLabelParser parser = MapOfParsers.getInstance().getValueFromKeyValue(c_type);
+ final LabelParser parser = this.registry.getLabelParser(c_type);
if (parser == null) {
throw new PCEPDeserializerException("Unknown C-TYPE for ero label subobject. Passed: " + c_type);
}
- return parser.parse(ByteArray.cutBytes(soContentsBytes, HEADER_LENGTH), reserved.get(U_FLAG_OFFSET), loose);
+ final LabelBuilder builder = new LabelBuilder();
+ builder.setUniDirectional(reserved.get(U_FLAG_OFFSET));
+ builder.setLabelType(parser.parseLabel(ByteArray.cutBytes(buffer, HEADER_LENGTH)));
+ return new SubobjectsBuilder().setLoose(loose).setSubobjectType(builder.build()).build();
}
- public static byte[] put(final EROLabelSubobject objToSerialize) {
- final Integer c_type = MapOfParsers.getInstance().getKeyValueFromKey(objToSerialize.getClass());
- final EROLabelParser parser = MapOfParsers.getInstance().getValueFromKeyValue(c_type);
+ @Override
+ public byte[] serializeSubobject(final Subobjects subobject) {
+ Preconditions.checkNotNull(subobject.getSubobjectType(), "Subobject type cannot be empty.");
- if (c_type == null || parser == null)
- throw new IllegalArgumentException("Unknown EROLabelSubobject instance. Passed " + objToSerialize.getClass());
+ final LabelSubobject label = (LabelSubobject) subobject.getSubobjectType();
- final byte[] labelbytes = parser.put(objToSerialize);
+ final LabelSerializer serializer = this.registry.getLabelSerializer((CLabel) label);
+
+ if (serializer == null)
+ throw new IllegalArgumentException("Unknown EROLabelSubobject instance. Passed " + label.getClass());
+
+ final byte[] labelbytes = serializer.serializeSubobject((CLabel) label);
final byte[] retBytes = new byte[labelbytes.length + HEADER_LENGTH];
System.arraycopy(labelbytes, 0, retBytes, HEADER_LENGTH, labelbytes.length);
final BitSet reserved = new BitSet();
- reserved.set(U_FLAG_OFFSET, objToSerialize.isUpStream());
+ reserved.set(U_FLAG_OFFSET, label.isUniDirectional());
System.arraycopy(ByteArray.bitSetToBytes(reserved, RES_F_LENGTH), 0, retBytes, RES_F_OFFSET, RES_F_LENGTH);
- retBytes[C_TYPE_F_OFFSET] = (byte) c_type.intValue();
+ retBytes[C_TYPE_F_OFFSET] = (byte) serializer.getType();
return retBytes;
}
+
+ @Override
+ public int getType() {
+ return TYPE;
+ }
}