Bumped to ietf-stateful-02.
[bgpcep.git] / pcep / spi / src / main / java / org / opendaylight / protocol / pcep / spi / AbstractMessageParser.java
1 /*
2  * Copyright (c) 2013 Cisco Systems, Inc. and others.  All rights reserved.
3  *
4  * This program and the accompanying materials are made available under the
5  * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6  * and is available at http://www.eclipse.org/legal/epl-v10.html
7  */
8 package org.opendaylight.protocol.pcep.spi;
9
10 import java.util.Arrays;
11 import java.util.BitSet;
12 import java.util.List;
13
14 import org.opendaylight.protocol.util.ByteArray;
15 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.message.rev131007.PcerrBuilder;
16 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.Message;
17 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.Object;
18 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.ObjectHeader;
19 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcep.error.object.ErrorObjectBuilder;
20 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.PcerrMessageBuilder;
21 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.ErrorsBuilder;
22 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.error.type.RequestCaseBuilder;
23 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.pcerr.message.pcerr.message.error.type.request._case.RequestBuilder;
24 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;
25 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev131005.rp.object.Rp;
26
27 import com.google.common.base.Preconditions;
28 import com.google.common.collect.Lists;
29 import com.google.common.primitives.UnsignedBytes;
30
31 public abstract class AbstractMessageParser implements MessageParser, MessageSerializer {
32
33         private static final int COMMON_OBJECT_HEADER_LENGTH = 4;
34
35         private static final int OC_F_LENGTH = 1;
36         private static final int OT_FLAGS_MF_LENGTH = 1;
37         private static final int OBJ_LENGTH_F_LENGTH = 2;
38
39         private static final int OT_SF_LENGTH = 4;
40         private static final int FLAGS_SF_LENGTH = 4;
41         /*
42          * offsets of fields inside of multi-field in bits
43          */
44         private static final int OT_SF_OFFSET = 0;
45         private static final int FLAGS_SF_OFFSET = OT_SF_OFFSET + OT_SF_LENGTH;
46         /*
47          * flags offsets inside multi-filed
48          */
49         private static final int P_FLAG_OFFSET = 6;
50         private static final int I_FLAG_OFFSET = 7;
51
52         private final ObjectHandlerRegistry registry;
53
54         protected AbstractMessageParser(final ObjectHandlerRegistry registry) {
55                 this.registry = Preconditions.checkNotNull(registry);
56         }
57
58         protected byte[] serializeObject(final Object object) {
59                 if (object == null) {
60                         throw new IllegalArgumentException("Null object passed.");
61                 }
62                 final ObjectSerializer serializer = this.registry.getObjectSerializer(object);
63                 return serializer.serializeObject(object);
64         }
65
66         private List<Object> parseObjects(final byte[] bytes) throws PCEPDeserializerException {
67                 int offset = 0;
68                 final List<Object> objs = Lists.newArrayList();
69                 while (bytes.length - offset > 0) {
70                         if (bytes.length - offset < COMMON_OBJECT_HEADER_LENGTH) {
71                                 throw new PCEPDeserializerException("Too few bytes in passed array. Passed: " + (bytes.length - offset) + " Expected: >= "
72                                                 + COMMON_OBJECT_HEADER_LENGTH + ".");
73                         }
74
75                         final int objClass = UnsignedBytes.toInt(bytes[offset]);
76
77                         offset += OC_F_LENGTH;
78
79                         final int objType = UnsignedBytes.toInt(ByteArray.copyBitsRange(bytes[offset], OT_SF_OFFSET, OT_SF_LENGTH));
80
81                         final byte[] flagsBytes = { ByteArray.copyBitsRange(bytes[offset], FLAGS_SF_OFFSET, FLAGS_SF_LENGTH) };
82
83                         final BitSet flags = ByteArray.bytesToBitSet(flagsBytes);
84
85                         offset += OT_FLAGS_MF_LENGTH;
86
87                         final int objLength = ByteArray.bytesToInt(ByteArray.subByte(bytes, offset, OBJ_LENGTH_F_LENGTH));
88
89                         if (bytes.length - offset < objLength - COMMON_OBJECT_HEADER_LENGTH) {
90                                 throw new PCEPDeserializerException("Too few bytes in passed array. Passed: " + (bytes.length - offset) + " Expected: >= "
91                                                 + objLength + ".");
92                         }
93
94                         offset += OBJ_LENGTH_F_LENGTH;
95
96                         // copy bytes for deeper parsing
97                         final byte[] bytesToPass = ByteArray.subByte(bytes, offset, objLength - COMMON_OBJECT_HEADER_LENGTH);
98
99                         offset += objLength - COMMON_OBJECT_HEADER_LENGTH;
100
101                         final ObjectParser parser = Preconditions.checkNotNull(this.registry.getObjectParser(objClass, objType));
102                         final ObjectHeader header = new ObjectHeaderImpl(flags.get(P_FLAG_OFFSET), flags.get(I_FLAG_OFFSET));
103
104                         // parseObject is required to return null for P=0 errored objects
105                         final Object o = parser.parseObject(header, bytesToPass);
106                         if (o != null) {
107                                 objs.add(o);
108                         }
109                 }
110
111                 return objs;
112         }
113
114         public static Message createErrorMsg(final PCEPErrors e) {
115                 final PCEPErrorMapping maping = PCEPErrorMapping.getInstance();
116                 return new PcerrBuilder().setPcerrMessage(
117                                 new PcerrMessageBuilder().setErrors(
118                                                 Arrays.asList(new ErrorsBuilder().setErrorObject(
119                                                                 new ErrorObjectBuilder().setType(maping.getFromErrorsEnum(e).type).setValue(
120                                                                                 maping.getFromErrorsEnum(e).value).build()).build())).build()).build();
121         }
122
123         public static Message createErrorMsg(final PCEPErrors e, final Rp rp) {
124                 final PCEPErrorMapping maping = PCEPErrorMapping.getInstance();
125                 return new PcerrBuilder().setPcerrMessage(
126                                 new PcerrMessageBuilder().setErrorType(
127                                                 new RequestCaseBuilder().setRequest(
128                                                                 new RequestBuilder().setRps(Lists.newArrayList(new RpsBuilder().setRp(rp).build())).build()).build()).setErrors(
129                                                 Arrays.asList(new ErrorsBuilder().setErrorObject(
130                                                                 new ErrorObjectBuilder().setType(maping.getFromErrorsEnum(e).type).setValue(
131                                                                                 maping.getFromErrorsEnum(e).value).build()).build())).build()).build();
132         }
133
134         protected abstract Message validate(final List<Object> objects, final List<Message> errors) throws PCEPDeserializerException;
135
136         @Override
137         public final Message parseMessage(final byte[] buffer, final List<Message> errors) throws PCEPDeserializerException {
138                 Preconditions.checkNotNull(buffer, "Buffer may not be null");
139
140                 // Parse objects first
141                 final List<Object> objs = parseObjects(buffer);
142
143                 // Run validation
144                 return validate(objs, errors);
145         }
146 }