/*
* * 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.bgp.parser.impl.message;
import com.google.common.base.Preconditions;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufUtil;
import io.netty.buffer.Unpooled;
import java.util.Arrays;
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.impl.message.update.ClusterIdAttributeParser;
import org.opendaylight.protocol.bgp.parser.impl.message.update.OriginatorIdAttributeParser;
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.concepts.Ipv4Util;
import org.opendaylight.protocol.util.ByteArray;
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.yangtools.yang.binding.Notification;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* LENGTH fields, that denote the length of the fields with variable length, have fixed SIZE.
*
* @see BGP-4 Update Message Format
*/
public class BGPUpdateMessageParser implements MessageParser, MessageSerializer {
public static final int TYPE = 2;
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;
private final AttributeRegistry reg;
// Constructors -------------------------------------------------------
public BGPUpdateMessageParser(final AttributeRegistry reg) {
this.reg = Preconditions.checkNotNull(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: {}", Arrays.toString(ByteArray.getAllBytes(buffer)));
final int withdrawnRoutesLength = buffer.readUnsignedShort();
final UpdateBuilder eventBuilder = new UpdateBuilder();
if (withdrawnRoutesLength > 0) {
final List withdrawnRoutes = Ipv4Util.prefixListForBytes(ByteArray.readBytes(buffer, withdrawnRoutesLength));
eventBuilder.setWithdrawnRoutes(new WithdrawnRoutesBuilder().setWithdrawnRoutes(withdrawnRoutes).build());
}
final int totalPathAttrLength = buffer.readUnsignedShort();
if (withdrawnRoutesLength == 0 && totalPathAttrLength == 0) {
return eventBuilder.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) {
// 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 nlri = Ipv4Util.prefixListForBytes(ByteArray.readAllBytes(buffer));
if (nlri != null && !nlri.isEmpty()) {
eventBuilder.setNlri(new NlriBuilder().setNlri(nlri).build());
}
Update msg = eventBuilder.build();
LOG.debug("BGP Update message was parsed {}.", msg);
return msg;
}
@Override
public void serializeMessage(Notification message, ByteBuf bytes) {
if (message == null) {
throw new IllegalArgumentException("BGPUpdate message cannot be null");
}
LOG.trace("Started serializing update message: {}", message);
final Update update = (Update) message;
ByteBuf messageBody = Unpooled.buffer();
WithdrawnRoutes withdrawnRoutes = update.getWithdrawnRoutes();
if (withdrawnRoutes != null) {
ByteBuf withdrawnRoutesBuf = Unpooled.buffer();
for (Ipv4Prefix withdrawnRoutePrefix : withdrawnRoutes.getWithdrawnRoutes()) {
int prefixBits = Ipv4Util.getPrefixLength(withdrawnRoutePrefix.getValue());
byte[] prefixBytes = ByteArray.subByte(Ipv4Util.bytesForPrefix(withdrawnRoutePrefix), 0,
Ipv4Util.getPrefixLengthBytes(withdrawnRoutePrefix.getValue()));
withdrawnRoutesBuf.writeByte(prefixBits);
withdrawnRoutesBuf.writeBytes(prefixBytes);
}
messageBody.writeShort(withdrawnRoutesBuf.writerIndex());
messageBody.writeBytes(withdrawnRoutesBuf);
} else {
messageBody.writeZero(2);
}
if (update.getPathAttributes() != null) {
ByteBuf pathAttributesBuf = Unpooled.buffer();
this.reg.serializeAttribute(update.getPathAttributes(), pathAttributesBuf);
ClusterIdAttributeParser clusterIdAttributeParser = new ClusterIdAttributeParser();
clusterIdAttributeParser.serializeAttribute(update.getPathAttributes(), pathAttributesBuf);
OriginatorIdAttributeParser originatorIdAttributeParser = new OriginatorIdAttributeParser();
originatorIdAttributeParser.serializeAttribute(update.getPathAttributes(), pathAttributesBuf);
messageBody.writeShort(pathAttributesBuf.writerIndex());
messageBody.writeBytes(pathAttributesBuf);
} else {
messageBody.writeZero(2);
}
Nlri nlri = update.getNlri();
if (nlri != null) {
for (Ipv4Prefix ipv4Prefix : nlri.getNlri()) {
int prefixBits = Ipv4Util.getPrefixLength(ipv4Prefix.getValue());
byte[] prefixBytes = ByteArray.subByte(Ipv4Util.bytesForPrefix(ipv4Prefix), 0,
Ipv4Util.getPrefixLengthBytes(ipv4Prefix.getValue()));
messageBody.writeByte(prefixBits);
messageBody.writeBytes(prefixBytes);
}
}
LOG.trace("Update message serialized to {}", ByteBufUtil.hexDump(messageBody));
//FIXME: switch to ByteBuf
bytes.writeBytes(MessageUtil.formatMessage(TYPE, ByteArray.getAllBytes(messageBody)));
}
}