blob: d3beaaeec65ea150da480952a550e4a57b6cb262 [file] [log] [blame]
Priyanka Bb2988fa2015-10-09 12:45:36 +05301/*
2 * Copyright 2015 Open Networking Laboratory
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Jonathan Hart317f4762015-11-09 16:05:36 -080017package org.onosproject.bgpio.protocol.linkstate;
Priyanka Bb2988fa2015-10-09 12:45:36 +053018
19import java.util.Iterator;
20import java.util.LinkedList;
Priyanka B02040732015-11-29 11:30:29 +053021import java.util.List;
22import java.util.ListIterator;
Priyanka Bb2988fa2015-10-09 12:45:36 +053023import java.util.Objects;
24
25import org.jboss.netty.buffer.ChannelBuffer;
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053026import org.onosproject.bgpio.exceptions.BgpParseException;
27import org.onosproject.bgpio.types.BgpErrorType;
28import org.onosproject.bgpio.types.BgpValueType;
Priyanka Bb2988fa2015-10-09 12:45:36 +053029import org.onosproject.bgpio.types.IPReachabilityInformationTlv;
30import org.onosproject.bgpio.types.OSPFRouteTypeTlv;
31import org.onosproject.bgpio.types.attr.BgpAttrNodeMultiTopologyId;
32import org.onosproject.bgpio.util.UnSupportedAttribute;
33import org.slf4j.Logger;
34import org.slf4j.LoggerFactory;
35
36import com.google.common.base.MoreObjects;
37
38/**
39 * Provides Implementation of Local node descriptors and prefix descriptors.
40 */
Priyanka B02040732015-11-29 11:30:29 +053041public class BgpPrefixLSIdentifier implements Comparable<Object> {
Priyanka Bb2988fa2015-10-09 12:45:36 +053042
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053043 protected static final Logger log = LoggerFactory.getLogger(BgpPrefixLSIdentifier.class);
Priyanka Bb2988fa2015-10-09 12:45:36 +053044 public static final int TYPE_AND_LEN = 4;
45 private NodeDescriptors localNodeDescriptors;
Priyanka B02040732015-11-29 11:30:29 +053046 private List<BgpValueType> prefixDescriptor;
Priyanka Bb2988fa2015-10-09 12:45:36 +053047
48 /**
49 * Resets parameters.
50 */
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053051 public BgpPrefixLSIdentifier() {
Priyanka Bb2988fa2015-10-09 12:45:36 +053052 this.localNodeDescriptors = null;
53 this.prefixDescriptor = null;
54 }
55
56 /**
57 * Constructor to initialize parameters.
58 *
59 * @param localNodeDescriptors Local node descriptors
60 * @param prefixDescriptor Prefix Descriptors
61 */
Priyanka B02040732015-11-29 11:30:29 +053062 public BgpPrefixLSIdentifier(NodeDescriptors localNodeDescriptors, List<BgpValueType> prefixDescriptor) {
Priyanka Bb2988fa2015-10-09 12:45:36 +053063 this.localNodeDescriptors = localNodeDescriptors;
64 this.prefixDescriptor = prefixDescriptor;
65 }
66
67 /**
68 * Reads the channel buffer and parses Prefix Identifier.
69 *
70 * @param cb ChannelBuffer
71 * @param protocolId protocol ID
72 * @return object of this class
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053073 * @throws BgpParseException while parsing Prefix Identifier
Priyanka Bb2988fa2015-10-09 12:45:36 +053074 */
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053075 public static BgpPrefixLSIdentifier parsePrefixIdendifier(ChannelBuffer cb, byte protocolId)
76 throws BgpParseException {
Priyanka Bb2988fa2015-10-09 12:45:36 +053077 //Parse Local Node descriptor
78 NodeDescriptors localNodeDescriptors = new NodeDescriptors();
79 localNodeDescriptors = parseLocalNodeDescriptors(cb, protocolId);
80
81 //Parse Prefix descriptor
Priyanka B02040732015-11-29 11:30:29 +053082 List<BgpValueType> prefixDescriptor = new LinkedList<>();
Priyanka Bb2988fa2015-10-09 12:45:36 +053083 prefixDescriptor = parsePrefixDescriptors(cb);
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053084 return new BgpPrefixLSIdentifier(localNodeDescriptors, prefixDescriptor);
Priyanka Bb2988fa2015-10-09 12:45:36 +053085 }
86
87 /**
88 * Parse local node descriptors.
89 *
90 * @param cb ChannelBuffer
91 * @param protocolId protocol identifier
92 * @return LocalNodeDescriptors
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053093 * @throws BgpParseException while parsing local node descriptors
Priyanka Bb2988fa2015-10-09 12:45:36 +053094 */
95 public static NodeDescriptors parseLocalNodeDescriptors(ChannelBuffer cb, byte protocolId)
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +053096 throws BgpParseException {
Priyanka B02040732015-11-29 11:30:29 +053097 ChannelBuffer tempBuf = cb.copy();
Priyanka Bb2988fa2015-10-09 12:45:36 +053098 short type = cb.readShort();
99 short length = cb.readShort();
100 if (cb.readableBytes() < length) {
101 //length + 4 implies data contains type, length and value
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530102 throw new BgpParseException(BgpErrorType.UPDATE_MESSAGE_ERROR, BgpErrorType.OPTIONAL_ATTRIBUTE_ERROR,
Priyanka Bb2988fa2015-10-09 12:45:36 +0530103 tempBuf.readBytes(cb.readableBytes() + TYPE_AND_LEN));
104 }
105 NodeDescriptors localNodeDescriptors = new NodeDescriptors();
106 ChannelBuffer tempCb = cb.readBytes(length);
107
108 if (type == NodeDescriptors.LOCAL_NODE_DES_TYPE) {
109 localNodeDescriptors = NodeDescriptors.read(tempCb, length, type, protocolId);
110 } else {
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530111 throw new BgpParseException(BgpErrorType.UPDATE_MESSAGE_ERROR,
112 BgpErrorType.MALFORMED_ATTRIBUTE_LIST, null);
Priyanka Bb2988fa2015-10-09 12:45:36 +0530113 }
114 return localNodeDescriptors;
115 }
116
117 /**
118 * Parse list of prefix descriptors.
119 *
120 * @param cb ChannelBuffer
121 * @return list of prefix descriptors
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530122 * @throws BgpParseException while parsing list of prefix descriptors
Priyanka Bb2988fa2015-10-09 12:45:36 +0530123 */
Priyanka B02040732015-11-29 11:30:29 +0530124 public static List<BgpValueType> parsePrefixDescriptors(ChannelBuffer cb) throws BgpParseException {
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530125 LinkedList<BgpValueType> prefixDescriptor = new LinkedList<>();
126 BgpValueType tlv = null;
Priyanka Bb2988fa2015-10-09 12:45:36 +0530127 boolean isIpReachInfo = false;
128 ChannelBuffer tempCb;
129 int count = 0;
130
131 while (cb.readableBytes() > 0) {
Priyanka B02040732015-11-29 11:30:29 +0530132 ChannelBuffer tempBuf = cb.copy();
Priyanka Bb2988fa2015-10-09 12:45:36 +0530133 short type = cb.readShort();
134 short length = cb.readShort();
135 if (cb.readableBytes() < length) {
136 //length + 4 implies data contains type, length and value
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530137 throw new BgpParseException(BgpErrorType.UPDATE_MESSAGE_ERROR, BgpErrorType.OPTIONAL_ATTRIBUTE_ERROR,
Priyanka Bb2988fa2015-10-09 12:45:36 +0530138 tempBuf.readBytes(cb.readableBytes() + TYPE_AND_LEN));
139 }
140 tempCb = cb.readBytes(length);
141 switch (type) {
142 case OSPFRouteTypeTlv.TYPE:
143 tlv = OSPFRouteTypeTlv.read(tempCb);
144 break;
145 case IPReachabilityInformationTlv.TYPE:
146 tlv = IPReachabilityInformationTlv.read(tempCb, length);
147 isIpReachInfo = true;
148 break;
149 case BgpAttrNodeMultiTopologyId.ATTRNODE_MULTITOPOLOGY:
150 tlv = BgpAttrNodeMultiTopologyId.read(tempCb);
151 count = count + 1;
152 if (count > 1) {
153 //length + 4 implies data contains type, length and value
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530154 throw new BgpParseException(BgpErrorType.UPDATE_MESSAGE_ERROR,
155 BgpErrorType.OPTIONAL_ATTRIBUTE_ERROR, tempBuf.readBytes(length + TYPE_AND_LEN));
Priyanka Bb2988fa2015-10-09 12:45:36 +0530156 }
157 break;
158 default:
159 UnSupportedAttribute.skipBytes(tempCb, length);
160 }
161 prefixDescriptor.add(tlv);
162 }
163
164 if (!isIpReachInfo) {
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530165 throw new BgpParseException(BgpErrorType.UPDATE_MESSAGE_ERROR, BgpErrorType.OPTIONAL_ATTRIBUTE_ERROR,
Priyanka Bb2988fa2015-10-09 12:45:36 +0530166 null);
167 }
168 return prefixDescriptor;
169 }
170
171 /**
172 * Returns local node descriptors.
173 *
174 * @return local node descriptors
175 */
176 public NodeDescriptors getLocalNodeDescriptors() {
177 return this.localNodeDescriptors;
178 }
179
180 /**
181 * Returns Prefix descriptors.
182 *
183 * @return Prefix descriptors
184 */
Priyanka B02040732015-11-29 11:30:29 +0530185 public List<BgpValueType> getPrefixdescriptor() {
Priyanka Bb2988fa2015-10-09 12:45:36 +0530186 return this.prefixDescriptor;
187 }
188
189 @Override
190 public int hashCode() {
191 return Objects.hash(prefixDescriptor.hashCode(), localNodeDescriptors);
192 }
193
194 @Override
195 public boolean equals(Object obj) {
196 if (this == obj) {
197 return true;
198 }
199
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530200 if (obj instanceof BgpPrefixLSIdentifier) {
Priyanka Bb2988fa2015-10-09 12:45:36 +0530201 int countObjSubTlv = 0;
202 int countOtherSubTlv = 0;
203 boolean isCommonSubTlv = true;
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530204 BgpPrefixLSIdentifier other = (BgpPrefixLSIdentifier) obj;
Priyanka Bb2988fa2015-10-09 12:45:36 +0530205
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530206 Iterator<BgpValueType> objListIterator = other.prefixDescriptor.iterator();
Priyanka Bb2988fa2015-10-09 12:45:36 +0530207 countOtherSubTlv = other.prefixDescriptor.size();
208 countObjSubTlv = prefixDescriptor.size();
209 if (countObjSubTlv != countOtherSubTlv) {
210 return false;
211 } else {
212 while (objListIterator.hasNext() && isCommonSubTlv) {
Shashikanth VH5dd8dbe2015-11-26 13:22:18 +0530213 BgpValueType subTlv = objListIterator.next();
Priyanka B02040732015-11-29 11:30:29 +0530214 if (prefixDescriptor.contains(subTlv) && other.prefixDescriptor.contains(subTlv)) {
215 isCommonSubTlv = Objects.equals(prefixDescriptor.get(prefixDescriptor.indexOf(subTlv)),
216 other.prefixDescriptor.get(other.prefixDescriptor.indexOf(subTlv)));
217 } else {
218 isCommonSubTlv = false;
219 }
Priyanka Bb2988fa2015-10-09 12:45:36 +0530220 }
221 return isCommonSubTlv && Objects.equals(this.localNodeDescriptors, other.localNodeDescriptors);
222 }
223 }
224 return false;
225 }
226
227 @Override
228 public String toString() {
229 return MoreObjects.toStringHelper(getClass())
230 .add("localNodeDescriptors", localNodeDescriptors)
231 .add("prefixDescriptor", prefixDescriptor)
232 .toString();
233 }
Priyanka B02040732015-11-29 11:30:29 +0530234
235 @Override
236 public int compareTo(Object o) {
237 if (this.equals(o)) {
238 return 0;
239 }
240 int result = this.localNodeDescriptors.compareTo(((BgpPrefixLSIdentifier) o).localNodeDescriptors);
241 if (result != 0) {
242 return result;
243 } else {
244 int countOtherSubTlv = ((BgpPrefixLSIdentifier) o).prefixDescriptor.size();
245 int countObjSubTlv = prefixDescriptor.size();
246 if (countOtherSubTlv != countObjSubTlv) {
247 if (countOtherSubTlv > countObjSubTlv) {
248 return 1;
249 } else {
250 return -1;
251 }
252 }
253
254 ListIterator<BgpValueType> listIterator = prefixDescriptor.listIterator();
255 ListIterator<BgpValueType> listIteratorOther = ((BgpPrefixLSIdentifier) o).prefixDescriptor.listIterator();
256 while (listIterator.hasNext()) {
257 BgpValueType tlv = listIterator.next();
258 BgpValueType tlv1 = listIteratorOther.next();
259 if (prefixDescriptor.contains(tlv) && ((BgpPrefixLSIdentifier) o).prefixDescriptor.contains(tlv1)) {
260 int res = prefixDescriptor.get(prefixDescriptor.indexOf(tlv)).compareTo(
261 ((BgpPrefixLSIdentifier) o).prefixDescriptor
262 .get(((BgpPrefixLSIdentifier) o).prefixDescriptor.indexOf(tlv1)));
263 if (res != 0) {
264 return res;
265 }
266 }
267 }
268 }
269 return 0;
270 }
Jonathan Hart317f4762015-11-09 16:05:36 -0800271}