OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
agent_uve_base.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 #include <base/cpuinfo.h>
6 #include <db/db.h>
7 #include <cmn/agent_cmn.h>
9 #include <oper/interface.h>
10 
11 #include "vr_genetlink.h"
12 #include "nl_util.h"
13 
14 #include <uve/agent_uve_base.h>
15 #include <uve/vn_uve_table_base.h>
16 #include <uve/stats_interval_types.h>
17 #include <init/agent_param.h>
18 #include <oper/mirror_table.h>
19 #include <oper/global_vrouter.h>
20 #include <oper/tag.h>
22 #include <cmn/agent_stats.h>
23 
24 using process::ConnectionInfo;
25 using process::ConnectionType;
26 using process::ProcessState;
27 using process::ConnectionStatus;
30 using process::g_process_info_constants;
31 
32 
33 const uint32_t AgentUveBase::kUveCountPerTimer;
34 
35 const uint32_t AgentUveBase::kDefaultInterval;
36 
38 
40 
41 
43 
44 AgentUveBase::AgentUveBase(Agent *agent, uint64_t intvl,
45  uint32_t default_intvl, uint32_t incremental_intvl)
46  : vn_uve_table_(NULL), vm_uve_table_(NULL), vrouter_uve_entry_(NULL),
47  prouter_uve_table_(new ProuterUveTable(agent, default_intvl)),
48  interface_uve_table_(NULL),
49  default_interval_(default_intvl),
50  incremental_interval_(incremental_intvl),
51  agent_(agent), bandwidth_intvl_(intvl),
52  vrouter_stats_collector_(new VrouterStatsCollector(
53  *(agent->event_manager()->io_service()),
54  this)) {
55  singleton_ = this;
56 }
57 
59 }
60 
62  const {
63  TagTable *table = agent_->tag_table();
64  TagList::const_iterator it = tl.begin();
65  while (it != tl.end()) {
66  uint32_t type = ((uint32_t)*it >> TagEntry::kTagTypeBitShift);
67  switch (type) {
69  info->application = table->TagName(*it);
70  break;
71  case TagTable::TIER:
72  info->tier = table->TagName(*it);
73  break;
74  case TagTable::SITE:
75  info->site = table->TagName(*it);
76  break;
78  info->deployment = table->TagName(*it);
79  break;
80  case TagTable::LABEL:
81  if (info->fill_type == UveTagData::SET) {
82  info->label_set.insert(table->TagName(*it));
83  break;
84  } else if (info->fill_type == UveTagData::VECTOR) {
85  info->label_vector.push_back(table->TagName(*it));
86  break;
87  }
88  if (!info->labels.empty()) {
89  info->labels += ";";
90  }
91  info->labels.append(table->TagName(*it));
92  break;
94  info->application = table->TagName(*it);
95  break;
96  default:
97  if (info->fill_type == UveTagData::SET) {
98  info->custom_tag_set.insert(table->TagName(*it));
99  break;
100  } else if (info->fill_type == UveTagData::VECTOR) {
101  info->custom_tag_vector.push_back(table->TagName(*it));
102  break;
103  }
104  if (!info->custom_tags.empty()) {
105  info->custom_tags += ";";
106  }
107  info->custom_tags.append(table->TagName(*it));
108  break;
109  }
110  ++it;
111  }
112 }
113 
114 /* Web-UI requires tag-id to returned as hex string. This should start with 0x
115  * and have exactly 8 digits/characters. Zero should be used for filling leading
116  * characters if the tag-id is not 8 digits wide. Web-UI uses this id to do
117  * lookup in API server */
118 string AgentUveBase::IntegerToHexString(uint32_t value) const {
119  std::stringstream ss;
120  ss << "0x" << std::setfill('0') <<
121  std::setw(8) << std::hex << (uint32_t)value;
122  return ss.str();
123 }
124 
126  const {
127  TagList::const_iterator it = tl.begin();
128  while (it != tl.end()) {
129  uint32_t type = ((uint32_t)*it >> TagEntry::kTagTypeBitShift);
130  switch (type) {
132  info->application = IntegerToHexString(*it);
133  break;
134  case TagTable::TIER:
135  info->tier = IntegerToHexString(*it);
136  break;
137  case TagTable::SITE:
138  info->site = IntegerToHexString(*it);
139  break;
141  info->deployment = IntegerToHexString(*it);
142  break;
143  case TagTable::LABEL:
144  if (info->fill_type == UveTagData::SET) {
145  info->label_set.insert(IntegerToHexString(*it));
146  break;
147  } else if (info->fill_type == UveTagData::VECTOR) {
148  info->label_vector.push_back(IntegerToHexString(*it));
149  break;
150  }
151  if (!info->labels.empty()) {
152  info->labels += ";";
153  }
154  info->labels.append(IntegerToHexString(*it));
155  break;
157  info->application = IntegerToHexString(*it);
158  break;
159  default:
160  if (info->fill_type == UveTagData::SET) {
161  info->custom_tag_set.insert(IntegerToHexString(*it));
162  break;
163  } else if (info->fill_type == UveTagData::VECTOR) {
164  info->custom_tag_vector.push_back(IntegerToHexString(*it));
165  break;
166  }
167  if (!info->custom_tags.empty()) {
168  info->custom_tags += ";";
169  }
170  info->custom_tags.append(IntegerToHexString(*it));
171  break;
172  }
173  ++it;
174  }
175 }
176 
178  vn_uve_table_.get()->Shutdown();
179  vm_uve_table_.get()->Shutdown();
180  vrouter_uve_entry_.get()->Shutdown();
181  prouter_uve_table_.get()->Shutdown();
182  interface_uve_table_.get()->Shutdown();
184  vrouter_stats_collector_->Shutdown();
185 }
186 
188  std::string module_id(agent_->module_name());
189  std::string instance_id(agent_->instance_id());
191  boost::asio::io_context &io = *evm->io_service();
192 
195  ConnectionStateManager::
196  GetInstance();
197  agent_->set_connection_state(ConnectionState::GetInstance());
199  module_id, instance_id,
201  this, _1, _2, _3), "ObjectVRouter");
202 }
203 
204 uint8_t AgentUveBase::ExpectedConnections(uint8_t &num_control_nodes,
205  uint8_t &num_dns_servers) {
206  uint8_t count = 0;
207  AgentParam *cfg = agent_->params();
208 
209  for (int i = 0; i < MAX_XMPP_SERVERS; i++) {
210  if (!agent_->controller_ifmap_xmpp_server(i).empty()) {
211  num_control_nodes++;
212  count++;
213  }
214  if (agent_->services() && !agent_->dns_server(i).empty()) {
215  num_dns_servers++;
216  count++;
217  }
218  }
219  //Increment 1 for collector service
220  if (cfg->collector_server_list().size() != 0) {
221  count++;
222  }
223 
224  return count;
225 }
226 
227 void AgentUveBase::UpdateMessage(const ConnectionInfo &cinfo,
228  std::string &message) {
229  if (message.empty()) {
230  message = cinfo.get_type();
231  } else {
232  message += ", " + cinfo.get_type();
233  }
234  const std::string &name(cinfo.get_name());
235  if (!name.empty()) {
236  message += ":" + name;
237  }
238 }
239 
241  if (!agent_ || !agent_->oper_db() || !agent_->oper_db()->global_vrouter()) {
242  return false;
243  }
244  return agent_->oper_db()->global_vrouter()->configured();
245 }
246 
248  (const std::vector<ConnectionInfo> &cinfos,
249  ProcessState::type &pstate, std::string &message) {
250  size_t num_conns = 0;
251  uint8_t num_control_nodes = 0, num_dns_servers = 0;
252  uint8_t down_control_nodes = 0;
253  uint8_t expected_conns = ExpectedConnections(num_control_nodes,
254  num_dns_servers);
255  std::string cup(g_process_info_constants.ConnectionStatusNames.
256  find(ConnectionStatus::UP)->second);
257  bool is_cup = true;
258  bool is_tor_connected = false;
259  string tor_type(g_process_info_constants.ConnectionTypeNames.
260  find(ConnectionType::TOR)->second);
261  // Iterate to determine process connectivity status
262  for (std::vector<ConnectionInfo>::const_iterator it = cinfos.begin();
263  it != cinfos.end(); it++) {
264  const ConnectionInfo &cinfo(*it);
265  const std::string &conn_status(cinfo.get_status());
266  if (cinfo.get_type() == tor_type) {
267  is_tor_connected = true;
268  continue;
269  }
270  /* Don't consider ConnectionType::TOR type for counting connections.
271  * contrail-tor-agent is not supposed to report as Non-Functional when
272  * it is in backup mode, but contrail-tor-agent does not have a way to
273  * figure out that it is in backup mode. Hence for contrail-tor-agent
274  * (both active and backup modes) we don't consider connection to TOR
275  * for reporting Node Status */
276  num_conns++;
277  if (conn_status != cup) {
278  if (cinfo.get_name().compare(0, 13,
279  agent_->xmpp_control_node_prefix()) == 0) {
280  down_control_nodes++;
281  }
282  is_cup = false;
283  UpdateMessage(cinfo, message);
284  }
285  }
286  if ((num_control_nodes == 0) || (num_control_nodes == down_control_nodes)) {
287  pstate = ProcessState::NON_FUNCTIONAL;
288  if ((num_control_nodes == 0) && message.empty()) {
289  message = "No control-nodes configured";
290  }
291  } else if (!is_tor_connected && agent_->tor_agent_enabled()) {
292  // waiting for first TOR config to arrive
293  pstate = ProcessState::NON_FUNCTIONAL;
294  message += " No ToR Config";
295  } else {
296  pstate = ProcessState::FUNCTIONAL;
297  }
298  if (!is_cup) {
299  message += " connection down";
300  }
301  if (!HasSelfConfiguration()) {
302  // waiting for Global vrouter config
303  pstate = ProcessState::NON_FUNCTIONAL;
304  if (message.empty()) {
305  message = "No Configuration for self";
306  } else {
307  message += ", No Configuration for self";
308  }
309  }
310  for (int i = 0; i < MAX_XMPP_SERVERS; i++) {
311  if (!agent_->controller_ifmap_xmpp_server(i).empty()) {
312  if (agent_->stats()->xmpp_reconnects(i) >= 1) {
313  break;
314  }
315  }
316  }
317 
318  if (num_conns != expected_conns) {
319  message += " Number of connections:" + integerToString(num_conns) +
320  ", Expected: " + integerToString(expected_conns);
321  return;
322  }
323  return;
324 }
325 
327  vn_uve_table_.get()->RegisterDBClients();
328  vm_uve_table_.get()->RegisterDBClients();
329  vrouter_uve_entry_.get()->RegisterDBClients();
330  prouter_uve_table_.get()->RegisterDBClients();
331  interface_uve_table_.get()->RegisterDBClients();
332 }
333 
335  vrouter_stats_collector_->InitDone();
336 }
std::string application
std::vector< string > label_vector
ServicesModule * services() const
Definition: agent.cc:973
void Init(boost::asio::io_context &service, const std::string &hostname, const std::string &module, const std::string &instance_id, ProcessStateFn status_cb, std::string table)
FillType fill_type
const std::string & TagName(uint32_t id)
std::string tier
std::string custom_tags
AgentUveBase(Agent *agent, uint64_t intvl, uint32_t default_intvl, uint32_t incremental_intvl)
virtual ~AgentUveBase()
const std::vector< std::string > collector_server_list() const
Definition: agent_param.h:306
boost::scoped_ptr< InterfaceUveTable > interface_uve_table_
const std::string & dns_server(uint8_t idx) const
Definition: agent.h:857
boost::asio::io_context * io_service()
Definition: event_manager.h:42
static const uint32_t kUveCountPerTimer
std::string site
void BuildTagIdsFromList(const TagList &tl, UveTagData *info) const
void set_connection_state(process::ConnectionState *state)
Definition: agent.h:956
boost::scoped_ptr< ProuterUveTable > prouter_uve_table_
static void Init()
bool HasSelfConfiguration() const
std::string IntegerToHexString(uint32_t value) const
void BuildTagNamesFromList(const TagList &tl, UveTagData *info) const
TagTable * tag_table() const
Definition: agent.h:505
OperDB * oper_db() const
Definition: agent.cc:1013
boost::scoped_ptr< VrouterStatsCollector > vrouter_stats_collector_
boost::scoped_ptr< VmUveTableBase > vm_uve_table_
GlobalVrouter * global_vrouter() const
Definition: operdb_init.h:54
uint8_t type
Definition: load_balance.h:109
const std::string & instance_id() const
Definition: agent.h:886
static const std::string integerToString(const NumberType &num)
Definition: string_util.h:19
Definition: agent.h:358
static const uint32_t kDefaultInterval
EventManager * event_manager() const
Definition: agent.h:1103
static const uint64_t kBandwidthInterval
process::ConnectionStateManager * connection_state_manager_
void UpdateMessage(const process::ConnectionInfo &info, std::string &message)
virtual void Shutdown()
std::string deployment
virtual void RegisterDBClients()
static const uint32_t kIncrementalInterval
AgentParam * params() const
Definition: agent.h:1218
const std::string & controller_ifmap_xmpp_server(uint8_t idx) const
Definition: agent.h:730
const std::string & agent_name() const
Definition: agent.h:878
std::vector< string > custom_tag_vector
#define MAX_XMPP_SERVERS
Definition: agent.h:291
std::set< string > custom_tag_set
boost::scoped_ptr< VrouterUveEntryBase > vrouter_uve_entry_
virtual void InitDone()
void VrouterAgentProcessState(const std::vector< process::ConnectionInfo > &c, process::ProcessState::type &state, std::string &message)
const std::string & module_name() const
Definition: agent.h:892
bool configured() const
boost::scoped_ptr< VnUveTableBase > vn_uve_table_
static const uint32_t kTagTypeBitShift
std::string labels
static AgentUveBase * singleton_
uint8_t ExpectedConnections(uint8_t &num_c_nodes, uint8_t &num_d_servers)
static EventManager evm
std::set< string > label_set
std::vector< int > TagList
Definition: agent.h:202