stasis_system.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. /*
  2. * Asterisk -- An open source telephony toolkit.
  3. *
  4. * Copyright (C) 2013, Digium, Inc.
  5. *
  6. * Jason Parker <jparker@digium.com>
  7. *
  8. * See http://www.asterisk.org for more information about
  9. * the Asterisk project. Please do not directly contact
  10. * any of the maintainers of this project for assistance;
  11. * the project provides a web site, mailing lists and IRC
  12. * channels for your use.
  13. *
  14. * This program is free software, distributed under the terms of
  15. * the GNU General Public License Version 2. See the LICENSE file
  16. * at the top of the source tree.
  17. */
  18. /*! \file
  19. *
  20. * \brief Stasis Messages and Data Types for System events
  21. *
  22. * \author Jason Parker <jparker@digium.com>
  23. */
  24. /*** MODULEINFO
  25. <support_level>core</support_level>
  26. ***/
  27. #include "asterisk.h"
  28. #include "asterisk/astobj2.h"
  29. #include "asterisk/stasis.h"
  30. #include "asterisk/stasis_system.h"
  31. /*** DOCUMENTATION
  32. <managerEvent language="en_US" name="Registry">
  33. <managerEventInstance class="EVENT_FLAG_SYSTEM">
  34. <synopsis>Raised when an outbound registration completes.</synopsis>
  35. <syntax>
  36. <parameter name="ChannelType">
  37. <para>The type of channel that was registered (or not).</para>
  38. </parameter>
  39. <parameter name="Username">
  40. <para>The username portion of the registration.</para>
  41. </parameter>
  42. <parameter name="Domain">
  43. <para>The address portion of the registration.</para>
  44. </parameter>
  45. <parameter name="Status">
  46. <para>The status of the registration request.</para>
  47. <enumlist>
  48. <enum name="Registered"/>
  49. <enum name="Unregistered"/>
  50. <enum name="Rejected"/>
  51. <enum name="Failed"/>
  52. </enumlist>
  53. </parameter>
  54. <parameter name="Cause">
  55. <para>What caused the rejection of the request, if available.</para>
  56. </parameter>
  57. </syntax>
  58. </managerEventInstance>
  59. </managerEvent>
  60. ***/
  61. /*! \brief The \ref stasis topic for system level changes */
  62. static struct stasis_topic *system_topic;
  63. static struct ast_manager_event_blob *system_registry_to_ami(struct stasis_message *message);
  64. static struct ast_manager_event_blob *cc_available_to_ami(struct stasis_message *message);
  65. static struct ast_manager_event_blob *cc_offertimerstart_to_ami(struct stasis_message *message);
  66. static struct ast_manager_event_blob *cc_requested_to_ami(struct stasis_message *message);
  67. static struct ast_manager_event_blob *cc_requestacknowledged_to_ami(struct stasis_message *message);
  68. static struct ast_manager_event_blob *cc_callerstopmonitoring_to_ami(struct stasis_message *message);
  69. static struct ast_manager_event_blob *cc_callerstartmonitoring_to_ami(struct stasis_message *message);
  70. static struct ast_manager_event_blob *cc_callerrecalling_to_ami(struct stasis_message *message);
  71. static struct ast_manager_event_blob *cc_recallcomplete_to_ami(struct stasis_message *message);
  72. static struct ast_manager_event_blob *cc_failure_to_ami(struct stasis_message *message);
  73. static struct ast_manager_event_blob *cc_monitorfailed_to_ami(struct stasis_message *message);
  74. STASIS_MESSAGE_TYPE_DEFN(ast_network_change_type);
  75. STASIS_MESSAGE_TYPE_DEFN(ast_system_registry_type,
  76. .to_ami = system_registry_to_ami,
  77. );
  78. STASIS_MESSAGE_TYPE_DEFN(ast_cc_available_type,
  79. .to_ami = cc_available_to_ami,
  80. );
  81. STASIS_MESSAGE_TYPE_DEFN(ast_cc_offertimerstart_type,
  82. .to_ami = cc_offertimerstart_to_ami,
  83. );
  84. STASIS_MESSAGE_TYPE_DEFN(ast_cc_requested_type,
  85. .to_ami = cc_requested_to_ami,
  86. );
  87. STASIS_MESSAGE_TYPE_DEFN(ast_cc_requestacknowledged_type,
  88. .to_ami = cc_requestacknowledged_to_ami,
  89. );
  90. STASIS_MESSAGE_TYPE_DEFN(ast_cc_callerstopmonitoring_type,
  91. .to_ami = cc_callerstopmonitoring_to_ami,
  92. );
  93. STASIS_MESSAGE_TYPE_DEFN(ast_cc_callerstartmonitoring_type,
  94. .to_ami = cc_callerstartmonitoring_to_ami,
  95. );
  96. STASIS_MESSAGE_TYPE_DEFN(ast_cc_callerrecalling_type,
  97. .to_ami = cc_callerrecalling_to_ami,
  98. );
  99. STASIS_MESSAGE_TYPE_DEFN(ast_cc_recallcomplete_type,
  100. .to_ami = cc_recallcomplete_to_ami,
  101. );
  102. STASIS_MESSAGE_TYPE_DEFN(ast_cc_failure_type,
  103. .to_ami = cc_failure_to_ami,
  104. );
  105. STASIS_MESSAGE_TYPE_DEFN(ast_cc_monitorfailed_type,
  106. .to_ami = cc_monitorfailed_to_ami,
  107. );
  108. STASIS_MESSAGE_TYPE_DEFN(ast_cluster_discovery_type);
  109. void ast_system_publish_registry(const char *channeltype, const char *username, const char *domain, const char *status, const char *cause)
  110. {
  111. struct ast_json *registry;
  112. struct ast_json_payload *payload;
  113. struct stasis_message *message;
  114. if (!ast_system_registry_type()) {
  115. return;
  116. }
  117. registry = ast_json_pack("{s: s, s: s, s: s, s: s, s: s, s: s}",
  118. "type", "registry",
  119. "channeltype", channeltype,
  120. "username", username,
  121. "domain", domain,
  122. "status", status,
  123. "cause", S_OR(cause, ""));
  124. payload = ast_json_payload_create(registry);
  125. ast_json_unref(registry);
  126. if (!payload) {
  127. return;
  128. }
  129. message = stasis_message_create(ast_system_registry_type(), payload);
  130. ao2_ref(payload, -1);
  131. if (!message) {
  132. return;
  133. }
  134. stasis_publish(ast_system_topic(), message);
  135. ao2_ref(message, -1);
  136. }
  137. static struct ast_manager_event_blob *system_registry_to_ami(struct stasis_message *message)
  138. {
  139. struct ast_json_payload *payload = stasis_message_data(message);
  140. const char *channeltype;
  141. const char *username;
  142. const char *domain;
  143. const char *status;
  144. const char *cause;
  145. RAII_VAR(struct ast_str *, cause_string, ast_str_create(32), ast_free);
  146. if (!cause_string) {
  147. return NULL;
  148. }
  149. channeltype = ast_json_string_get(ast_json_object_get(payload->json, "channeltype"));
  150. username = ast_json_string_get(ast_json_object_get(payload->json, "username"));
  151. domain = ast_json_string_get(ast_json_object_get(payload->json, "domain"));
  152. status = ast_json_string_get(ast_json_object_get(payload->json, "status"));
  153. cause = ast_json_string_get(ast_json_object_get(payload->json, "cause"));
  154. if (!ast_strlen_zero(cause)) {
  155. ast_str_set(&cause_string, 0, "Cause: %s\r\n", cause);
  156. }
  157. return ast_manager_event_blob_create(EVENT_FLAG_SYSTEM, "Registry",
  158. "ChannelType: %s\r\n"
  159. "Username: %s\r\n"
  160. "Domain: %s\r\n"
  161. "Status: %s\r\n"
  162. "%s",
  163. channeltype, username, domain, status, ast_str_buffer(cause_string));
  164. }
  165. static struct ast_manager_event_blob *cc_available_to_ami(struct stasis_message *message)
  166. {
  167. struct ast_json_payload *payload = stasis_message_data(message);
  168. int core_id;
  169. const char *callee;
  170. const char *service;
  171. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  172. callee = ast_json_string_get(ast_json_object_get(payload->json, "callee"));
  173. service = ast_json_string_get(ast_json_object_get(payload->json, "service"));
  174. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCAvailable",
  175. "CoreID: %d\r\n"
  176. "Callee: %s\r\n"
  177. "Service: %s\r\n",
  178. core_id, callee, service);
  179. }
  180. static struct ast_manager_event_blob *cc_offertimerstart_to_ami(struct stasis_message *message)
  181. {
  182. struct ast_json_payload *payload = stasis_message_data(message);
  183. int core_id;
  184. const char *caller;
  185. unsigned int expires;
  186. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  187. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  188. expires = ast_json_integer_get(ast_json_object_get(payload->json, "expires"));
  189. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCOfferTimerStart",
  190. "CoreID: %d\r\n"
  191. "Caller: %s\r\n"
  192. "Expires: %u\r\n",
  193. core_id, caller, expires);
  194. }
  195. static struct ast_manager_event_blob *cc_requested_to_ami(struct stasis_message *message)
  196. {
  197. struct ast_json_payload *payload = stasis_message_data(message);
  198. int core_id;
  199. const char *caller;
  200. const char *callee;
  201. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  202. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  203. callee = ast_json_string_get(ast_json_object_get(payload->json, "callee"));
  204. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCRequested",
  205. "CoreID: %d\r\n"
  206. "Caller: %s\r\n"
  207. "Callee: %s\r\n",
  208. core_id, caller, callee);
  209. }
  210. static struct ast_manager_event_blob *cc_requestacknowledged_to_ami(struct stasis_message *message)
  211. {
  212. struct ast_json_payload *payload = stasis_message_data(message);
  213. int core_id;
  214. const char *caller;
  215. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  216. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  217. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCRequestAcknowledged",
  218. "CoreID: %d\r\n"
  219. "Caller: %s\r\n",
  220. core_id, caller);
  221. }
  222. static struct ast_manager_event_blob *cc_callerstopmonitoring_to_ami(struct stasis_message *message)
  223. {
  224. struct ast_json_payload *payload = stasis_message_data(message);
  225. int core_id;
  226. const char *caller;
  227. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  228. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  229. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCCallerStopMonitoring",
  230. "CoreID: %d\r\n"
  231. "Caller: %s\r\n",
  232. core_id, caller);
  233. }
  234. static struct ast_manager_event_blob *cc_callerstartmonitoring_to_ami(struct stasis_message *message)
  235. {
  236. struct ast_json_payload *payload = stasis_message_data(message);
  237. int core_id;
  238. const char *caller;
  239. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  240. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  241. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCCallerStartMonitoring",
  242. "CoreID: %d\r\n"
  243. "Caller: %s\r\n",
  244. core_id, caller);
  245. }
  246. static struct ast_manager_event_blob *cc_callerrecalling_to_ami(struct stasis_message *message)
  247. {
  248. struct ast_json_payload *payload = stasis_message_data(message);
  249. int core_id;
  250. const char *caller;
  251. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  252. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  253. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCCallerRecalling",
  254. "CoreID: %d\r\n"
  255. "Caller: %s\r\n",
  256. core_id, caller);
  257. }
  258. static struct ast_manager_event_blob *cc_recallcomplete_to_ami(struct stasis_message *message)
  259. {
  260. struct ast_json_payload *payload = stasis_message_data(message);
  261. int core_id;
  262. const char *caller;
  263. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  264. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  265. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCRecallComplete",
  266. "CoreID: %d\r\n"
  267. "Caller: %s\r\n",
  268. core_id, caller);
  269. }
  270. static struct ast_manager_event_blob *cc_failure_to_ami(struct stasis_message *message)
  271. {
  272. struct ast_json_payload *payload = stasis_message_data(message);
  273. int core_id;
  274. const char *caller;
  275. const char *reason;
  276. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  277. caller = ast_json_string_get(ast_json_object_get(payload->json, "caller"));
  278. reason = ast_json_string_get(ast_json_object_get(payload->json, "reason"));
  279. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCFailure",
  280. "CoreID: %d\r\n"
  281. "Caller: %s\r\n"
  282. "Reason: %s\r\n",
  283. core_id, caller, reason);
  284. }
  285. static struct ast_manager_event_blob *cc_monitorfailed_to_ami(struct stasis_message *message)
  286. {
  287. struct ast_json_payload *payload = stasis_message_data(message);
  288. int core_id;
  289. const char *callee;
  290. core_id = ast_json_integer_get(ast_json_object_get(payload->json, "core_id"));
  291. callee = ast_json_string_get(ast_json_object_get(payload->json, "callee"));
  292. return ast_manager_event_blob_create(EVENT_FLAG_CC, "CCMonitorFailed",
  293. "CoreID: %d\r\n"
  294. "Callee: %s\r\n",
  295. core_id, callee);
  296. }
  297. struct stasis_topic *ast_system_topic(void)
  298. {
  299. return system_topic;
  300. }
  301. /*! \brief Cleanup the \ref stasis system level items */
  302. static void stasis_system_cleanup(void)
  303. {
  304. ao2_cleanup(system_topic);
  305. system_topic = NULL;
  306. STASIS_MESSAGE_TYPE_CLEANUP(ast_network_change_type);
  307. STASIS_MESSAGE_TYPE_CLEANUP(ast_system_registry_type);
  308. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_available_type);
  309. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_offertimerstart_type);
  310. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_requested_type);
  311. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_requestacknowledged_type);
  312. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_callerstopmonitoring_type);
  313. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_callerstartmonitoring_type);
  314. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_callerrecalling_type);
  315. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_recallcomplete_type);
  316. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_failure_type);
  317. STASIS_MESSAGE_TYPE_CLEANUP(ast_cc_monitorfailed_type);
  318. STASIS_MESSAGE_TYPE_CLEANUP(ast_cluster_discovery_type);
  319. }
  320. /*! \brief Initialize the system level items for \ref stasis */
  321. int ast_stasis_system_init(void)
  322. {
  323. ast_register_cleanup(stasis_system_cleanup);
  324. system_topic = stasis_topic_create("system:all");
  325. if (!system_topic) {
  326. return 1;
  327. }
  328. if (STASIS_MESSAGE_TYPE_INIT(ast_network_change_type) != 0) {
  329. return -1;
  330. }
  331. if (STASIS_MESSAGE_TYPE_INIT(ast_system_registry_type) != 0) {
  332. return -1;
  333. }
  334. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_available_type) != 0) {
  335. return -1;
  336. }
  337. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_offertimerstart_type) != 0) {
  338. return -1;
  339. }
  340. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_requested_type) != 0) {
  341. return -1;
  342. }
  343. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_requestacknowledged_type) != 0) {
  344. return -1;
  345. }
  346. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_callerstopmonitoring_type) != 0) {
  347. return -1;
  348. }
  349. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_callerstartmonitoring_type) != 0) {
  350. return -1;
  351. }
  352. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_callerrecalling_type) != 0) {
  353. return -1;
  354. }
  355. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_recallcomplete_type) != 0) {
  356. return -1;
  357. }
  358. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_failure_type) != 0) {
  359. return -1;
  360. }
  361. if (STASIS_MESSAGE_TYPE_INIT(ast_cc_monitorfailed_type) != 0) {
  362. return -1;
  363. }
  364. if (STASIS_MESSAGE_TYPE_INIT(ast_cluster_discovery_type) != 0) {
  365. return -1;
  366. }
  367. return 0;
  368. }