PubSubWriter.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  1. #include "PubSubWriter.hpp"
  2. #include "../../osdep/OSUtils.hpp"
  3. #include "CtlUtil.hpp"
  4. #include "member.pb.h"
  5. #include "member_status.pb.h"
  6. #include "network.pb.h"
  7. #include <chrono>
  8. #include <google/cloud/options.h>
  9. #include <google/cloud/pubsub/message.h>
  10. #include <google/cloud/pubsub/publisher.h>
  11. #include <google/cloud/pubsub/topic.h>
  12. #include <opentelemetry/trace/provider.h>
  13. namespace pubsub = ::google::cloud::pubsub;
  14. namespace ZeroTier {
  15. pbmessages::NetworkChange*
  16. networkChangeFromJson(std::string controllerID, const nlohmann::json& oldNetwork, const nlohmann::json& newNetwork);
  17. pbmessages::MemberChange*
  18. memberChangeFromJson(std::string controllerID, const nlohmann::json& oldMember, const nlohmann::json& newMember);
  19. PubSubWriter::PubSubWriter(std::string project, std::string topic, std::string controller_id)
  20. : _controller_id(controller_id)
  21. , _project(project)
  22. , _topic(topic)
  23. {
  24. fprintf(
  25. stderr, "PubSubWriter for controller %s project %s topic %s\n", controller_id.c_str(), project.c_str(),
  26. topic.c_str());
  27. GOOGLE_PROTOBUF_VERIFY_VERSION;
  28. // If PUBSUB_EMULATOR_HOST is set, create the topic if it doesn't exist
  29. const char* emulatorHost = std::getenv("PUBSUB_EMULATOR_HOST");
  30. if (emulatorHost != nullptr) {
  31. create_gcp_pubsub_topic_if_needed(project, topic);
  32. }
  33. auto options =
  34. ::google::cloud::Options {}
  35. .set<pubsub::RetryPolicyOption>(pubsub::LimitedTimeRetryPolicy(std::chrono::seconds(5)).clone())
  36. .set<pubsub::BackoffPolicyOption>(
  37. pubsub::ExponentialBackoffPolicy(std::chrono::milliseconds(100), std::chrono::seconds(2), 1.3).clone());
  38. auto publisher = pubsub::MakePublisherConnection(pubsub::Topic(project, topic), std::move(options));
  39. _publisher = std::make_shared<pubsub::Publisher>(std::move(publisher));
  40. }
  41. PubSubWriter::~PubSubWriter()
  42. {
  43. }
  44. bool PubSubWriter::publishMessage(const std::string& payload, const std::string& frontend)
  45. {
  46. fprintf(stderr, "Publishing message to %s\n", _topic.c_str());
  47. std::vector<std::pair<std::string, std::string> > attributes;
  48. attributes.emplace_back("controller_id", _controller_id);
  49. if (! frontend.empty()) {
  50. attributes.emplace_back("frontend", frontend);
  51. }
  52. auto msg = pubsub::MessageBuilder {}.SetData(payload).SetAttributes(attributes).Build();
  53. auto message_id = _publisher->Publish(std::move(msg)).get();
  54. if (! message_id) {
  55. fprintf(stderr, "Failed to publish message: %s\n", std::move(message_id).status().message().c_str());
  56. return false;
  57. }
  58. fprintf(stderr, "Published message to %s\n", _topic.c_str());
  59. return true;
  60. }
  61. bool PubSubWriter::publishNetworkChange(
  62. const nlohmann::json& oldNetwork,
  63. const nlohmann::json& newNetwork,
  64. const std::string& frontend)
  65. {
  66. fprintf(stderr, "Publishing network change\n");
  67. pbmessages::NetworkChange* nc = networkChangeFromJson(_controller_id, oldNetwork, newNetwork);
  68. std::string payload;
  69. if (! nc->SerializeToString(&payload)) {
  70. fprintf(stderr, "Failed to serialize NetworkChange protobuf message\n");
  71. delete nc;
  72. return false;
  73. }
  74. delete nc;
  75. return publishMessage(payload, frontend);
  76. }
  77. bool PubSubWriter::publishMemberChange(
  78. const nlohmann::json& oldMember,
  79. const nlohmann::json& newMember,
  80. const std::string& frontend)
  81. {
  82. fprintf(stderr, "Publishing member change\n");
  83. pbmessages::MemberChange* mc = memberChangeFromJson(_controller_id, oldMember, newMember);
  84. std::string payload;
  85. if (! mc->SerializeToString(&payload)) {
  86. fprintf(stderr, "Failed to serialize MemberChange protobuf message\n");
  87. delete mc;
  88. return false;
  89. }
  90. delete mc;
  91. return publishMessage(payload, frontend);
  92. }
  93. bool PubSubWriter::publishStatusChange(
  94. std::string frontend,
  95. std::string network_id,
  96. std::string node_id,
  97. std::string os,
  98. std::string arch,
  99. std::string version,
  100. int64_t last_seen)
  101. {
  102. auto provider = opentelemetry::trace::Provider::GetTracerProvider();
  103. auto tracer = provider->GetTracer("PubSubWriter");
  104. auto span = tracer->StartSpan("PubSubWriter::publishStatusChange");
  105. auto scope = tracer->WithActiveSpan(span);
  106. pbmessages::MemberStatus_MemberStatusMetadata* metadata = new pbmessages::MemberStatus_MemberStatusMetadata();
  107. metadata->set_controller_id(_controller_id);
  108. metadata->set_trace_id(""); // TODO: generate a trace ID
  109. pbmessages::MemberStatus ms;
  110. ms.set_network_id(network_id);
  111. ms.set_member_id(node_id);
  112. ms.set_os(os);
  113. ms.set_arch(arch);
  114. ms.set_version(version);
  115. ms.set_timestamp(last_seen);
  116. ms.set_allocated_metadata(metadata);
  117. std::string payload;
  118. if (! ms.SerializeToString(&payload)) {
  119. fprintf(stderr, "Failed to serialize StatusChange protobuf message\n");
  120. return false;
  121. }
  122. return publishMessage(payload, "");
  123. }
  124. pbmessages::NetworkChange_Network* networkFromJson(const nlohmann::json& j)
  125. {
  126. if (! j.is_object()) {
  127. return nullptr;
  128. }
  129. pbmessages::NetworkChange_Network* n = new pbmessages::NetworkChange_Network();
  130. try {
  131. n->set_network_id(OSUtils::jsonString(j["id"], ""));
  132. n->set_name(OSUtils::jsonString(j["name"], ""));
  133. n->set_capabilities(OSUtils::jsonDump(j.value("capabilities", "[]"), -1));
  134. n->set_creation_time(OSUtils::jsonInt(j["creationTime"], 0));
  135. n->set_enable_broadcast(OSUtils::jsonBool(j["enableBroadcast"], false));
  136. for (const auto& p : j["ipAssignmentPools"]) {
  137. if (p.is_object()) {
  138. auto pool = n->add_assignment_pools();
  139. pool->set_start_ip(OSUtils::jsonString(p["ipRangeStart"], ""));
  140. pool->set_end_ip(OSUtils::jsonString(p["ipRangeEnd"], ""));
  141. }
  142. }
  143. n->set_mtu(OSUtils::jsonInt(j["mtu"], 2800));
  144. n->set_multicast_limit(OSUtils::jsonInt(j["multicastLimit"], 32));
  145. n->set_is_private(OSUtils::jsonBool(j["private"], true));
  146. n->set_remote_trace_level(OSUtils::jsonInt(j["remoteTraceLevel"], 0));
  147. n->set_remote_trace_target(OSUtils::jsonString(j["remoteTraceTarget"], ""));
  148. n->set_revision(OSUtils::jsonInt(j["revision"], 0));
  149. for (const auto& p : j["routes"]) {
  150. if (p.is_object()) {
  151. auto r = n->add_routes();
  152. r->set_target(OSUtils::jsonString(p["target"], ""));
  153. r->set_via(OSUtils::jsonString(p["via"], ""));
  154. }
  155. }
  156. std::string rules;
  157. if (j["rules"].is_array()) {
  158. rules = OSUtils::jsonDump(j["rules"], -1);
  159. }
  160. else {
  161. rules = "[]";
  162. }
  163. n->set_rules(rules);
  164. std::string tags;
  165. if (j["tags"].is_array()) {
  166. tags = OSUtils::jsonDump(j["tags"], -1);
  167. }
  168. else {
  169. tags = "[]";
  170. }
  171. n->set_tags(tags);
  172. pbmessages::NetworkChange_IPV4AssignMode* v4am = new pbmessages::NetworkChange_IPV4AssignMode();
  173. if (j["v4AssignMode"].is_object()) {
  174. nlohmann::json am = j["v4AssignMode"];
  175. v4am->set_zt(OSUtils::jsonBool(am["zt"], false));
  176. }
  177. n->set_allocated_ipv4_assign_mode(v4am);
  178. pbmessages::NetworkChange_IPV6AssignMode* v6am = new pbmessages::NetworkChange_IPV6AssignMode();
  179. if (j["v6AssignMode"].is_object()) {
  180. nlohmann::json am = j["v6AssignMode"];
  181. v6am->set_zt(OSUtils::jsonBool(am["zt"], false));
  182. v6am->set_six_plane(OSUtils::jsonBool(am["6plane"], false));
  183. v6am->set_rfc4193(OSUtils::jsonBool(am["rfc4193"], false));
  184. }
  185. n->set_allocated_ipv6_assign_mode(v6am);
  186. nlohmann::json jdns = j["dns"];
  187. if (jdns.is_object()) {
  188. pbmessages::NetworkChange_DNS* dns = new pbmessages::NetworkChange_DNS();
  189. dns->set_domain(jdns.value("domain", ""));
  190. for (const auto& s : jdns["servers"]) {
  191. if (s.is_string()) {
  192. auto server = dns->add_nameservers();
  193. *server = s;
  194. }
  195. }
  196. n->set_allocated_dns(dns);
  197. }
  198. n->set_sso_enabled(OSUtils::jsonBool(j["ssoEnabled"], false));
  199. nlohmann::json ssocfg = j["ssoConfig"];
  200. if (ssocfg.is_object()) {
  201. n->set_sso_provider(OSUtils::jsonString(ssocfg["provider"], ""));
  202. n->set_sso_client_id(OSUtils::jsonString(ssocfg["clientId"], ""));
  203. n->set_sso_authorization_endpoint(OSUtils::jsonString(ssocfg["authorizationEndpoint"], ""));
  204. n->set_sso_issuer(OSUtils::jsonString(ssocfg["issuer"], ""));
  205. n->set_sso_provider(OSUtils::jsonString(ssocfg["provider"], ""));
  206. }
  207. n->set_rules_source(OSUtils::jsonString(j["rulesSource"], ""));
  208. }
  209. catch (const std::exception& e) {
  210. fprintf(stderr, "Exception parsing network JSON: %s\n", e.what());
  211. delete n;
  212. return nullptr;
  213. }
  214. return n;
  215. }
  216. pbmessages::NetworkChange*
  217. networkChangeFromJson(std::string controllerID, const nlohmann::json& oldNetwork, const nlohmann::json& newNetwork)
  218. {
  219. pbmessages::NetworkChange* nc = new pbmessages::NetworkChange();
  220. nc->set_allocated_old(networkFromJson(oldNetwork));
  221. nc->set_allocated_new_(networkFromJson(newNetwork));
  222. nc->set_change_source(pbmessages::NetworkChange_ChangeSource::NetworkChange_ChangeSource_CONTROLLER);
  223. pbmessages::NetworkChange_NetworkChangeMetadata* metadata = new pbmessages::NetworkChange_NetworkChangeMetadata();
  224. metadata->set_controller_id(controllerID);
  225. metadata->set_trace_id(""); // TODO: generate a trace ID
  226. nc->set_allocated_metadata(metadata);
  227. return nc;
  228. }
  229. pbmessages::MemberChange_Member* memberFromJson(const nlohmann::json& j)
  230. {
  231. if (! j.is_object()) {
  232. fprintf(stderr, "memberFromJson: JSON is not an object\n");
  233. return nullptr;
  234. }
  235. fprintf(stderr, "memberFromJSON: %s\n", j.dump().c_str());
  236. pbmessages::MemberChange_Member* m = new pbmessages::MemberChange_Member();
  237. try {
  238. m->set_network_id(OSUtils::jsonString(j["nwid"], ""));
  239. m->set_device_id(OSUtils::jsonString(j["id"], ""));
  240. m->set_identity(OSUtils::jsonString(j["identity"], ""));
  241. m->set_authorized(OSUtils::jsonBool(j["authorized"], false));
  242. if (j["ipAssignments"].is_array()) {
  243. for (const auto& addr : j["ipAssignments"]) {
  244. if (addr.is_string()) {
  245. auto a = m->add_ip_assignments();
  246. std::string address = addr.get<std::string>();
  247. *a = address;
  248. }
  249. }
  250. }
  251. m->set_active_bridge(OSUtils::jsonBool(j["activeBridge"], false));
  252. if (j["tags"].is_array()) {
  253. nlohmann::json tags = j["tags"];
  254. std::string tagsStr = OSUtils::jsonDump(tags, -1);
  255. m->set_tags(tagsStr);
  256. }
  257. else {
  258. nlohmann::json tags = nlohmann::json::array();
  259. std::string tagsStr = OSUtils::jsonDump(tags, -1);
  260. m->set_tags(tagsStr);
  261. }
  262. if (j["capabilities"].is_array()) {
  263. nlohmann::json caps = j["capabilities"];
  264. std::string capsStr = OSUtils::jsonDump(caps, -1);
  265. m->set_capabilities(capsStr);
  266. }
  267. else {
  268. nlohmann::json caps = nlohmann::json::array();
  269. std::string capsStr = OSUtils::jsonDump(caps, -1);
  270. m->set_capabilities(capsStr);
  271. }
  272. m->set_creation_time(OSUtils::jsonInt(j["creationTime"], 0));
  273. m->set_no_auto_assign_ips(OSUtils::jsonBool(j["noAutoAssignIps"], false));
  274. m->set_revision(OSUtils::jsonInt(j["revision"], 0));
  275. m->set_last_authorized_time(OSUtils::jsonInt(j["lastAuthorizedTime"], 0));
  276. m->set_last_deauthorized_time(OSUtils::jsonInt(j["lastDeauthorizedTime"], 0));
  277. m->set_last_authorized_credential_type(OSUtils::jsonString(j["lastAuthorizedCredentialType"], ""));
  278. m->set_last_authorized_credential(OSUtils::jsonString(j["lastAuthorizedCredential"], ""));
  279. m->set_version_major(OSUtils::jsonInt(j["versionMajor"], 0));
  280. m->set_version_minor(OSUtils::jsonInt(j["versionMinor"], 0));
  281. m->set_version_rev(OSUtils::jsonInt(j["versionRev"], 0));
  282. m->set_version_protocol(OSUtils::jsonInt(j["versionProtocol"], 0));
  283. m->set_remote_trace_level(OSUtils::jsonInt(j["remoteTraceLevel"], 0));
  284. m->set_remote_trace_target(OSUtils::jsonString(j["remoteTraceTarget"], ""));
  285. m->set_sso_exempt(OSUtils::jsonBool(j["ssoExempt"], false));
  286. m->set_auth_expiry_time(OSUtils::jsonInt(j["authExpiryTime"], 0));
  287. }
  288. catch (const std::exception& e) {
  289. fprintf(stderr, "Exception parsing member JSON: %s\n", e.what());
  290. delete m;
  291. return nullptr;
  292. }
  293. fprintf(stderr, "memberFromJSON complete\n");
  294. return m;
  295. }
  296. pbmessages::MemberChange*
  297. memberChangeFromJson(std::string controllerID, const nlohmann::json& oldMember, const nlohmann::json& newMember)
  298. {
  299. fprintf(stderr, "memberrChangeFromJson: old: %s\n", oldMember.dump().c_str());
  300. fprintf(stderr, "memberrChangeFromJson: new: %s\n", newMember.dump().c_str());
  301. pbmessages::MemberChange* mc = new pbmessages::MemberChange();
  302. pbmessages::MemberChange_Member* om = memberFromJson(oldMember);
  303. if (om != nullptr) {
  304. mc->set_allocated_old(om);
  305. }
  306. pbmessages::MemberChange_Member* nm = memberFromJson(newMember);
  307. if (nm != nullptr) {
  308. mc->set_allocated_new_(nm);
  309. }
  310. mc->set_change_source(pbmessages::MemberChange_ChangeSource::MemberChange_ChangeSource_CONTROLLER);
  311. pbmessages::MemberChange_MemberChangeMetadata* metadata = new pbmessages::MemberChange_MemberChangeMetadata();
  312. metadata->set_controller_id(controllerID);
  313. metadata->set_trace_id(""); // TODO: generate a trace ID
  314. mc->set_allocated_metadata(metadata);
  315. return mc;
  316. }
  317. } // namespace ZeroTier