*/
package org.opendaylight.protocol.bgp.parser.impl.message;
+import static java.util.Objects.requireNonNull;
import com.google.common.base.Preconditions;
import io.netty.buffer.ByteBuf;
-import io.netty.buffer.ByteBufUtil;
import io.netty.buffer.Unpooled;
+import java.util.ArrayList;
import java.util.List;
import org.opendaylight.protocol.bgp.parser.BGPDocumentedException;
import org.opendaylight.protocol.bgp.parser.BGPError;
import org.opendaylight.protocol.bgp.parser.BGPParsingException;
+import org.opendaylight.protocol.bgp.parser.BgpTableTypeImpl;
+import org.opendaylight.protocol.bgp.parser.impl.message.update.AsPathAttributeParser;
+import org.opendaylight.protocol.bgp.parser.impl.message.update.NextHopAttributeParser;
+import org.opendaylight.protocol.bgp.parser.impl.message.update.OriginAttributeParser;
import org.opendaylight.protocol.bgp.parser.spi.AttributeRegistry;
import org.opendaylight.protocol.bgp.parser.spi.MessageParser;
import org.opendaylight.protocol.bgp.parser.spi.MessageSerializer;
import org.opendaylight.protocol.bgp.parser.spi.MessageUtil;
-import org.opendaylight.protocol.util.ByteArray;
+import org.opendaylight.protocol.bgp.parser.spi.MultiPathSupportUtil;
+import org.opendaylight.protocol.bgp.parser.spi.PathIdUtil;
+import org.opendaylight.protocol.bgp.parser.spi.PeerSpecificParserConstraint;
+import org.opendaylight.protocol.util.ByteBufWriteUtil;
import org.opendaylight.protocol.util.Ipv4Util;
-import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev100924.Ipv4Prefix;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.Update;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.UpdateBuilder;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.update.Nlri;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.update.NlriBuilder;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.update.PathAttributes;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.update.WithdrawnRoutes;
-import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.update.WithdrawnRoutesBuilder;
+import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4Prefix;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.PathId;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.Update;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.UpdateBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.path.attributes.Attributes;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.update.message.Nlri;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.update.message.NlriBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.update.message.WithdrawnRoutes;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev180329.update.message.WithdrawnRoutesBuilder;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev180329.Ipv4AddressFamily;
+import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev180329.UnicastSubsequentAddressFamily;
import org.opendaylight.yangtools.yang.binding.Notification;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
*
* @see <a href="http://tools.ietf.org/html/rfc4271#section-4.3">BGP-4 Update Message Format</a>
*/
-public class BGPUpdateMessageParser implements MessageParser, MessageSerializer {
- public static final int TYPE = 2;
+public final class BGPUpdateMessageParser implements MessageParser, MessageSerializer {
private static final Logger LOG = LoggerFactory.getLogger(BGPUpdateMessageParser.class);
- /**
- * Size of the withdrawn_routes_length field, in bytes.
- */
- public static final int WITHDRAWN_ROUTES_LENGTH_SIZE = 2;
- /**
- * Size of the total_path_attr_length field, in bytes.
- */
- public static final int TOTAL_PATH_ATTR_LENGTH_SIZE = 2;
+ public static final int TYPE = 2;
+
+ private static final int WITHDRAWN_ROUTES_LENGTH_SIZE = 2;
+
+ private static final int TOTAL_PATH_ATTR_LENGTH_SIZE = 2;
private final AttributeRegistry reg;
- // Constructors -------------------------------------------------------
public BGPUpdateMessageParser(final AttributeRegistry reg) {
- this.reg = Preconditions.checkNotNull(reg);
+ this.reg = requireNonNull(reg);
}
- // Getters & setters --------------------------------------------------
-
@Override
public Update parseMessageBody(final ByteBuf buffer, final int messageLength) throws BGPDocumentedException {
- Preconditions.checkArgument(buffer != null && buffer.readableBytes() != 0, "Byte array cannot be null or empty.");
- LOG.trace("Started parsing of update message: {}", ByteBufUtil.hexDump(buffer));
+ return parseMessageBody(buffer, messageLength, null);
+ }
- final int withdrawnRoutesLength = buffer.readUnsignedShort();
- final UpdateBuilder eventBuilder = new UpdateBuilder();
+ @Override
+ public void serializeMessage(final Notification message, final ByteBuf bytes) {
+ Preconditions.checkArgument(message instanceof Update, "Message needs to be of type Update");
+ final Update update = (Update) message;
+
+ final ByteBuf messageBody = Unpooled.buffer();
+ final List<WithdrawnRoutes> withdrawnRoutes = update.getWithdrawnRoutes();
+ if (withdrawnRoutes != null) {
+ final ByteBuf withdrawnRoutesBuf = Unpooled.buffer();
+ withdrawnRoutes.forEach(withdrawnRoute -> writePathIdPrefix(withdrawnRoutesBuf, withdrawnRoute.getPathId(),
+ withdrawnRoute.getPrefix()));
+ messageBody.writeShort(withdrawnRoutesBuf.writerIndex());
+ messageBody.writeBytes(withdrawnRoutesBuf);
+ } else {
+ messageBody.writeZero(WITHDRAWN_ROUTES_LENGTH_SIZE);
+ }
+ if (update.getAttributes() != null) {
+ final ByteBuf pathAttributesBuf = Unpooled.buffer();
+ this.reg.serializeAttribute(update.getAttributes(), pathAttributesBuf);
+ messageBody.writeShort(pathAttributesBuf.writerIndex());
+ messageBody.writeBytes(pathAttributesBuf);
+ } else {
+ messageBody.writeZero(TOTAL_PATH_ATTR_LENGTH_SIZE);
+ }
+ final List<Nlri> nlris = update.getNlri();
+ if (nlris != null) {
+ nlris.forEach(nlri -> writePathIdPrefix(messageBody, nlri.getPathId(), nlri.getPrefix()));
+ }
+ MessageUtil.formatMessage(TYPE, messageBody, bytes);
+ }
+
+ private void writePathIdPrefix(final ByteBuf byteBuf, final PathId pathId, final Ipv4Prefix ipv4Prefix) {
+ PathIdUtil.writePathId(pathId, byteBuf);
+ ByteBufWriteUtil.writeMinimalPrefix(ipv4Prefix, byteBuf);
+ }
+
+ /**
+ * Parse Update message from buffer.
+ * Calls {@link #checkMandatoryAttributesPresence(Update)} to check for presence of mandatory attributes.
+ *
+ * @param buffer Encoded BGP message in ByteBuf
+ * @param messageLength Length of the BGP message
+ * @param constraint Peer specific constraints
+ * @return Parsed Update message body
+ */
+ @Override
+ public Update parseMessageBody(final ByteBuf buffer, final int messageLength,
+ final PeerSpecificParserConstraint constraint) throws BGPDocumentedException {
+ Preconditions.checkArgument(buffer != null && buffer.isReadable(),
+ "Buffer cannot be null or empty.");
+ final UpdateBuilder builder = new UpdateBuilder();
+ final boolean isMultiPathSupported = MultiPathSupportUtil.isTableTypeSupported(constraint,
+ new BgpTableTypeImpl(Ipv4AddressFamily.class, UnicastSubsequentAddressFamily.class));
+
+ final int withdrawnRoutesLength = buffer.readUnsignedShort();
if (withdrawnRoutesLength > 0) {
- final List<Ipv4Prefix> withdrawnRoutes = Ipv4Util.prefixListForBytes(ByteArray.readBytes(buffer, withdrawnRoutesLength));
- eventBuilder.setWithdrawnRoutes(new WithdrawnRoutesBuilder().setWithdrawnRoutes(withdrawnRoutes).build());
+ final List<WithdrawnRoutes> withdrawnRoutes = new ArrayList<>();
+ final ByteBuf withdrawnRoutesBuffer = buffer.readBytes(withdrawnRoutesLength);
+ while (withdrawnRoutesBuffer.isReadable()) {
+ final WithdrawnRoutesBuilder withdrawnRoutesBuilder = new WithdrawnRoutesBuilder();
+ if (isMultiPathSupported) {
+ withdrawnRoutesBuilder.setPathId(PathIdUtil.readPathId(withdrawnRoutesBuffer));
+ }
+ withdrawnRoutesBuilder.setPrefix(Ipv4Util.prefixForByteBuf(withdrawnRoutesBuffer));
+ withdrawnRoutes.add(withdrawnRoutesBuilder.build());
+ }
+ builder.setWithdrawnRoutes(withdrawnRoutes);
}
final int totalPathAttrLength = buffer.readUnsignedShort();
if (withdrawnRoutesLength == 0 && totalPathAttrLength == 0) {
- return eventBuilder.build();
+ return builder.build();
}
if (totalPathAttrLength > 0) {
try {
- final PathAttributes pathAttributes = this.reg.parseAttributes(buffer.slice(buffer.readerIndex(), totalPathAttrLength));
- buffer.skipBytes(totalPathAttrLength);
- eventBuilder.setPathAttributes(pathAttributes);
- } catch (final BGPParsingException | RuntimeException e) {
+ final Attributes attributes
+ = this.reg.parseAttributes(buffer.readSlice(totalPathAttrLength), constraint);
+ builder.setAttributes(attributes);
+ } catch (final RuntimeException | BGPParsingException e) {
// Catch everything else and turn it into a BGPDocumentedException
- LOG.warn("Could not parse BGP attributes", e);
throw new BGPDocumentedException("Could not parse BGP attributes.", BGPError.MALFORMED_ATTR_LIST, e);
}
}
- final List<Ipv4Prefix> nlri = Ipv4Util.prefixListForBytes(ByteArray.readAllBytes(buffer));
- if (nlri != null && !nlri.isEmpty()) {
- eventBuilder.setNlri(new NlriBuilder().setNlri(nlri).build());
+ final List<Nlri> nlri = new ArrayList<>();
+ while (buffer.isReadable()) {
+ final NlriBuilder nlriBuilder = new NlriBuilder();
+ if (isMultiPathSupported) {
+ nlriBuilder.setPathId(PathIdUtil.readPathId(buffer));
+ }
+ nlriBuilder.setPrefix(Ipv4Util.prefixForByteBuf(buffer));
+ nlri.add(nlriBuilder.build());
}
- final Update msg = eventBuilder.build();
+ if (!nlri.isEmpty()) {
+ builder.setNlri(nlri);
+ }
+ final Update msg = builder.build();
+ checkMandatoryAttributesPresence(msg);
LOG.debug("BGP Update message was parsed {}.", msg);
return msg;
}
- @Override
- public void serializeMessage(final Notification message, final ByteBuf bytes) {
- Preconditions.checkArgument(message instanceof Update, "BGPUpdate message cannot be null");
- LOG.trace("Started serializing update message: {}", message);
- final Update update = (Update) message;
+ /**
+ * Check for presence of well known mandatory path attributes
+ * ORIGIN, AS_PATH and NEXT_HOP in Update message
+ *
+ * @param message Update message
+ * @throws BGPDocumentedException
+ */
+ private static void checkMandatoryAttributesPresence(final Update message) throws BGPDocumentedException {
+ requireNonNull(message, "Update message cannot be null");
- final ByteBuf messageBody = Unpooled.buffer();
- final WithdrawnRoutes withdrawnRoutes = update.getWithdrawnRoutes();
- if (withdrawnRoutes != null) {
- final ByteBuf withdrawnRoutesBuf = Unpooled.buffer();
- for (final Ipv4Prefix prefix : withdrawnRoutes.getWithdrawnRoutes()) {
- withdrawnRoutesBuf.writeBytes(Ipv4Util.bytesForPrefixBegin(prefix));
+ final Attributes attrs = message.getAttributes();
+
+ if (message.getNlri() != null) {
+ if (attrs == null || attrs.getCNextHop() == null) {
+ throw new BGPDocumentedException(BGPError.MANDATORY_ATTR_MISSING_MSG + "NEXT_HOP",
+ BGPError.WELL_KNOWN_ATTR_MISSING,
+ new byte[] { NextHopAttributeParser.TYPE });
}
- messageBody.writeShort(withdrawnRoutesBuf.writerIndex());
- messageBody.writeBytes(withdrawnRoutesBuf);
- } else {
- messageBody.writeZero(WITHDRAWN_ROUTES_LENGTH_SIZE);
- }
- if (update.getPathAttributes() != null) {
- final ByteBuf pathAttributesBuf = Unpooled.buffer();
- this.reg.serializeAttribute(update.getPathAttributes(), pathAttributesBuf);
- messageBody.writeShort(pathAttributesBuf.writerIndex());
- messageBody.writeBytes(pathAttributesBuf);
- } else {
- messageBody.writeZero(TOTAL_PATH_ATTR_LENGTH_SIZE);
}
- final Nlri nlri = update.getNlri();
- if (nlri != null) {
- for (final Ipv4Prefix prefix : nlri.getNlri()) {
- messageBody.writeBytes(Ipv4Util.bytesForPrefixBegin(prefix));
+
+ if (MessageUtil.isAnyNlriPresent(message)) {
+ if (attrs == null || attrs.getOrigin() == null) {
+ throw new BGPDocumentedException(BGPError.MANDATORY_ATTR_MISSING_MSG + "ORIGIN",
+ BGPError.WELL_KNOWN_ATTR_MISSING,
+ new byte[] { OriginAttributeParser.TYPE });
+ }
+
+ if (attrs.getAsPath() == null) {
+ throw new BGPDocumentedException(BGPError.MANDATORY_ATTR_MISSING_MSG + "AS_PATH",
+ BGPError.WELL_KNOWN_ATTR_MISSING,
+ new byte[] { AsPathAttributeParser.TYPE });
}
}
- LOG.trace("Update message serialized to {}", ByteBufUtil.hexDump(messageBody));
- MessageUtil.formatMessage(TYPE, messageBody, bytes);
}
}