Merge pull request #655 from dothebart/statsd_fix_division_by_zero
[collectd.git] / src / utils_cmd_putnotif.c
1 /**
2  * collectd - src/utils_cms_putnotif.c
3  * Copyright (C) 2008  Florian octo Forster
4  *
5  * This program is free software; you can redistribute it and/or modify it
6  * under the terms of the GNU General Public License as published by the
7  * Free Software Foundation; only version 2 of the License is applicable.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License along
15  * with this program; if not, write to the Free Software Foundation, Inc.,
16  * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
17  *
18  * Author:
19  *   Florian octo Forster <octo at verplant.org>
20  **/
21
22 #include "collectd.h"
23 #include "common.h"
24 #include "plugin.h"
25
26 #include "utils_parse_option.h"
27
28 #define print_to_socket(fh, ...) \
29   if (fprintf (fh, __VA_ARGS__) < 0) { \
30     char errbuf[1024]; \
31     WARNING ("handle_putnotif: failed to write to socket #%i: %s", \
32         fileno (fh), sstrerror (errno, errbuf, sizeof (errbuf))); \
33     return -1; \
34   }
35
36 static int set_option_severity (notification_t *n, const char *value)
37 {
38   if (strcasecmp (value, "Failure") == 0)
39     n->severity = NOTIF_FAILURE;
40   else if (strcasecmp (value, "Warning") == 0)
41     n->severity = NOTIF_WARNING;
42   else if (strcasecmp (value, "Okay") == 0)
43     n->severity = NOTIF_OKAY;
44   else
45     return (-1);
46
47   return (0);
48 } /* int set_option_severity */
49
50 static int set_option_time (notification_t *n, const char *value)
51 {
52   char *endptr = NULL;
53   double tmp;
54
55   errno = 0;
56   tmp = strtod (value, &endptr);
57   if ((errno != 0)         /* Overflow */
58       || (endptr == value) /* Invalid string */
59       || (endptr == NULL)  /* This should not happen */
60       || (*endptr != 0))   /* Trailing chars */
61     return (-1);
62
63   n->time = DOUBLE_TO_CDTIME_T (tmp);
64
65   return (0);
66 } /* int set_option_time */
67
68 static int set_option (notification_t *n, const char *option, const char *value)
69 {
70   if ((n == NULL) || (option == NULL) || (value == NULL))
71     return (-1);
72
73   DEBUG ("utils_cmd_putnotif: set_option (option = %s, value = %s);",
74       option, value);
75
76   /* Add a meta option in the form: <type>:<key> */
77   if (option[0] != '\0' && option[1] == ':') {
78     /* Refuse empty key */
79     if (option[2] == '\0')
80       return (1);
81
82     if (option[0] == 's')
83       return plugin_notification_meta_add_string (n, option + 2, value);
84     else
85       return (1);
86   }
87
88   if (strcasecmp ("severity", option) == 0)
89     return (set_option_severity (n, value));
90   else if (strcasecmp ("time", option) == 0)
91     return (set_option_time (n, value));
92   else if (strcasecmp ("message", option) == 0)
93     sstrncpy (n->message, value, sizeof (n->message));
94   else if (strcasecmp ("host", option) == 0)
95     sstrncpy (n->host, value, sizeof (n->host));
96   else if (strcasecmp ("plugin", option) == 0)
97     sstrncpy (n->plugin, value, sizeof (n->plugin));
98   else if (strcasecmp ("plugin_instance", option) == 0)
99     sstrncpy (n->plugin_instance, value, sizeof (n->plugin_instance));
100   else if (strcasecmp ("type", option) == 0)
101     sstrncpy (n->type, value, sizeof (n->type));
102   else if (strcasecmp ("type_instance", option) == 0)
103     sstrncpy (n->type_instance, value, sizeof (n->type_instance));
104   else
105     return (1);
106
107   return (0);
108 } /* int set_option */
109
110 int handle_putnotif (FILE *fh, char *buffer)
111 {
112   char *command;
113   notification_t n;
114   int status;
115
116   if ((fh == NULL) || (buffer == NULL))
117     return (-1);
118
119   DEBUG ("utils_cmd_putnotif: handle_putnotif (fh = %p, buffer = %s);",
120       (void *) fh, buffer);
121
122   command = NULL;
123   status = parse_string (&buffer, &command);
124   if (status != 0)
125   {
126     print_to_socket (fh, "-1 Cannot parse command.\n");
127     return (-1);
128   }
129   assert (command != NULL);
130
131   if (strcasecmp ("PUTNOTIF", command) != 0)
132   {
133     print_to_socket (fh, "-1 Unexpected command: `%s'.\n", command);
134     return (-1);
135   }
136
137   memset (&n, '\0', sizeof (n));
138
139   status = 0;
140   while (*buffer != 0)
141   {
142     char *key;
143     char *value;
144
145     status = parse_option (&buffer, &key, &value);
146     if (status != 0)
147     {
148       print_to_socket (fh, "-1 Malformed option.\n");
149       break;
150     }
151
152     status = set_option (&n, key, value);
153     if (status != 0)
154     {
155       print_to_socket (fh, "-1 Error parsing option `%s'\n", key);
156       break;
157     }
158   } /* for (i) */
159
160   /* Check for required fields and complain if anything is missing. */
161   if ((status == 0) && (n.severity == 0))
162   {
163     print_to_socket (fh, "-1 Option `severity' missing.\n");
164     status = -1;
165   }
166   if ((status == 0) && (n.time == 0))
167   {
168     print_to_socket (fh, "-1 Option `time' missing.\n");
169     status = -1;
170   }
171   if ((status == 0) && (strlen (n.message) == 0))
172   {
173     print_to_socket (fh, "-1 No message or message of length 0 given.\n");
174     status = -1;
175   }
176
177   /* If status is still zero the notification is fine and we can finally
178    * dispatch it. */
179   if (status == 0)
180   {
181     plugin_dispatch_notification (&n);
182     print_to_socket (fh, "0 Success\n");
183   }
184
185   return (0);
186 } /* int handle_putnotif */
187
188 /* vim: set shiftwidth=2 softtabstop=2 tabstop=8 : */