res_indications.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. /** @file res_indications.c
  2. *
  3. * Asterisk -- A telephony toolkit for Linux.
  4. *
  5. * Load the indications
  6. *
  7. * Copyright (C) 2002, Pauline Middelink
  8. *
  9. * Pauline Middelink <middelink@polyware.nl>
  10. *
  11. * This program is free software, distributed under the terms of
  12. * the GNU General Public License
  13. *
  14. * Load the country specific dialtones into the asterisk PBX.
  15. */
  16. #include <unistd.h>
  17. #include <string.h>
  18. #include <ctype.h>
  19. #include <stdlib.h>
  20. #include <pthread.h>
  21. #include <errno.h>
  22. #include <sys/types.h>
  23. #include <sys/stat.h>
  24. #include <asterisk/lock.h>
  25. #include <asterisk/file.h>
  26. #include <asterisk/cli.h>
  27. #include <asterisk/logger.h>
  28. #include <asterisk/config.h>
  29. #include <asterisk/channel.h>
  30. #include <asterisk/pbx.h>
  31. #include <asterisk/module.h>
  32. #include <asterisk/translate.h>
  33. #include <asterisk/indications.h>
  34. // Globals
  35. static const char dtext[] = "Indications Configuration";
  36. static const char config[] = "indications.conf";
  37. /*
  38. * Help for commands provided by this module ...
  39. */
  40. static char help_add_indication[] =
  41. "Usage: add indication <country> <indication> \"<tonelist>\"\n"
  42. " Add the given indication to the country.\n";
  43. static char help_remove_indication[] =
  44. "Usage: remove indication <country> <indication>\n"
  45. " Remove the given indication from the country.\n";
  46. static char help_show_indications[] =
  47. "Usage: show indications [<country> ...]\n"
  48. " Show either a condensed for of all country/indications, or the\n"
  49. " indications for the specified countries.\n";
  50. /*
  51. * Implementation of functions provided by this module
  52. */
  53. /*
  54. * ADD INDICATION command stuff
  55. */
  56. static int handle_add_indication(int fd, int argc, char *argv[])
  57. {
  58. struct tone_zone *tz;
  59. int created_country = 0;
  60. if (argc != 5) return RESULT_SHOWUSAGE;
  61. tz = ast_get_indication_zone(argv[2]);
  62. if (!tz) {
  63. /* country does not exist, create it */
  64. ast_log(LOG_NOTICE, "Country '%s' does not exist, creating it.\n",argv[2]);
  65. tz = malloc(sizeof(struct tone_zone));
  66. if (!tz) {
  67. ast_log(LOG_WARNING, "Out of memory\n");
  68. return -1;
  69. }
  70. memset(tz,0,sizeof(struct tone_zone));
  71. strncpy(tz->country,argv[2],sizeof(tz->country)-1);
  72. if (ast_register_indication_country(tz)) {
  73. ast_log(LOG_WARNING, "Unable to register new country\n");
  74. free(tz);
  75. return -1;
  76. }
  77. created_country = 1;
  78. }
  79. if (ast_register_indication(tz,argv[3],argv[4])) {
  80. ast_log(LOG_WARNING, "Unable to register indication %s/%s\n",argv[2],argv[3]);
  81. if (created_country)
  82. ast_unregister_indication_country(argv[2]);
  83. return -1;
  84. }
  85. return 0;
  86. }
  87. /*
  88. * REMOVE INDICATION command stuff
  89. */
  90. static int handle_remove_indication(int fd, int argc, char *argv[])
  91. {
  92. struct tone_zone *tz;
  93. if (argc != 3 && argc != 4) return RESULT_SHOWUSAGE;
  94. if (argc == 3) {
  95. /* remove entiry country */
  96. if (ast_unregister_indication_country(argv[2])) {
  97. ast_log(LOG_WARNING, "Unable to unregister indication country %s\n",argv[2]);
  98. return -1;
  99. }
  100. return 0;
  101. }
  102. tz = ast_get_indication_zone(argv[2]);
  103. if (!tz) {
  104. ast_log(LOG_WARNING, "Unable to unregister indication %s/%s, country does not exists\n",argv[2],argv[3]);
  105. return -1;
  106. }
  107. if (ast_unregister_indication(tz,argv[3])) {
  108. ast_log(LOG_WARNING, "Unable to unregister indication %s/%s\n",argv[2],argv[3]);
  109. return -1;
  110. }
  111. return 0;
  112. }
  113. /*
  114. * SHOW INDICATIONS command stuff
  115. */
  116. static int handle_show_indications(int fd, int argc, char *argv[])
  117. {
  118. struct tone_zone *tz;
  119. char buf[256];
  120. int found_country = 0;
  121. if (ast_mutex_lock(&tzlock)) {
  122. ast_log(LOG_WARNING, "Unable to lock tone_zones list\n");
  123. return 0;
  124. }
  125. if (argc == 2) {
  126. /* no arguments, show a list of countries */
  127. ast_cli(fd,"Country Alias Description\n"
  128. "===========================\n");
  129. for (tz=tone_zones; tz; tz=tz->next) {
  130. ast_cli(fd,"%-7.7s %-7.7s %s\n", tz->country, tz->alias, tz->description);
  131. }
  132. ast_mutex_unlock(&tzlock);
  133. return 0;
  134. }
  135. /* there was a request for specific country(ies), lets humor them */
  136. for (tz=tone_zones; tz; tz=tz->next) {
  137. int i,j;
  138. for (i=2; i<argc; i++) {
  139. if (strcasecmp(tz->country,argv[i])==0 &&
  140. !tz->alias[0]) {
  141. struct tone_zone_sound* ts;
  142. if (!found_country) {
  143. found_country = 1;
  144. ast_cli(fd,"Country Indication PlayList\n"
  145. "=====================================\n");
  146. }
  147. j = snprintf(buf,sizeof(buf),"%-7.7s %-15.15s ",tz->country,"<ringcadance>");
  148. for (i=0; i<tz->nrringcadance; i++) {
  149. j += snprintf(buf+j,sizeof(buf)-j,"%d,",tz->ringcadance[i]);
  150. }
  151. if (tz->nrringcadance) j--;
  152. strncpy(buf+j,"\n",sizeof(buf)-j);
  153. ast_cli(fd,buf);
  154. for (ts=tz->tones; ts; ts=ts->next)
  155. ast_cli(fd,"%-7.7s %-15.15s %s\n",tz->country,ts->name,ts->data);
  156. break;
  157. }
  158. }
  159. }
  160. if (!found_country)
  161. ast_cli(fd,"No countries matched your criteria.\n");
  162. ast_mutex_unlock(&tzlock);
  163. return -1;
  164. }
  165. /*
  166. * Playtones command stuff
  167. */
  168. static int handle_playtones(struct ast_channel *chan, void *data)
  169. {
  170. struct tone_zone_sound *ts;
  171. int res;
  172. if (!data || !((char*)data)[0]) {
  173. ast_log(LOG_NOTICE,"Nothing to play\n");
  174. return -1;
  175. }
  176. ts = ast_get_indication_tone(chan->zone, (const char*)data);
  177. if (ts && ts->data[0])
  178. res = ast_playtones_start(chan, 0, ts->data, 0);
  179. else
  180. res = ast_playtones_start(chan, 0, (const char*)data, 0);
  181. if (res)
  182. ast_log(LOG_NOTICE,"Unable to start playtones\n");
  183. return res;
  184. }
  185. /*
  186. * StopPlaylist command stuff
  187. */
  188. static int handle_stopplaytones(struct ast_channel *chan, void *data)
  189. {
  190. ast_playtones_stop(chan);
  191. return 0;
  192. }
  193. /*
  194. * Load module stuff
  195. */
  196. static int ind_load_module(void)
  197. {
  198. struct ast_config *cfg;
  199. struct ast_variable *v;
  200. char *cxt;
  201. char *c;
  202. struct tone_zone *tones;
  203. const char *country = NULL;
  204. /* that the following cast is needed, is yuk! */
  205. /* yup, checked it out. It is NOT written to. */
  206. cfg = ast_load((char *)config);
  207. if (!cfg)
  208. return 0;
  209. /* Use existing config to populate the Indication table */
  210. cxt = ast_category_browse(cfg, NULL);
  211. while(cxt) {
  212. /* All categories but "general" are considered countries */
  213. if (!strcasecmp(cxt, "general")) {
  214. cxt = ast_category_browse(cfg, cxt);
  215. continue;
  216. }
  217. tones = malloc(sizeof(struct tone_zone));
  218. if (!tones) {
  219. ast_log(LOG_WARNING,"Out of memory\n");
  220. ast_destroy(cfg);
  221. return -1;
  222. }
  223. memset(tones,0,sizeof(struct tone_zone));
  224. strncpy(tones->country,cxt,sizeof(tones->country));
  225. v = ast_variable_browse(cfg, cxt);
  226. while(v) {
  227. if (!strcasecmp(v->name, "description")) {
  228. strncpy(tones->description, v->value, sizeof(tones->description)-1);
  229. } else if (!strcasecmp(v->name,"ringcadance")) {
  230. char *ring,*rings = ast_strdupa(v->value);
  231. c = rings;
  232. ring = strsep(&c,",");
  233. while (ring) {
  234. int *tmp, val;
  235. if (!isdigit(ring[0]) || (val=atoi(ring))==-1) {
  236. ast_log(LOG_WARNING,"Invalid ringcadance given '%s' at line %d.\n",ring,v->lineno);
  237. ring = strsep(&c,",");
  238. continue;
  239. }
  240. tmp = realloc(tones->ringcadance,(tones->nrringcadance+1)*sizeof(int));
  241. if (!tmp) {
  242. ast_log(LOG_WARNING, "Out of memory\n");
  243. ast_destroy(cfg);
  244. return -1;
  245. }
  246. tones->ringcadance = tmp;
  247. tmp[tones->nrringcadance] = val;
  248. tones->nrringcadance++;
  249. /* next item */
  250. ring = strsep(&c,",");
  251. }
  252. } else if (!strcasecmp(v->name,"alias")) {
  253. char *countries = ast_strdupa(v->value);
  254. c = countries;
  255. country = strsep(&c,",");
  256. while (country) {
  257. struct tone_zone* azone = malloc(sizeof(struct tone_zone));
  258. if (!azone) {
  259. ast_log(LOG_WARNING,"Out of memory\n");
  260. ast_destroy(cfg);
  261. return -1;
  262. }
  263. memset(azone,0,sizeof(struct tone_zone));
  264. strncpy(azone->country,country,sizeof(azone->country));
  265. strncpy(azone->alias, cxt, sizeof(azone->alias)-1);
  266. if (ast_register_indication_country(azone)) {
  267. ast_log(LOG_WARNING, "Unable to register indication alias at line %d.\n",v->lineno);
  268. free(tones);
  269. }
  270. /* next item */
  271. country = strsep(&c,",");
  272. }
  273. } else {
  274. // add tone to country
  275. struct tone_zone_sound *ps,*ts;
  276. for (ps=NULL,ts=tones->tones; ts; ps=ts, ts=ts->next) {
  277. if (strcasecmp(v->name,ts->name)==0) {
  278. /* already there */
  279. ast_log(LOG_NOTICE,"Duplicate entry '%s', skipped.\n",v->name);
  280. goto out;
  281. }
  282. }
  283. /* not there, add it to the back */
  284. ts = malloc(sizeof(struct tone_zone_sound));
  285. if (!ts) {
  286. ast_log(LOG_WARNING, "Out of memory\n");
  287. ast_destroy(cfg);
  288. return -1;
  289. }
  290. ts->next = NULL;
  291. ts->name = strdup(v->name);
  292. ts->data = strdup(v->value);
  293. if (ps)
  294. ps->next = ts;
  295. else
  296. tones->tones = ts;
  297. }
  298. out: v = v->next;
  299. }
  300. if (tones->description[0] || tones->alias[0] || tones->tones) {
  301. if (ast_register_indication_country(tones)) {
  302. ast_log(LOG_WARNING, "Unable to register indication at line %d.\n",v->lineno);
  303. free(tones);
  304. }
  305. } else free(tones);
  306. cxt = ast_category_browse(cfg, cxt);
  307. }
  308. /* determine which country is the default */
  309. country = ast_variable_retrieve(cfg,"general","country");
  310. if (!country || !*country || ast_set_indication_country(country))
  311. ast_log(LOG_WARNING,"Unable to set the default country (for indication tones)\n");
  312. ast_destroy(cfg);
  313. return 0;
  314. }
  315. /*
  316. * CLI entries for commands provided by this module
  317. */
  318. static struct ast_cli_entry add_indication_cli =
  319. { { "add", "indication", NULL }, handle_add_indication,
  320. "Add the given indication to the country", help_add_indication,
  321. NULL };
  322. static struct ast_cli_entry remove_indication_cli =
  323. { { "remove", "indication", NULL }, handle_remove_indication,
  324. "Remove the given indication from the country", help_remove_indication,
  325. NULL };
  326. static struct ast_cli_entry show_indications_cli =
  327. { { "show", "indications", NULL }, handle_show_indications,
  328. "Show a list of all country/indications", help_show_indications,
  329. NULL };
  330. /*
  331. * Standard module functions ...
  332. */
  333. int unload_module(void)
  334. {
  335. /* remove the registed indications... */
  336. ast_unregister_indication_country(NULL);
  337. /* and the functions */
  338. ast_cli_unregister(&add_indication_cli);
  339. ast_cli_unregister(&remove_indication_cli);
  340. ast_cli_unregister(&show_indications_cli);
  341. ast_unregister_application("Playlist");
  342. ast_unregister_application("StopPlaylist");
  343. return 0;
  344. }
  345. int load_module(void)
  346. {
  347. if (ind_load_module()) return -1;
  348. ast_cli_register(&add_indication_cli);
  349. ast_cli_register(&remove_indication_cli);
  350. ast_cli_register(&show_indications_cli);
  351. ast_register_application("Playtones", handle_playtones, "Play a tone list","Play a tone list, either registered (through indications.conf) or a direct list of tones and durations.");
  352. ast_register_application("StopPlaytones", handle_stopplaytones, "Stop playing a tone list","Stop playing a tone list");
  353. return 0;
  354. }
  355. int reload(void)
  356. {
  357. /* remove the registed indications... */
  358. ast_unregister_indication_country(NULL);
  359. return ind_load_module();
  360. }
  361. char *description(void)
  362. {
  363. /* that the following cast is needed, is yuk! */
  364. return (char*)dtext;
  365. }
  366. int usecount(void)
  367. {
  368. return 0;
  369. }
  370. char *key()
  371. {
  372. return ASTERISK_GPL_KEY;
  373. }