readconf.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. /* $Id: readconf.c,v 1.22 2005/03/20 16:17:22 dijkstra Exp $ */
  2. /*
  3. * Copyright (c) 2001-2004 Willem Dijkstra
  4. * All rights reserved.
  5. *
  6. * Redistribution and use in source and binary forms, with or without
  7. * modification, are permitted provided that the following conditions
  8. * are met:
  9. *
  10. * - Redistributions of source code must retain the above copyright
  11. * notice, this list of conditions and the following disclaimer.
  12. * - Redistributions in binary form must reproduce the above
  13. * copyright notice, this list of conditions and the following
  14. * disclaimer in the documentation and/or other materials provided
  15. * with the distribution.
  16. *
  17. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  18. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  19. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
  20. * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
  21. * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
  22. * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
  23. * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  24. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  25. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  26. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
  27. * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  28. * POSSIBILITY OF SUCH DAMAGE.
  29. *
  30. */
  31. #include <string.h>
  32. #include "conf.h"
  33. #include "data.h"
  34. #include "error.h"
  35. #include "lex.h"
  36. #include "net.h"
  37. #include "symon.h"
  38. #include "xmalloc.h"
  39. __BEGIN_DECLS
  40. int read_host_port(struct muxlist *, struct mux *, struct lex *);
  41. int read_symon_args(struct mux *, struct lex *);
  42. int read_monitor(struct muxlist *, struct lex *);
  43. __END_DECLS
  44. const char *default_symux_port = SYMUX_PORT;
  45. /* <hostname> (port|:|,| ) <number> */
  46. int
  47. read_host_port(struct muxlist * mul, struct mux * mux, struct lex * l)
  48. {
  49. char muxname[_POSIX2_LINE_MAX];
  50. lex_nexttoken(l);
  51. if (!getip(l->token, AF_INET) && !getip(l->token, AF_INET6)) {
  52. warning("%.200s:%d: could not resolve '%.200s'",
  53. l->filename, l->cline, l->token);
  54. return 0;
  55. }
  56. mux->addr = xstrdup((const char *) &res_host);
  57. /* check for port statement */
  58. if (!lex_nexttoken(l))
  59. return 1;
  60. if (l->op == LXT_PORT || l->op == LXT_COMMA)
  61. lex_nexttoken(l);
  62. if (l->type != LXY_NUMBER) {
  63. lex_ungettoken(l);
  64. mux->port = xstrdup(default_symux_port);
  65. } else {
  66. mux->port = xstrdup((const char *) l->token);
  67. }
  68. bzero(&muxname, sizeof(muxname));
  69. snprintf(&muxname[0], sizeof(muxname), "%s %s", mux->addr, mux->port);
  70. if (rename_mux(mul, mux, muxname) == NULL) {
  71. warning("%.200s:%d: monitored data for host '%.200s' has already been specified",
  72. l->filename, l->cline, muxname);
  73. return 0;
  74. }
  75. return 1;
  76. }
  77. /* parse "<cpu(arg)|mem|if(arg)|io(arg)|debug|pf|pfq(arg)|proc(arg)>", end condition == "}" */
  78. int
  79. read_symon_args(struct mux * mux, struct lex * l)
  80. {
  81. char sn[_POSIX2_LINE_MAX];
  82. char sa[_POSIX2_LINE_MAX];
  83. int st;
  84. EXPECT(l, LXT_BEGIN)
  85. while (lex_nexttoken(l) && l->op != LXT_END) {
  86. switch (l->op) {
  87. case LXT_CPU:
  88. case LXT_IF:
  89. case LXT_IO:
  90. case LXT_IO1:
  91. case LXT_MEM:
  92. case LXT_PF:
  93. case LXT_PFQ:
  94. case LXT_MBUF:
  95. case LXT_DEBUG:
  96. case LXT_PROC:
  97. case LXT_SENSOR:
  98. st = token2type(l->op);
  99. strncpy(&sn[0], l->token, _POSIX2_LINE_MAX);
  100. /* parse arg */
  101. lex_nexttoken(l);
  102. if (l->op == LXT_OPEN) {
  103. lex_nexttoken(l);
  104. if (l->op == LXT_CLOSE) {
  105. parse_error(l, "<stream argument>");
  106. return 0;
  107. }
  108. strncpy(&sa[0], l->token, _POSIX2_LINE_MAX);
  109. lex_nexttoken(l);
  110. if (l->op != LXT_CLOSE) {
  111. parse_error(l, ")");
  112. return 0;
  113. }
  114. } else {
  115. lex_ungettoken(l);
  116. sa[0] = '\0';
  117. }
  118. if (strlen(sa) > (SYMON_PS_ARGLEN - 1)) {
  119. warning("%.200s:%d: argument '%.200s' too long for network format, "
  120. "will send leading " SYMON_PS_ARGLENSTR " chars only",
  121. l->filename, l->cline, sa);
  122. }
  123. if ((add_mux_stream(mux, st, sa)) == NULL) {
  124. warning("%.200s:%d: stream %.200s(%.200s) redefined",
  125. l->filename, l->cline, sn, sa);
  126. return 0;
  127. }
  128. break; /* LXT_CPU/IF/IO/IO1/MEM/PF/PFQ/MBUF/DEBUG/PROC/SENSOR */
  129. case LXT_COMMA:
  130. break;
  131. default:
  132. parse_error(l, "{cpu|mem|if|io|pf|debug|sensor}");
  133. return 0;
  134. break;
  135. }
  136. }
  137. return 1;
  138. }
  139. /* parse monitor <args> stream [to] <host>:<port> */
  140. int
  141. read_monitor(struct muxlist * mul, struct lex * l)
  142. {
  143. struct mux *mux;
  144. mux = add_mux(mul, SYMON_UNKMUX);
  145. /* parse [stream(streamarg)]+ */
  146. if (!read_symon_args(mux, l))
  147. return 0;
  148. lex_nexttoken(l);
  149. /* parse [every x seconds]? */
  150. if (l->op == LXT_EVERY) {
  151. lex_nexttoken(l);
  152. if (l->op == LXT_SECOND) {
  153. symon_interval = 1;
  154. } else if (l->type == LXY_NUMBER) {
  155. symon_interval = l->value;
  156. lex_nexttoken(l);
  157. if (l->op != LXT_SECONDS) {
  158. parse_error(l, "seconds");
  159. }
  160. } else {
  161. parse_error(l, "<number> ");
  162. return 0;
  163. }
  164. lex_nexttoken(l);
  165. }
  166. /* parse [stream to] */
  167. if (l->op != LXT_STREAM) {
  168. parse_error(l, "stream");
  169. return 0;
  170. }
  171. lex_nexttoken(l);
  172. if (l->op != LXT_TO)
  173. lex_ungettoken(l);
  174. /* parse [host [port]?] */
  175. return read_host_port(mul, mux, l);
  176. }
  177. /* Read symon.conf */
  178. int
  179. read_config_file(struct muxlist *muxlist, char *filename)
  180. {
  181. struct lex *l;
  182. struct mux *mux;
  183. SLIST_INIT(muxlist);
  184. l = open_lex(filename);
  185. while (lex_nexttoken(l)) {
  186. /* expecting keyword now */
  187. switch (l->op) {
  188. case LXT_MONITOR:
  189. if (!read_monitor(muxlist, l))
  190. return 0;
  191. break;
  192. default:
  193. parse_error(l, "monitor");
  194. return 0;
  195. break;
  196. }
  197. }
  198. /* sanity checks */
  199. SLIST_FOREACH(mux, muxlist, muxes) {
  200. if (strncmp(SYMON_UNKMUX, mux->name, sizeof(SYMON_UNKMUX)) == 0) {
  201. /* mux was not initialised for some reason */
  202. return 0;
  203. }
  204. if (SLIST_EMPTY(&mux->sl)) {
  205. warning("%.200s: no monitors selected for mux '%.200s'",
  206. l->filename, mux->name);
  207. return 0;
  208. }
  209. }
  210. if (symon_interval < SYMON_DEFAULT_INTERVAL) {
  211. warning("%.200s: monitoring set to every %d s", l->filename, symon_interval);
  212. }
  213. close_lex(l);
  214. return 1;
  215. }