Line data Source code
1 : /*
2 : * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3 : */
4 :
5 : #include "bgp/xmpp_message_builder.h"
6 :
7 : #include <boost/foreach.hpp>
8 :
9 : #include <algorithm>
10 :
11 : #include "bgp/ipeer.h"
12 : #include "bgp/bgp_server.h"
13 : #include "bgp/bgp_table.h"
14 : #include "bgp/extended-community/etree.h"
15 : #include "bgp/extended-community/mac_mobility.h"
16 : #include "bgp/ermvpn/ermvpn_route.h"
17 : #include "bgp/evpn/evpn_route.h"
18 : #include "bgp/mvpn/mvpn_route.h"
19 : #include "bgp/routing-instance/routing_instance.h"
20 : #include "bgp/security_group/security_group.h"
21 : #include "db/db.h"
22 : #include "net/community_type.h"
23 : #include "schema/xmpp_multicast_types.h"
24 : #include "schema/xmpp_mvpn_types.h"
25 : #include "schema/xmpp_enet_types.h"
26 : #include "xmpp/xmpp_init.h"
27 :
28 : using pugi::xml_attribute;
29 : using pugi::xml_document;
30 : using pugi::xml_node;
31 : using std::ostringstream;
32 : using std::copy;
33 : using std::fill;
34 : using std::string;
35 : using std::stringstream;
36 : using std::vector;
37 :
38 469534 : static inline const char *AfiName(uint16_t afi) {
39 469534 : switch (afi) {
40 259878 : case BgpAf::IPv4:
41 259878 : return "1";
42 : break;
43 199233 : case BgpAf::IPv6:
44 199233 : return "2";
45 : break;
46 10448 : case BgpAf::L2Vpn:
47 10448 : return "25";
48 : break;
49 : }
50 0 : assert(false);
51 : return NULL;
52 : }
53 :
54 469551 : static inline const char *XmppSafiName(uint8_t safi) {
55 469551 : switch (safi) {
56 449871 : case BgpAf::Unicast:
57 449871 : return "1";
58 : break;
59 12 : case BgpAf::Mpls:
60 12 : return "4";
61 : break;
62 6477 : case BgpAf::MVpn:
63 6477 : return "5";
64 : break;
65 2752 : case BgpAf::Mcast:
66 2752 : return "241";
67 : break;
68 10448 : case BgpAf::Enet:
69 10448 : return "242";
70 : break;
71 : }
72 0 : assert(false);
73 : return NULL;
74 : }
75 :
76 144 : BgpXmppMessage::BgpXmppMessage()
77 144 : : table_(NULL),
78 144 : writer_(XmlWriter(&repr_)),
79 144 : is_reachable_(false),
80 144 : cache_routes_(false),
81 144 : repr_valid_(false),
82 144 : mobility_(0, false),
83 288 : etree_leaf_(false) {
84 144 : msg_begin_.reserve(kMaxFromToLength);
85 144 : }
86 :
87 288 : BgpXmppMessage::~BgpXmppMessage() {
88 288 : }
89 :
90 469451 : void BgpXmppMessage::Reset() {
91 469451 : Message::Reset();
92 469454 : table_ = NULL;
93 469454 : is_reachable_ = false;
94 469454 : cache_routes_ = false;
95 469454 : repr_valid_ = false;
96 469454 : repr_.clear();
97 469451 : }
98 :
99 469447 : bool BgpXmppMessage::Start(const RibOut *ribout, bool cache_routes,
100 : const RibOutAttr *roattr, const BgpRoute *route) {
101 469447 : Reset();
102 469444 : table_ = ribout->table();
103 469447 : is_reachable_ = roattr->IsReachable();
104 469444 : cache_routes_ = cache_routes;
105 469444 : Address::Family family = table_->family();
106 :
107 469436 : if (is_reachable_) {
108 316879 : const BgpAttr *attr = roattr->attr();
109 316875 : ProcessCommunity(attr->community());
110 316870 : ProcessExtCommunity(attr->ext_community());
111 : }
112 :
113 : // Reserve space for the begin line that contains the message opening tag
114 : // with from and to attributes. Actual value gets patched in when GetData
115 : // is called.
116 469478 : repr_.append(kMaxFromToLength, ' ');
117 :
118 : // Add opening tags for event and items. The closing tags are added when
119 : // GetData is called.
120 469481 : repr_ += "\n\t<event xmlns=\"http://jabber.org/protocol/pubsub\">";
121 469491 : repr_ += "\n\t\t<items node=\"";
122 469532 : repr_ += AfiName(BgpAf::FamilyToAfi(family));
123 469554 : repr_ += "/";
124 469568 : repr_ += XmppSafiName(BgpAf::FamilyToXmppSafi(family));
125 469556 : repr_ += "/";
126 469558 : repr_ += table_->routing_instance()->name();
127 469548 : repr_ += "\">\n";
128 :
129 469553 : if (table_->family() == Address::ERMVPN) {
130 2752 : AddMcastRoute(route, roattr);
131 466795 : } else if (table_->family() == Address::EVPN) {
132 10448 : AddEnetRoute(route, roattr);
133 456337 : } else if (table_->family() == Address::INET6) {
134 199238 : AddInet6Route(route, roattr);
135 257096 : } else if (table_->family() == Address::MVPN) {
136 6477 : AddMvpnRoute(route, roattr);
137 : } else {
138 250634 : AddInetRoute(route, roattr);
139 : }
140 469583 : return true;
141 : }
142 :
143 469606 : void BgpXmppMessage::Finish() {
144 469606 : }
145 :
146 620542 : bool BgpXmppMessage::AddRoute(const BgpRoute *route, const RibOutAttr *roattr) {
147 620542 : assert(is_reachable_ == roattr->IsReachable());
148 620538 : if (is_reachable_ && num_reach_route_ >= kMaxReachCount)
149 224 : return false;
150 620314 : if (!is_reachable_ && num_unreach_route_ >= kMaxUnreachCount)
151 8 : return false;
152 :
153 620306 : if (is_reachable_) {
154 519859 : const BgpAttr *attr = roattr->attr();
155 519856 : ProcessCommunity(attr->community());
156 519854 : ProcessExtCommunity(attr->ext_community());
157 : }
158 :
159 620316 : if (table_->family() == Address::ERMVPN) {
160 139 : return AddMcastRoute(route, roattr);
161 620174 : } else if (table_->family() == Address::EVPN) {
162 2171 : return AddEnetRoute(route, roattr);
163 618002 : } else if (table_->family() == Address::INET6) {
164 121251 : return AddInet6Route(route, roattr);
165 496750 : } else if (table_->family() == Address::MVPN) {
166 190464 : return AddMvpnRoute(route, roattr);
167 : } else {
168 306321 : return AddInetRoute(route, roattr);
169 : }
170 : }
171 :
172 530904 : void BgpXmppMessage::EncodeNextHop(const BgpRoute *route,
173 : const RibOutAttr::NextHop &nexthop,
174 : autogen::ItemType *item) {
175 530904 : autogen::NextHopType item_nexthop;
176 :
177 530892 : const IpAddress &address = nexthop.address();
178 530891 : if (address.is_v4()) {
179 527572 : item_nexthop.af = BgpAf::IPv4;
180 527572 : item_nexthop.address = address.to_v4().to_string();
181 : } else {
182 3321 : item_nexthop.af = BgpAf::IPv6;
183 3321 : item_nexthop.address = address.to_v6().to_string();
184 : }
185 531138 : item_nexthop.label = nexthop.label();
186 531134 : item_nexthop.virtual_network = GetVirtualNetwork(nexthop);
187 531124 : item_nexthop.tag_list.tag = nexthop.tag_list();
188 530968 : item_nexthop.is_new_tags_list = true;
189 :
190 : // If there's a non-zero label and encap list is empty use mpls over gre
191 : // as default encap.
192 530968 : if (item_nexthop.label) {
193 522003 : vector<string> &encap_list =
194 : item_nexthop.tunnel_encapsulation_list.tunnel_encapsulation;
195 522003 : if (nexthop.encap().empty()) {
196 100648 : encap_list.push_back(string("gre"));
197 : } else {
198 421363 : encap_list = nexthop.encap();
199 : }
200 : }
201 :
202 530927 : item->entry.next_hops.next_hop.push_back(item_nexthop);
203 530913 : }
204 :
205 626932 : void BgpXmppMessage::AddIpReach(const BgpRoute *route,
206 : const RibOutAttr *roattr) {
207 626932 : if (!roattr->repr().empty()) {
208 98208 : repr_ += roattr->repr();
209 98208 : return;
210 : }
211 528725 : Address::Family family = table_->family();
212 :
213 528718 : autogen::ItemType item;
214 528668 : item.entry.nlri.af = BgpAf::FamilyToAfi(family);
215 528668 : item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
216 528665 : item.entry.nlri.address = route->ToString();
217 528749 : item.entry.version = 1;
218 528749 : item.entry.virtual_network = GetVirtualNetwork(route, roattr);
219 528744 : item.entry.local_preference = roattr->attr()->local_pref();
220 528716 : item.entry.med = roattr->attr()->med();
221 528693 : item.entry.sequence_number = mobility_.sequence_number;
222 528693 : item.entry.mobility.seqno = mobility_.sequence_number;
223 528693 : item.entry.mobility.sticky = mobility_.sticky;
224 :
225 528693 : assert(!roattr->nexthop_list().empty());
226 :
227 : // Encode all next-hops in the list.
228 1590540 : BOOST_FOREACH(const RibOutAttr::NextHop &nexthop, roattr->nexthop_list()) {
229 530904 : EncodeNextHop(route, nexthop, &item);
230 : }
231 :
232 528634 : for (vector<int>::const_iterator it = security_group_list_.begin();
233 851220 : it != security_group_list_.end(); ++it) {
234 322620 : item.entry.security_group_list.security_group.push_back(*it);
235 : }
236 :
237 528602 : for (vector<string>::const_iterator it = community_list_.begin();
238 540477 : it != community_list_.end(); ++it) {
239 11887 : item.entry.community_tag_list.community_tag.push_back(*it);
240 : }
241 :
242 : // Encode load balance attribute.
243 528582 : if (!load_balance_attribute_.IsDefault())
244 4 : load_balance_attribute_.Encode(&item.entry.load_balance);
245 :
246 528581 : xml_node node = doc_.append_child("item");
247 528550 : node.append_attribute("id") = route->ToXmppIdString().c_str();
248 :
249 : // Remember the previous size.
250 : // Using remove_child instead of reset allows memory pages allocated for
251 : // the xml_document to be reused during the lifetime of the xml_document.
252 528636 : size_t pos = repr_.size();
253 528638 : item.Encode(&node);
254 528598 : doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
255 528755 : doc_.remove_child(node);
256 :
257 : // Cache the substring starting at the previous size.
258 528744 : if (cache_routes_)
259 28438 : roattr->set_repr(repr_, pos);
260 528746 : }
261 :
262 250456 : void BgpXmppMessage::AddIpUnreach(const BgpRoute *route) {
263 250456 : repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
264 250557 : }
265 :
266 556951 : bool BgpXmppMessage::AddInetRoute(const BgpRoute *route,
267 : const RibOutAttr *roattr) {
268 556951 : if (is_reachable_) {
269 424653 : num_reach_route_++;
270 424653 : AddIpReach(route, roattr);
271 : } else {
272 132298 : num_unreach_route_++;
273 132298 : AddIpUnreach(route);
274 : }
275 557014 : return true;
276 : }
277 :
278 320485 : bool BgpXmppMessage::AddInet6Route(const BgpRoute *route,
279 : const RibOutAttr *roattr) {
280 320485 : if (is_reachable_) {
281 202296 : num_reach_route_++;
282 202296 : AddIpReach(route, roattr);
283 : } else {
284 118189 : num_unreach_route_++;
285 118189 : AddIpUnreach(route);
286 : }
287 320504 : return true;
288 : }
289 :
290 9070 : void BgpXmppMessage::EncodeEnetNextHop(const BgpRoute *route,
291 : const RibOutAttr::NextHop &nexthop,
292 : autogen::EnetItemType *item) {
293 9070 : autogen::EnetNextHopType item_nexthop;
294 :
295 9070 : item_nexthop.af = BgpAf::IPv4;
296 9070 : item_nexthop.address = nexthop.address().to_v4().to_string();
297 9070 : item_nexthop.label = nexthop.label();
298 9070 : item_nexthop.l3_label = nexthop.l3_label();
299 9070 : if (!nexthop.mac().IsZero())
300 20 : item_nexthop.mac = nexthop.mac().ToString();
301 :
302 : // If encap list is empty use mpls over gre as default encap.
303 9070 : vector<string> &encap_list =
304 : item_nexthop.tunnel_encapsulation_list.tunnel_encapsulation;
305 9070 : if (nexthop.encap().empty()) {
306 0 : encap_list.push_back(string("gre"));
307 : } else {
308 9070 : encap_list = nexthop.encap();
309 : }
310 9070 : item_nexthop.tag_list.tag = nexthop.tag_list();
311 9070 : item_nexthop.is_new_tags_list = true;
312 9070 : item->entry.next_hops.next_hop.push_back(item_nexthop);
313 9070 : }
314 :
315 10627 : void BgpXmppMessage::AddEnetReach(const BgpRoute *route,
316 : const RibOutAttr *roattr) {
317 10627 : if (!roattr->repr().empty()) {
318 0 : repr_ += roattr->repr();
319 0 : return;
320 : }
321 10627 : Address::Family family = table_->family();
322 :
323 10627 : autogen::EnetItemType item;
324 10627 : item.entry.nlri.af = BgpAf::FamilyToAfi(family);
325 10627 : item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
326 :
327 10627 : EvpnRoute *evpn_route =
328 : static_cast<EvpnRoute *>(const_cast<BgpRoute *>(route));
329 10627 : const EvpnPrefix &evpn_prefix = evpn_route->GetPrefix();
330 10627 : item.entry.nlri.ethernet_tag = evpn_prefix.tag();
331 10627 : item.entry.nlri.mac = evpn_prefix.mac_addr().ToString();
332 21254 : item.entry.nlri.address = evpn_prefix.ip_address().to_string() + "/" +
333 31881 : integerToString(evpn_prefix.ip_address_length());
334 10627 : item.entry.nlri.source = evpn_prefix.source().to_string();
335 10627 : item.entry.nlri.group = evpn_prefix.group().to_string();
336 :
337 10627 : item.entry.virtual_network = GetVirtualNetwork(route, roattr);
338 10627 : item.entry.local_preference = roattr->attr()->local_pref();
339 10627 : item.entry.med = roattr->attr()->med();
340 10627 : item.entry.sequence_number = mobility_.sequence_number;
341 10627 : item.entry.mobility.seqno = mobility_.sequence_number;
342 10627 : item.entry.mobility.sticky = mobility_.sticky;
343 10627 : item.entry.etree_leaf = etree_leaf_;
344 :
345 10627 : for (vector<int>::const_iterator it = security_group_list_.begin();
346 21262 : it != security_group_list_.end(); ++it) {
347 10635 : item.entry.security_group_list.security_group.push_back(*it);
348 : }
349 :
350 10627 : const BgpOList *olist = roattr->attr()->olist().get();
351 10627 : assert((olist == NULL) != roattr->nexthop_list().empty());
352 :
353 10627 : if (olist) {
354 1685 : assert(olist->olist().subcode == BgpAttribute::OList);
355 15809 : BOOST_FOREACH(const BgpOListElem *elem, olist->elements()) {
356 7062 : autogen::EnetNextHopType nh;
357 7062 : nh.af = BgpAf::IPv4;
358 7062 : nh.address = elem->address.to_string();
359 7062 : nh.label = elem->label;
360 7062 : nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
361 7062 : item.entry.olist.next_hop.push_back(nh);
362 7062 : }
363 : }
364 :
365 10627 : const BgpOList *leaf_olist = roattr->attr()->leaf_olist().get();
366 10627 : assert((leaf_olist == NULL) != roattr->nexthop_list().empty());
367 :
368 10627 : if (leaf_olist) {
369 1685 : assert(leaf_olist->olist().subcode == BgpAttribute::LeafOList);
370 1685 : BOOST_FOREACH(const BgpOListElem *elem, leaf_olist->elements()) {
371 0 : autogen::EnetNextHopType nh;
372 0 : nh.af = BgpAf::IPv4;
373 0 : nh.address = elem->address.to_string();
374 0 : nh.label = elem->label;
375 0 : nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
376 0 : item.entry.leaf_olist.next_hop.push_back(nh);
377 0 : }
378 : }
379 :
380 28767 : BOOST_FOREACH(const RibOutAttr::NextHop &nexthop, roattr->nexthop_list()) {
381 9070 : EncodeEnetNextHop(route, nexthop, &item);
382 : }
383 :
384 11550 : for (const auto &peer_name : route->peer_sources()) {
385 923 : item.entry.peers.peer.push_back(peer_name);
386 : }
387 :
388 10627 : xml_node node = doc_.append_child("item");
389 10627 : node.append_attribute("id") = route->ToXmppIdString().c_str();
390 :
391 : // Remember the previous size.
392 : // Using remove_child instead of reset allows memory pages allocated for
393 : // the xml_document to be reused during the lifetime of the xml_document.
394 10627 : size_t pos = repr_.size();
395 10627 : item.Encode(&node);
396 10627 : doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
397 10627 : doc_.remove_child(node);
398 :
399 : // Cache the substring starting at the previous size.
400 10627 : if (cache_routes_)
401 1904 : roattr->set_repr(repr_, pos);
402 10627 : }
403 :
404 1992 : void BgpXmppMessage::AddEnetUnreach(const BgpRoute *route) {
405 1992 : repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
406 1992 : }
407 :
408 12619 : bool BgpXmppMessage::AddEnetRoute(const BgpRoute *route,
409 : const RibOutAttr *roattr) {
410 12619 : if (is_reachable_) {
411 10627 : num_reach_route_++;
412 10627 : AddEnetReach(route, roattr);
413 : } else {
414 1992 : num_unreach_route_++;
415 1992 : AddEnetUnreach(route);
416 : }
417 12619 : return true;
418 : }
419 :
420 : //
421 : // Note that there's no need to cache the string representation since a given
422 : // mcast route is sent to exactly one xmpp peer.
423 : //
424 2426 : void BgpXmppMessage::AddMcastReach(const BgpRoute *route,
425 : const RibOutAttr *roattr) {
426 2426 : Address::Family family = table_->family();
427 2426 : autogen::McastItemType item;
428 2426 : item.entry.nlri.af = BgpAf::FamilyToAfi(family);
429 2426 : item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
430 :
431 2426 : ErmVpnRoute *ermvpn_route =
432 : static_cast<ErmVpnRoute *>(const_cast<BgpRoute *>(route));
433 2426 : item.entry.nlri.group = ermvpn_route->GetPrefix().group().to_string();
434 2426 : item.entry.nlri.source = ermvpn_route->GetPrefix().source().to_string();
435 2426 : item.entry.nlri.source_label = roattr->label();
436 2426 : if (!roattr->source_address().is_unspecified()) {
437 : item.entry.nlri.source_address =
438 146 : roattr->source_address().to_string();
439 : }
440 :
441 2426 : const BgpOList *olist = roattr->attr()->olist().get();
442 2426 : assert(olist->olist().subcode == BgpAttribute::OList);
443 7624 : BOOST_FOREACH(const BgpOListElem *elem, olist->elements()) {
444 2599 : autogen::McastNextHopType nh;
445 2599 : nh.af = BgpAf::IPv4;
446 2599 : nh.address = elem->address.to_string();
447 2599 : nh.label = integerToString(elem->label);
448 2599 : nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
449 2599 : item.entry.olist.next_hop.push_back(nh);
450 2599 : }
451 :
452 : // Using remove_child instead of reset allows memory pages allocated for
453 : // the xml_document to be reused during the lifetime of the xml_document.
454 2426 : xml_node node = doc_.append_child("item");
455 2426 : node.append_attribute("id") = route->ToXmppIdString().c_str();
456 2426 : item.Encode(&node);
457 2426 : doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
458 2426 : doc_.remove_child(node);
459 2426 : }
460 :
461 465 : void BgpXmppMessage::AddMcastUnreach(const BgpRoute *route) {
462 465 : repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
463 465 : }
464 :
465 2891 : bool BgpXmppMessage::AddMcastRoute(const BgpRoute *route,
466 : const RibOutAttr *roattr) {
467 2891 : if (is_reachable_) {
468 2426 : num_reach_route_++;
469 2426 : AddMcastReach(route, roattr);
470 : } else {
471 465 : num_unreach_route_++;
472 465 : AddMcastUnreach(route);
473 : }
474 2891 : return true;
475 : }
476 :
477 196817 : void BgpXmppMessage::AddMvpnReach(const BgpRoute *route,
478 : const RibOutAttr *roattr) {
479 196817 : Address::Family family = table_->family();
480 196817 : autogen::MvpnItemType item;
481 196817 : item.entry.nlri.af = BgpAf::FamilyToAfi(family);
482 196817 : item.entry.nlri.safi = BgpAf::FamilyToXmppSafi(family);
483 :
484 196817 : MvpnRoute *mvpn_route =
485 : static_cast<MvpnRoute *>(const_cast<BgpRoute *>(route));
486 196817 : item.entry.nlri.group = mvpn_route->GetPrefix().group().to_string();
487 196817 : item.entry.nlri.source = mvpn_route->GetPrefix().source().to_string();
488 196817 : item.entry.nlri.route_type = mvpn_route->GetPrefix().type();
489 196817 : assert((item.entry.nlri.route_type == MvpnPrefix::SourceActiveADRoute) ||
490 : (item.entry.nlri.route_type == MvpnPrefix::SourceTreeJoinRoute));
491 :
492 196817 : if (item.entry.nlri.route_type == MvpnPrefix::SourceActiveADRoute) {
493 196723 : const BgpOList *olist = roattr->attr()->olist().get();
494 196723 : assert(olist->olist().subcode == BgpAttribute::OList);
495 590169 : BOOST_FOREACH(const BgpOListElem *elem, olist->elements()) {
496 196723 : autogen::MvpnNextHopType nh;
497 196723 : nh.af = BgpAf::IPv4;
498 196723 : nh.address = elem->address.to_string();
499 196723 : nh.label = elem->label;
500 196723 : nh.tunnel_encapsulation_list.tunnel_encapsulation = elem->encap;
501 196723 : item.entry.olist.next_hop.push_back(nh);
502 196723 : }
503 : }
504 :
505 : // Using remove_child instead of reset allows memory pages allocated for
506 : // the xml_document to be reused during the lifetime of the xml_document.
507 196817 : xml_node node = doc_.append_child("item");
508 196817 : node.append_attribute("id") = route->ToXmppIdString().c_str();
509 196817 : item.Encode(&node);
510 196817 : doc_.print(writer_, "\t", pugi::format_default, pugi::encoding_auto, 3);
511 196817 : doc_.remove_child(node);
512 196817 : }
513 :
514 124 : void BgpXmppMessage::AddMvpnUnreach(const BgpRoute *route) {
515 124 : repr_ += "\t\t\t<retract id=\"" + route->ToXmppIdString() + "\" />\n";
516 124 : }
517 :
518 196941 : bool BgpXmppMessage::AddMvpnRoute(const BgpRoute *route,
519 : const RibOutAttr *roattr) {
520 196941 : if (is_reachable_) {
521 196817 : num_reach_route_++;
522 196817 : AddMvpnReach(route, roattr);
523 : } else {
524 124 : num_unreach_route_++;
525 124 : AddMvpnUnreach(route);
526 : }
527 196941 : return true;
528 : }
529 :
530 14156244 : const uint8_t *BgpXmppMessage::GetData(IPeerUpdate *peer, size_t *lenp,
531 : const string **msg_str, string *temp) {
532 : // Build begin line that contains message opening tag with from and to
533 : // attributes.
534 14156244 : msg_begin_.clear();
535 14156301 : msg_begin_ += "\n<message from=\"";
536 14156362 : msg_begin_ += XmppInit::kControlNodeJID;
537 14156454 : msg_begin_ += "\" to=\"";
538 14156525 : msg_begin_ += peer->ToString();
539 14156576 : msg_begin_ += "/";
540 14156576 : msg_begin_ += XmppInit::kBgpPeer;
541 14156563 : msg_begin_ += "\">";
542 :
543 : // Add closing tags if this is the first peer to which the message will
544 : // be sent.
545 14156570 : if (!repr_valid_) {
546 469607 : repr_ += "\t\t</items>\n\t</event>\n</message>\n";
547 469609 : repr_valid_ = true;
548 : }
549 :
550 : // Replace the begin line if it fits in the space reserved at the start
551 : // of repr_. Use fill and copy instead of string::replace as the latter
552 : // seems to construct a new temporary string to hold the input data to
553 : // be copied.
554 : // Otherwise build a new string with the begin line and the rest of the
555 : // message in repr_.
556 14156572 : size_t begin_size = msg_begin_.size();
557 14156505 : if (begin_size <= kMaxFromToLength) {
558 9962224 : size_t extra = kMaxFromToLength - begin_size;
559 9962224 : char *data = const_cast<char *>(repr_.c_str());
560 9962224 : fill(data, data + extra, ' ');
561 9962094 : copy(msg_begin_.c_str(), msg_begin_.c_str() + begin_size, data + extra);
562 9962043 : *lenp = repr_.size() - extra;
563 9962016 : *msg_str = &repr_;
564 9962016 : return reinterpret_cast<const uint8_t *>(repr_.c_str()) + extra;
565 : } else {
566 4194281 : *temp = msg_begin_ + string(repr_, kMaxFromToLength);
567 4194304 : *lenp = temp->size();
568 4194304 : *msg_str = NULL;
569 4194304 : return reinterpret_cast<const uint8_t *>(temp->c_str());
570 : }
571 : }
572 :
573 836710 : void BgpXmppMessage::ProcessCommunity(const Community *community) {
574 836710 : community_list_.clear();
575 836719 : if (community == NULL)
576 824856 : return;
577 35633 : BOOST_FOREACH(uint32_t value, community->communities()) {
578 11885 : community_list_.push_back(CommunityType::CommunityToString(value));
579 : }
580 : }
581 :
582 836707 : void BgpXmppMessage::ProcessExtCommunity(const ExtCommunity *ext_community) {
583 836707 : mobility_.sequence_number = 0;
584 836707 : mobility_.sticky = false;
585 836707 : etree_leaf_ = false;
586 836707 : security_group_list_.clear();
587 836718 : load_balance_attribute_ = LoadBalance::LoadBalanceAttribute();
588 836884 : if (ext_community == NULL)
589 208115 : return;
590 :
591 628769 : as_t as_number = table_->server()->autonomous_system();
592 628738 : bool sg_asn_match = true;
593 628704 : for (ExtCommunity::ExtCommunityList::const_iterator iter =
594 628738 : ext_community->communities().begin();
595 3689381 : iter != ext_community->communities().end(); ++iter) {
596 3060660 : if (ExtCommunity::is_security_group(*iter)) {
597 431604 : SecurityGroup sg(*iter);
598 431591 : if (as_number <= AS2_MAX)
599 234983 : if (sg.as_number() != as_number && !sg.IsGlobal())
600 2 : continue;
601 431605 : security_group_list_.push_back(sg.security_group_id());
602 2629168 : } else if (ExtCommunity::is_security_group4(*iter)) {
603 196608 : SecurityGroup4ByteAs sg(*iter);
604 196608 : if (sg.as_number() != as_number)
605 0 : sg_asn_match = false;
606 2432388 : } else if (ExtCommunity::is_mac_mobility(*iter)) {
607 1079 : MacMobility mm(*iter);
608 1079 : mobility_.sequence_number = mm.sequence_number();
609 1079 : mobility_.sticky = mm.sticky();
610 2431440 : } else if (ExtCommunity::is_load_balance(*iter)) {
611 4 : LoadBalance load_balance(*iter);
612 4 : load_balance.FillAttribute(&load_balance_attribute_);
613 2431417 : } else if (ExtCommunity::is_etree(*iter)) {
614 9480 : ETree etree(*iter);
615 9480 : etree_leaf_ = etree.leaf();
616 : }
617 : }
618 628648 : if (!sg_asn_match)
619 0 : security_group_list_.clear();
620 : }
621 :
622 1068695 : string BgpXmppMessage::GetVirtualNetwork(
623 : const RibOutAttr::NextHop &nexthop) const {
624 1068695 : int index = nexthop.origin_vn_index();
625 1068681 : if (index > 0) {
626 : const RoutingInstanceMgr *manager =
627 69095 : table_->routing_instance()->manager();
628 69099 : return manager->GetVirtualNetworkByVnIndex(index);
629 999586 : } else if (index == 0) {
630 61270 : return table_->routing_instance()->GetVirtualNetworkName();
631 : } else {
632 938316 : return "unresolved";
633 : }
634 : }
635 :
636 539357 : string BgpXmppMessage::GetVirtualNetwork(const BgpRoute *route,
637 : const RibOutAttr *roattr) const {
638 539357 : if (!is_reachable_) {
639 0 : return "unresolved";
640 539357 : } else if (roattr->nexthop_list().empty()) {
641 1685 : if (roattr->vrf_originated()) {
642 1685 : return table_->routing_instance()->GetVirtualNetworkName();
643 : } else {
644 0 : return "unresolved";
645 : }
646 : } else {
647 537631 : return GetVirtualNetwork(roattr->nexthop_list().front());
648 : }
649 : }
650 :
651 50 : BgpXmppMessageBuilder::BgpXmppMessageBuilder() {
652 52 : }
653 :
654 144 : Message *BgpXmppMessageBuilder::Create() const {
655 144 : return new BgpXmppMessage;
656 : }
|