/* * Copyright (c) 2015 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.bmp.impl.app; import static org.opendaylight.protocol.bmp.impl.app.TablesUtil.BMP_TABLES_QNAME; import com.google.common.base.Preconditions; import com.google.common.collect.Sets; import java.util.Set; import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType; import org.opendaylight.controller.md.sal.dom.api.DOMDataWriteTransaction; import org.opendaylight.controller.md.sal.dom.api.DOMTransactionChain; import org.opendaylight.protocol.bgp.rib.spi.RIBExtensionConsumerContext; import org.opendaylight.protocol.bmp.impl.spi.BmpRouterPeer; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev100924.IpAddress; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.yang.types.rev100924.Timestamp; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.open.message.BgpParameters; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.open.message.bgp.parameters.OptionalCapabilities; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev130919.open.message.bgp.parameters.optional.capabilities.CParameters; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.CParameters1; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.open.bgp.parameters.optional.capabilities.c.parameters.MultiprotocolCapability; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.PeerId; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.rib.TablesKey; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.Ipv4AddressFamily; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.UnicastSubsequentAddressFamily; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.AdjRibInType; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.Mirror; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.MirrorInformationCode; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.PeerDownNotification; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.PeerUp; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.PeerUpNotification; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.RouteMirroringMessage; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.RouteMonitoringMessage; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.Stat; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.StatsReportsMessage; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.peer.header.PeerHeader; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.peer.up.ReceivedOpen; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.peer.up.SentOpen; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.message.rev150512.stat.Tlvs; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.BmpMonitor; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.bmp.monitor.Monitor; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.peers.Peer; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.peers.peer.Mirrors; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.peers.peer.PeerSession; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.peers.peer.PostPolicyRib; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.peers.peer.PrePolicyRib; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.peers.peer.Stats; import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bmp.monitor.rev150512.routers.Router; import org.opendaylight.yangtools.binding.data.codec.api.BindingCodecTree; import org.opendaylight.yangtools.binding.data.codec.api.BindingCodecTreeNode; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier; import org.opendaylight.yangtools.yang.binding.Notification; import org.opendaylight.yangtools.yang.common.QName; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates; import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode; import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode; import org.opendaylight.yangtools.yang.data.impl.schema.Builders; import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes; import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.DataContainerNodeAttrBuilder; public final class BmpRouterPeerImpl implements BmpRouterPeer { private static final String TIMESTAMP_SEC = "timestamp-sec"; private static final QName PEER_ID_QNAME = QName.create(Peer.QNAME, "peer-id").intern(); private static final QName PEER_TYPE_QNAME = QName.create(Peer.QNAME, "type"); private static final QName PEER_ADDRESS_QNAME = QName.create(Peer.QNAME, "address").intern(); private static final QName PEER_AS_QNAME = QName.create(Peer.QNAME, "as").intern(); private static final QName PEER_BGP_ID_QNAME = QName.create(Peer.QNAME, "bgp-id").intern(); private static final QName PEER_DISTINGUISHER_QNAME = QName.create(Peer.QNAME, "router-distinguisher").intern(); private static final QName PEER_LOCAL_ADDRESS_QNAME = QName.create(PeerSession.QNAME, "local-address").intern(); private static final QName PEER_LOCAL_PORT_QNAME = QName.create(PeerSession.QNAME, "local-port").intern(); private static final QName PEER_REMOTE_PORT_QNAME = QName.create(PeerSession.QNAME, "remote-port").intern(); private static final QName PEER_STATUS_QNAME = QName.create(PeerSession.QNAME, "status").intern(); private static final QName PEER_UP_TIMESTAMP_QNAME = QName.create(PeerSession.QNAME, TIMESTAMP_SEC).intern(); private static final QName PEER_STATS_TIMESTAMP_QNAME = QName.create(Stats.QNAME, TIMESTAMP_SEC).intern(); private static final QName PEER_MIRROR_INFORMATION_QNAME = QName.create(Mirrors.QNAME, "information").intern(); private static final QName PEER_MIRROR_TIMESTAMP_QNAME = QName.create(Mirrors.QNAME, TIMESTAMP_SEC).intern(); private static final QName STAT0_QNAME = QName.create(Stats.QNAME, "rejected-prefixes").intern(); private static final QName STAT1_QNAME = QName.create(Stats.QNAME, "duplicate-prefix-advertisements").intern(); private static final QName STAT2_QNAME = QName.create(Stats.QNAME, "duplicate-withdraws").intern(); private static final QName STAT3_QNAME = QName.create(Stats.QNAME, "invalidated-cluster-list-loop").intern(); private static final QName STAT4_QNAME = QName.create(Stats.QNAME, "invalidated-as-path-loop").intern(); private static final QName STAT5_QNAME = QName.create(Stats.QNAME, "invalidated-originator-id").intern(); private static final QName STAT6_QNAME = QName.create(Stats.QNAME, "invalidated-as-confed-loop").intern(); private static final QName STAT7_QNAME = QName.create(Stats.QNAME, "adj-ribs-in-routes"); private static final QName STAT8_QNAME = QName.create(Stats.QNAME, "loc-rib-routes"); private static final QName STAT9_QNAME = QName.create(Stats.QNAME, "per-afi-safi-adj-rib-in-routes").intern(); private static final QName AF_QNAME = QName.create(Stats.QNAME, "afi-safi").intern(); private static final QName COUNT_QNAME = QName.create(Stats.QNAME, "count").intern(); private static final QName STAT10_QNAME = QName.create(Stats.QNAME, "per-afi-safi-loc-rib-routes").intern(); private static final QName STAT11_QNAME = QName.create(Stats.QNAME, "updates-treated-as-withdraw").intern(); private static final QName STAT13_QNAME = QName.create(Stats.QNAME, "duplicate-updates").intern(); private static final TablesKey DEFAULT_TABLE = new TablesKey(Ipv4AddressFamily.class, UnicastSubsequentAddressFamily.class); private static final InstanceIdentifier SENT_OPEN_IID = InstanceIdentifier.builder(BmpMonitor.class) .child(Monitor.class) .child(Router.class) .child(Peer.class) .child(PeerSession.class) .child(SentOpen.class) .build(); private static final InstanceIdentifier RECEIVED_OPEN_IID = InstanceIdentifier.builder(BmpMonitor.class) .child(Monitor.class) .child(Router.class) .child(Peer.class) .child(PeerSession.class) .child(ReceivedOpen.class) .build(); private final DOMTransactionChain domTxChain; private final PeerId peerId; private final YangInstanceIdentifier peerYangIId; private final BmpRibInWriter prePolicyWriter; private final BmpRibInWriter postPolicyWriter; private final BindingCodecTreeNode sentOpenCodec; private final BindingCodecTreeNode receivedOpenCodec; private boolean up = true; private BmpRouterPeerImpl(final DOMTransactionChain domTxChain, final YangInstanceIdentifier peersYangIId, final PeerId peerId, final RIBExtensionConsumerContext extensions, final PeerUpNotification peerUp, final BindingCodecTree tree) { this.domTxChain = Preconditions.checkNotNull(domTxChain); this.peerId = peerId; this.peerYangIId = YangInstanceIdentifier.builder(peersYangIId).nodeWithKey(Peer.QNAME, PEER_ID_QNAME, this.peerId.getValue()).build(); final Set peerTables = setPeerTables(peerUp.getReceivedOpen()); final DOMDataWriteTransaction wTx = this.domTxChain.newWriteOnlyTransaction(); wTx.put(LogicalDatastoreType.OPERATIONAL, this.peerYangIId, createPeerEntry(peerUp)); wTx.submit(); this.prePolicyWriter = BmpRibInWriter.create(this.peerYangIId.node(PrePolicyRib.QNAME).node(BMP_TABLES_QNAME), this.domTxChain, extensions, peerTables, tree); this.postPolicyWriter = BmpRibInWriter.create(this.peerYangIId.node(PostPolicyRib.QNAME).node(BMP_TABLES_QNAME), this.domTxChain, extensions, peerTables, tree); this.sentOpenCodec = tree.getSubtreeCodec(SENT_OPEN_IID); this.receivedOpenCodec = tree.getSubtreeCodec(RECEIVED_OPEN_IID); } public static BmpRouterPeer createRouterPeer(final DOMTransactionChain domTxChain, final YangInstanceIdentifier peersYangIId, final PeerUpNotification peerUp, final RIBExtensionConsumerContext extensions, final BindingCodecTree tree, final PeerId peerId) { return new BmpRouterPeerImpl(domTxChain, peersYangIId, peerId, extensions, peerUp, tree); } @Override public void onPeerMessage(final Notification message) { if (message instanceof PeerDownNotification) { onPeerDown(); } else if (message instanceof RouteMonitoringMessage) { onRouteMonitoring((RouteMonitoringMessage) message); } else if (message instanceof StatsReportsMessage) { onStatsReports((StatsReportsMessage) message); } else if (message instanceof RouteMirroringMessage) { onRouteMirror((RouteMirroringMessage) message); } } private void onRouteMonitoring(final RouteMonitoringMessage routeMonitoring) { if (this.up) { final AdjRibInType ribType = routeMonitoring.getPeerHeader().getAdjRibInType(); switch (ribType) { case PrePolicy: this.prePolicyWriter.onMessage(routeMonitoring.getUpdate()); break; case PostPolicy: this.postPolicyWriter.onMessage(routeMonitoring.getUpdate()); break; default: break; } } } private void onStatsReports(final StatsReportsMessage statsReports) { if (this.up) { final DOMDataWriteTransaction wTx = this.domTxChain.newWriteOnlyTransaction(); wTx.merge(LogicalDatastoreType.OPERATIONAL, this.peerYangIId.node(Stats.QNAME), createStats(statsReports, statsReports.getPeerHeader().getTimestampSec())); wTx.submit(); } } private void onRouteMirror(final RouteMirroringMessage mirror) { final DOMDataWriteTransaction wTx = this.domTxChain.newWriteOnlyTransaction(); wTx.merge(LogicalDatastoreType.OPERATIONAL, this.peerYangIId.node(Mirrors.QNAME), createMirrors(mirror, mirror.getPeerHeader().getTimestampSec())); wTx.submit(); } private void onPeerDown() { final DOMDataWriteTransaction wTx = this.domTxChain.newWriteOnlyTransaction(); wTx.delete(LogicalDatastoreType.OPERATIONAL, this.peerYangIId); wTx.submit(); close(); } @Override public PeerId getPeerId() { return this.peerId; } @Override public void close() { Preconditions.checkState(this.up, "Already closed."); this.up = false; } private Set setPeerTables(final ReceivedOpen open) { final Set tables = Sets.newHashSet(DEFAULT_TABLE); for (final BgpParameters param : open.getBgpParameters()) { for (final OptionalCapabilities optCapa : param.getOptionalCapabilities()) { final CParameters cParam = optCapa.getCParameters(); if ( cParam.getAugmentation(CParameters1.class) == null || cParam.getAugmentation(CParameters1.class).getMultiprotocolCapability() == null ) { continue; } final MultiprotocolCapability multi = cParam.getAugmentation(CParameters1.class).getMultiprotocolCapability(); final TablesKey tt = new TablesKey(multi.getAfi(), multi.getSafi()); tables.add(tt); } } return tables; } private ContainerNode createPeerSessionUp(final PeerUp peerUp, final Timestamp timestamp) { final DataContainerNodeAttrBuilder builder = Builders .containerBuilder() .withNodeIdentifier(new NodeIdentifier(PeerSession.QNAME)) .withChild( ImmutableNodes.leafNode(PEER_LOCAL_ADDRESS_QNAME, getStringIpAddress(peerUp.getLocalAddress()))) .withChild( ImmutableNodes.leafNode(PEER_LOCAL_PORT_QNAME, peerUp .getLocalPort().getValue())) .withChild( ImmutableNodes.leafNode(PEER_REMOTE_PORT_QNAME, peerUp .getRemotePort().getValue())) .withChild(ImmutableNodes.leafNode(PEER_STATUS_QNAME, "up")) .withChild( ImmutableNodes.leafNode(PEER_UP_TIMESTAMP_QNAME, timestamp.getValue())); if (this.receivedOpenCodec != null) { builder.withChild(Builders.containerBuilder( (ContainerNode) this.receivedOpenCodec.serialize(peerUp .getReceivedOpen())).build()); } if (this.sentOpenCodec != null) { builder.withChild(Builders.containerBuilder( (ContainerNode) this.sentOpenCodec.serialize(peerUp .getSentOpen())).build()); } return builder.build(); } private MapEntryNode createPeerEntry(final PeerUpNotification peerUp) { final PeerHeader peerHeader = peerUp.getPeerHeader(); final DataContainerNodeAttrBuilder mapEntryBuilder = Builders.mapEntryBuilder() .withNodeIdentifier(new NodeIdentifierWithPredicates(Peer.QNAME, PEER_ID_QNAME, this.peerId.getValue())) .withChild(ImmutableNodes.leafNode(PEER_ID_QNAME, this.peerId.getValue())) .withChild(ImmutableNodes.leafNode(PEER_TYPE_QNAME, peerHeader.getType().name().toLowerCase())) .withChild(ImmutableNodes.leafNode(PEER_ADDRESS_QNAME, getStringIpAddress(peerHeader.getAddress()))) .withChild(ImmutableNodes.leafNode(PEER_AS_QNAME, peerHeader.getAs().getValue())) .withChild(ImmutableNodes.leafNode(PEER_BGP_ID_QNAME, peerHeader.getBgpId().getValue())) .withChild(createPeerSessionUp(peerUp, peerHeader.getTimestampSec())) .withChild( Builders.containerBuilder().withNodeIdentifier(new NodeIdentifier(PrePolicyRib.QNAME)) .withChild(ImmutableNodes.mapNodeBuilder(BMP_TABLES_QNAME).build()).build()) .withChild( Builders.containerBuilder().withNodeIdentifier(new NodeIdentifier(PostPolicyRib.QNAME)) .withChild(ImmutableNodes.mapNodeBuilder(BMP_TABLES_QNAME).build()).build()); if (peerHeader.getDistinguisher() != null) { mapEntryBuilder.withChild(ImmutableNodes.leafNode(PEER_DISTINGUISHER_QNAME, peerHeader.getDistinguisher().getDistinguisher())); } return mapEntryBuilder.build(); } private ContainerNode createStats(final Stat stat, final Timestamp timestamp) { final DataContainerNodeAttrBuilder builder = Builders.containerBuilder().withNodeIdentifier(new NodeIdentifier(Stats.QNAME)); builder.withChild(ImmutableNodes.leafNode(PEER_STATS_TIMESTAMP_QNAME, timestamp.getValue())); final Tlvs tlvs = stat.getTlvs(); if (tlvs != null) { statsForTlvs(tlvs, builder); } return builder.build(); } private static void statsForTlvs(final Tlvs tlvs, final DataContainerNodeAttrBuilder builder) { if (tlvs.getRejectedPrefixesTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT0_QNAME, tlvs.getRejectedPrefixesTlv().getCount().getValue())); } if (tlvs.getDuplicatePrefixAdvertisementsTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT1_QNAME, tlvs.getDuplicatePrefixAdvertisementsTlv().getCount().getValue())); } if (tlvs.getDuplicateWithdrawsTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT2_QNAME, tlvs.getDuplicateWithdrawsTlv().getCount().getValue())); } if (tlvs.getInvalidatedClusterListLoopTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT3_QNAME, tlvs.getInvalidatedClusterListLoopTlv().getCount().getValue())); } if (tlvs.getInvalidatedAsPathLoopTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT4_QNAME, tlvs.getInvalidatedAsPathLoopTlv().getCount().getValue())); } if (tlvs.getInvalidatedOriginatorIdTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT5_QNAME, tlvs.getInvalidatedOriginatorIdTlv().getCount().getValue())); } if (tlvs.getInvalidatedAsConfedLoopTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT6_QNAME, tlvs.getInvalidatedAsConfedLoopTlv().getCount().getValue())); } if (tlvs.getAdjRibsInRoutesTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT7_QNAME, tlvs.getAdjRibsInRoutesTlv().getCount().getValue())); } if (tlvs.getLocRibRoutesTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT8_QNAME, tlvs.getLocRibRoutesTlv().getCount().getValue())); } if (tlvs.getPerAfiSafiAdjRibInTlv() != null) { builder.withChild(Builders.containerBuilder().withNodeIdentifier(new NodeIdentifier(STAT9_QNAME)) .withChild(ImmutableNodes.mapNodeBuilder(AF_QNAME) .withChild(Builders.mapEntryBuilder() .withChild(ImmutableNodes.leafNode(COUNT_QNAME, tlvs.getPerAfiSafiAdjRibInTlv().getCount().getValue())) .withNodeIdentifier(TablesUtil.toYangTablesKey(AF_QNAME, Ipv4AddressFamily.class, UnicastSubsequentAddressFamily.class)) .build()).build()).build()); } if (tlvs.getPerAfiSafiLocRibTlv() != null) { builder.withChild(Builders.containerBuilder().withNodeIdentifier(new NodeIdentifier(STAT10_QNAME)) .withChild(ImmutableNodes.mapNodeBuilder(AF_QNAME) .withChild(Builders.mapEntryBuilder() .withChild(ImmutableNodes.leafNode(COUNT_QNAME, tlvs.getPerAfiSafiLocRibTlv().getCount().getValue())) .withNodeIdentifier(TablesUtil.toYangTablesKey(AF_QNAME, Ipv4AddressFamily.class, UnicastSubsequentAddressFamily.class)) .build()).build()).build()); } if (tlvs.getUpdatesTreatedAsWithdrawTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT11_QNAME, tlvs.getUpdatesTreatedAsWithdrawTlv().getCount().getValue())); } if (tlvs.getPrefixesTreatedAsWithdrawTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT11_QNAME, tlvs.getPrefixesTreatedAsWithdrawTlv().getCount().getValue())); } if (tlvs.getDuplicateUpdatesTlv() != null) { builder.withChild(ImmutableNodes.leafNode(STAT13_QNAME, tlvs.getDuplicateUpdatesTlv().getCount().getValue())); } } private ContainerNode createMirrors(final Mirror mirror, final Timestamp timestamp) { final DataContainerNodeAttrBuilder builder = Builders.containerBuilder().withNodeIdentifier(new NodeIdentifier(Mirrors.QNAME)); builder.withChild(ImmutableNodes.leafNode(PEER_MIRROR_INFORMATION_QNAME, MirrorInformationCode.forValue(mirror.getTlvs().getMirrorInformationTlv().getCode().getIntValue()))); builder.withChild(ImmutableNodes.leafNode(PEER_MIRROR_TIMESTAMP_QNAME, timestamp.getValue())); return builder.build(); } private static String getStringIpAddress(final IpAddress ipAddress) { if (ipAddress.getIpv4Address() != null) { return ipAddress.getIpv4Address().getValue(); } return ipAddress.getIpv6Address().getValue(); } }