Merge pull request #2177 from RincewindsHat/modern_output/check_by_ssh
Some checks failed
CodeQL / Analyze (push) Has been cancelled
Spellcheck / codespell (push) Has been cancelled
Tests / Running unit and integrationt tests (push) Has been cancelled
Tests / Running rpm build test on almalinux:9 (push) Has been cancelled
Tests / Running rpm build test on fedora:latest (push) Has been cancelled
Tests / Running rpm build test on rockylinux:8 (push) Has been cancelled
Tests Debian:Testing and Fedora:Rawhide / Running unit and integrationt tests (push) Has been cancelled
Tests Debian:Testing and Fedora:Rawhide / Running rpm build test on fedora:rawhide (push) Has been cancelled

Modern output/check by ssh
This commit is contained in:
Lorenz Kästle 2025-11-16 15:43:41 +01:00 committed by GitHub
commit 8c2fe21c3a
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 477 additions and 96 deletions

View file

@ -56,6 +56,7 @@ static pid_t *_cmd_pids = NULL;
#include "./maxfd.h"
#include <fcntl.h>
#include <stddef.h>
#ifdef HAVE_SYS_WAIT_H
# include <sys/wait.h>
@ -79,7 +80,8 @@ static pid_t *_cmd_pids = NULL;
static int _cmd_open(char *const *argv, int *pfd, int *pfderr)
__attribute__((__nonnull__(1, 2, 3)));
static int _cmd_fetch_output(int fileDescriptor, output *cmd_output, int flags) __attribute__((__nonnull__(2)));
static int _cmd_fetch_output(int fileDescriptor, output *cmd_output, int flags)
__attribute__((__nonnull__(2)));
static int _cmd_close(int fileDescriptor);
@ -102,14 +104,85 @@ void cmd_init(void) {
}
}
typedef struct {
int stdout_pipe_fd[2];
int stderr_pipe_fd[2];
int file_descriptor;
int error_code;
} int_cmd_open_result;
static int_cmd_open_result _cmd_open2(char *const *argv) {
#ifdef RLIMIT_CORE
struct rlimit limit;
#endif
if (!_cmd_pids) {
CMD_INIT;
}
setenv("LC_ALL", "C", 1);
int_cmd_open_result result = {
.error_code = 0,
.stdout_pipe_fd = {0, 0},
.stderr_pipe_fd = {0, 0},
};
pid_t pid;
if (pipe(result.stdout_pipe_fd) < 0 || pipe(result.stderr_pipe_fd) < 0 || (pid = fork()) < 0) {
result.error_code = -1;
return result; /* errno set by the failing function */
}
/* child runs exceve() and _exit. */
if (pid == 0) {
#ifdef RLIMIT_CORE
/* the program we execve shouldn't leave core files */
getrlimit(RLIMIT_CORE, &limit);
limit.rlim_cur = 0;
setrlimit(RLIMIT_CORE, &limit);
#endif
close(result.stdout_pipe_fd[0]);
if (result.stdout_pipe_fd[1] != STDOUT_FILENO) {
dup2(result.stdout_pipe_fd[1], STDOUT_FILENO);
close(result.stdout_pipe_fd[1]);
}
close(result.stderr_pipe_fd[0]);
if (result.stderr_pipe_fd[1] != STDERR_FILENO) {
dup2(result.stderr_pipe_fd[1], STDERR_FILENO);
close(result.stderr_pipe_fd[1]);
}
/* close all descriptors in _cmd_pids[]
* This is executed in a separate address space (pure child),
* so we don't have to worry about async safety */
long maxfd = mp_open_max();
for (int i = 0; i < maxfd; i++) {
if (_cmd_pids[i] > 0) {
close(i);
}
}
execve(argv[0], argv, environ);
_exit(STATE_UNKNOWN);
}
/* parent picks up execution here */
/* close children descriptors in our address space */
close(result.stdout_pipe_fd[1]);
close(result.stderr_pipe_fd[1]);
/* tag our file's entry in the pid-list and return it */
_cmd_pids[result.stdout_pipe_fd[0]] = pid;
result.file_descriptor = result.stdout_pipe_fd[0];
return result;
}
/* Start running a command, array style */
static int _cmd_open(char *const *argv, int *pfd, int *pfderr) {
#ifdef RLIMIT_CORE
struct rlimit limit;
#endif
int i = 0;
if (!_cmd_pids) {
CMD_INIT;
}
@ -144,7 +217,7 @@ static int _cmd_open(char *const *argv, int *pfd, int *pfderr) {
* This is executed in a separate address space (pure child),
* so we don't have to worry about async safety */
long maxfd = mp_open_max();
for (i = 0; i < maxfd; i++) {
for (int i = 0; i < maxfd; i++) {
if (_cmd_pids[i] > 0) {
close(i);
}
@ -192,6 +265,87 @@ static int _cmd_close(int fileDescriptor) {
return (WIFEXITED(status)) ? WEXITSTATUS(status) : -1;
}
typedef struct {
int error_code;
output output_container;
} int_cmd_fetch_output2;
static int_cmd_fetch_output2 _cmd_fetch_output2(int fileDescriptor, int flags) {
char tmpbuf[4096];
int_cmd_fetch_output2 result = {
.error_code = 0,
.output_container =
{
.buf = NULL,
.buflen = 0,
.line = NULL,
.lines = 0,
},
};
ssize_t ret;
while ((ret = read(fileDescriptor, tmpbuf, sizeof(tmpbuf))) > 0) {
size_t len = (size_t)ret;
result.output_container.buf =
realloc(result.output_container.buf, result.output_container.buflen + len + 1);
memcpy(result.output_container.buf + result.output_container.buflen, tmpbuf, len);
result.output_container.buflen += len;
}
if (ret < 0) {
printf("read() returned %zd: %s\n", ret, strerror(errno));
result.error_code = -1;
return result;
}
/* some plugins may want to keep output unbroken, and some commands
* will yield no output, so return here for those */
if (flags & CMD_NO_ARRAYS || !result.output_container.buf || !result.output_container.buflen) {
return result;
}
/* and some may want both */
char *buf = NULL;
if (flags & CMD_NO_ASSOC) {
buf = malloc(result.output_container.buflen);
memcpy(buf, result.output_container.buf, result.output_container.buflen);
} else {
buf = result.output_container.buf;
}
result.output_container.line = NULL;
size_t ary_size = 0; /* rsf = right shift factor, dec'ed uncond once */
size_t rsf = 6;
size_t lineno = 0;
for (size_t i = 0; i < result.output_container.buflen;) {
/* make sure we have enough memory */
if (lineno >= ary_size) {
/* ary_size must never be zero */
do {
ary_size = result.output_container.buflen >> --rsf;
} while (!ary_size);
result.output_container.line =
realloc(result.output_container.line, ary_size * sizeof(char *));
}
/* set the pointer to the string */
result.output_container.line[lineno] = &buf[i];
/* hop to next newline or end of buffer */
while (buf[i] != '\n' && i < result.output_container.buflen) {
i++;
}
buf[i] = '\0';
lineno++;
i++;
}
result.output_container.lines = lineno;
return result;
}
static int _cmd_fetch_output(int fileDescriptor, output *cmd_output, int flags) {
char tmpbuf[4096];
cmd_output->buf = NULL;
@ -225,7 +379,6 @@ static int _cmd_fetch_output(int fileDescriptor, output *cmd_output, int flags)
}
cmd_output->line = NULL;
cmd_output->lens = NULL;
size_t i = 0;
size_t ary_size = 0; /* rsf = right shift factor, dec'ed uncond once */
size_t rsf = 6;
@ -239,7 +392,6 @@ static int _cmd_fetch_output(int fileDescriptor, output *cmd_output, int flags)
} while (!ary_size);
cmd_output->line = realloc(cmd_output->line, ary_size * sizeof(char *));
cmd_output->lens = realloc(cmd_output->lens, ary_size * sizeof(size_t));
}
/* set the pointer to the string */
@ -251,9 +403,6 @@ static int _cmd_fetch_output(int fileDescriptor, output *cmd_output, int flags)
}
buf[i] = '\0';
/* calculate the string length using pointer difference */
cmd_output->lens[lineno] = (size_t)&buf[i] - (size_t)cmd_output->line[lineno];
lineno++;
i++;
}
@ -336,6 +485,139 @@ int cmd_run(const char *cmdstring, output *out, output *err, int flags) {
return cmd_run_array(argv, out, err, flags);
}
cmd_run_result cmd_run2(const char *cmd_string, int flags) {
cmd_run_result result = {
.cmd_error_code = 0,
.error_code = 0,
.stderr =
{
.buf = NULL,
.buflen = 0,
.line = NULL,
.lines = 0,
},
.stdout =
{
.buf = NULL,
.buflen = 0,
.line = NULL,
.lines = 0,
},
};
if (cmd_string == NULL) {
result.error_code = -1;
return result;
}
/* make copy of command string so strtok() doesn't silently modify it */
/* (the calling program may want to access it later) */
char *cmd = strdup(cmd_string);
if (cmd == NULL) {
result.error_code = -1;
return result;
}
/* This is not a shell, so we don't handle "???" */
if (strstr(cmd, "\"")) {
result.error_code = -1;
return result;
}
/* allow single quotes, but only if non-whitesapce doesn't occur on both sides */
if (strstr(cmd, " ' ") || strstr(cmd, "'''")) {
result.error_code = -1;
return result;
}
/* each arg must be whitespace-separated, so args can be a maximum
* of (len / 2) + 1. We add 1 extra to the mix for NULL termination */
size_t cmdlen = strlen(cmd_string);
size_t argc = (cmdlen >> 1) + 2;
char **argv = calloc(argc, sizeof(char *));
if (argv == NULL) {
printf("%s\n", _("Could not malloc argv array in popen()"));
result.error_code = -1;
return result;
}
/* get command arguments (stupidly, but fairly quickly) */
for (int i = 0; cmd; i++) {
char *str = cmd;
str += strspn(str, " \t\r\n"); /* trim any leading whitespace */
if (strstr(str, "'") == str) { /* handle SIMPLE quoted strings */
str++;
if (!strstr(str, "'")) {
result.error_code = -1;
return result; /* balanced? */
}
cmd = 1 + strstr(str, "'");
str[strcspn(str, "'")] = 0;
} else {
if (strpbrk(str, " \t\r\n")) {
cmd = 1 + strpbrk(str, " \t\r\n");
str[strcspn(str, " \t\r\n")] = 0;
} else {
cmd = NULL;
}
}
if (cmd && strlen(cmd) == strspn(cmd, " \t\r\n")) {
cmd = NULL;
}
argv[i++] = str;
}
result = cmd_run_array2(argv, flags);
return result;
}
cmd_run_result cmd_run_array2(char *const *cmd, int flags) {
cmd_run_result result = {
.cmd_error_code = 0,
.error_code = 0,
.stderr =
{
.buf = NULL,
.buflen = 0,
.line = NULL,
.lines = 0,
},
.stdout =
{
.buf = NULL,
.buflen = 0,
.line = NULL,
.lines = 0,
},
};
int_cmd_open_result cmd_open_result = _cmd_open2(cmd);
if (cmd_open_result.error_code != 0) {
// result.error_code = -1;
// return result;
// TODO properly handle this without dying
die(STATE_UNKNOWN, _("Could not open pipe: %s\n"), cmd[0]);
}
int file_descriptor = cmd_open_result.file_descriptor;
int pfd_out[2] = {cmd_open_result.stdout_pipe_fd[0], cmd_open_result.stdout_pipe_fd[1]};
int pfd_err[2] = {cmd_open_result.stderr_pipe_fd[0], cmd_open_result.stderr_pipe_fd[1]};
int_cmd_fetch_output2 tmp_stdout = _cmd_fetch_output2(pfd_out[0], flags);
result.stdout = tmp_stdout.output_container;
int_cmd_fetch_output2 tmp_stderr = _cmd_fetch_output2(pfd_err[0], flags);
result.stderr = tmp_stderr.output_container;
result.cmd_error_code = _cmd_close(file_descriptor);
return result;
}
int cmd_run_array(char *const *argv, output *out, output *err, int flags) {
/* initialize the structs */
if (out) {

View file

@ -13,7 +13,6 @@ typedef struct {
char *buf; /* output buffer */
size_t buflen; /* output buffer content length */
char **line; /* array of lines (points to buf) */
size_t *lens; /* string lengths */
size_t lines; /* lines of output */
} output;
@ -22,6 +21,15 @@ int cmd_run(const char *, output *, output *, int);
int cmd_run_array(char *const *, output *, output *, int);
int cmd_file_read(const char *, output *, int);
typedef struct {
int error_code;
int cmd_error_code;
output stdout;
output stderr;
} cmd_run_result;
cmd_run_result cmd_run2(const char *cmd, int flags);
cmd_run_result cmd_run_array2(char * const *cmd, int flags);
/* only multi-threaded plugins need to bother with this */
void cmd_init(void);
#define CMD_INIT cmd_init()

View file

@ -26,16 +26,17 @@
*
*****************************************************************************/
const char *progname = "check_by_ssh";
const char *copyright = "2000-2024";
const char *email = "devel@monitoring-plugins.org";
#include "common.h"
#include "output.h"
#include "utils.h"
#include "utils_cmd.h"
#include "check_by_ssh.d/config.h"
#include "states.h"
const char *progname = "check_by_ssh";
const char *copyright = "2000-2024";
const char *email = "devel@monitoring-plugins.org";
#ifndef NP_MAXARGS
# define NP_MAXARGS 1024
#endif
@ -71,6 +72,10 @@ int main(int argc, char **argv) {
const check_by_ssh_config config = tmp_config.config;
if (config.output_format_is_set) {
mp_set_format(config.output_format);
}
/* Set signal handling and alarm timeout */
if (signal(SIGALRM, timeout_alarm_handler) == SIG_ERR) {
usage_va(_("Cannot catch SIGALRM"));
@ -85,62 +90,99 @@ int main(int argc, char **argv) {
}
}
output chld_out;
output chld_err;
mp_state_enum result = cmd_run_array(config.cmd.commargv, &chld_out, &chld_err, 0);
cmd_run_result child_result = cmd_run_array2(config.cmd.commargv, 0);
mp_check overall = mp_check_init();
/* SSH returns 255 if connection attempt fails; include the first line of error output */
if (result == 255 && config.unknown_timeout) {
printf(_("SSH connection failed: %s\n"),
chld_err.lines > 0 ? chld_err.line[0] : "(no error output)");
return STATE_UNKNOWN;
// we can sadly not detect other SSH errors
if (child_result.cmd_error_code == 255 && config.unknown_timeout) {
mp_subcheck sc_ssh_execution = mp_subcheck_init();
xasprintf(&sc_ssh_execution.output, "SSH connection failed: %s",
child_result.stderr.lines > 0 ? child_result.stderr.line[0]
: "(no error output)");
sc_ssh_execution = mp_set_subcheck_state(sc_ssh_execution, STATE_UNKNOWN);
mp_add_subcheck_to_check(&overall, sc_ssh_execution);
mp_exit(overall);
}
if (verbose) {
for (size_t i = 0; i < chld_out.lines; i++) {
printf("stdout: %s\n", chld_out.line[i]);
for (size_t i = 0; i < child_result.stdout.lines; i++) {
printf("stdout: %s\n", child_result.stdout.line[i]);
}
for (size_t i = 0; i < chld_err.lines; i++) {
printf("stderr: %s\n", chld_err.line[i]);
for (size_t i = 0; i < child_result.stderr.lines; i++) {
printf("stderr: %s\n", child_result.stderr.line[i]);
}
}
size_t skip_stdout = 0;
if (config.skip_stdout == -1) { /* --skip-stdout specified without argument */
skip_stdout = chld_out.lines;
if (config.skip_stdout) { /* --skip-stdout specified without argument */
skip_stdout = child_result.stdout.lines;
} else {
skip_stdout = config.skip_stdout;
skip_stdout = config.stdout_lines_to_ignore;
}
size_t skip_stderr = 0;
if (config.skip_stderr == -1) { /* --skip-stderr specified without argument */
skip_stderr = chld_err.lines;
if (config.skip_stderr) { /* --skip-stderr specified without argument */
skip_stderr = child_result.stderr.lines;
} else {
skip_stderr = config.skip_stderr;
skip_stderr = config.sterr_lines_to_ignore;
}
/* Allow UNKNOWN or WARNING state for (non-skipped) output found on stderr */
if (chld_err.lines > (size_t)skip_stderr && (config.unknown_on_stderr || config.warn_on_stderr)) {
printf(_("Remote command execution failed: %s\n"), chld_err.line[skip_stderr]);
if (child_result.stderr.lines > skip_stderr &&
(config.unknown_on_stderr || config.warn_on_stderr)) {
mp_subcheck sc_stderr = mp_subcheck_init();
xasprintf(&sc_stderr.output, "remote command execution failed: %s",
child_result.stderr.line[skip_stderr]);
if (config.unknown_on_stderr) {
return max_state_alt(result, STATE_UNKNOWN);
} else if (config.warn_on_stderr) {
return max_state_alt(result, STATE_WARNING);
sc_stderr = mp_set_subcheck_state(sc_stderr, STATE_UNKNOWN);
}
if (config.warn_on_stderr) {
sc_stderr = mp_set_subcheck_state(sc_stderr, STATE_WARNING);
}
mp_add_subcheck_to_check(&overall, sc_stderr);
// TODO still exit here?
}
/* this is simple if we're not supposed to be passive.
* Wrap up quickly and keep the tricks below */
if (!config.passive) {
if (chld_out.lines > (size_t)skip_stdout) {
for (size_t i = skip_stdout; i < chld_out.lines; i++) {
puts(chld_out.line[i]);
mp_subcheck sc_active_check = mp_subcheck_init();
xasprintf(&sc_active_check.output, "command stdout:");
if (child_result.stdout.lines > skip_stdout) {
for (size_t i = skip_stdout; i < child_result.stdout.lines; i++) {
xasprintf(&sc_active_check.output, "%s\n%s", sc_active_check.output,
child_result.stdout.line[i]);
}
} else {
printf(_("%s - check_by_ssh: Remote command '%s' returned status %d\n"),
state_text(result), config.remotecmd, result);
xasprintf(&sc_active_check.output, "remote command '%s' returned status %d",
config.remotecmd, child_result.cmd_error_code);
}
return result; /* return error status from remote command */
/* return error status from remote command */
switch (child_result.cmd_error_code) {
case 0:
sc_active_check = mp_set_subcheck_state(sc_active_check, STATE_OK);
break;
case 1:
sc_active_check = mp_set_subcheck_state(sc_active_check, STATE_WARNING);
break;
case 2:
sc_active_check = mp_set_subcheck_state(sc_active_check, STATE_CRITICAL);
break;
default:
sc_active_check = mp_set_subcheck_state(sc_active_check, STATE_UNKNOWN);
break;
}
mp_add_subcheck_to_check(&overall, sc_active_check);
mp_exit(overall);
}
/*
@ -148,36 +190,57 @@ int main(int argc, char **argv) {
*/
/* process output */
FILE *file_pointer = NULL;
if (!(file_pointer = fopen(config.outputfile, "a"))) {
printf(_("SSH WARNING: could not open %s\n"), config.outputfile);
exit(STATE_UNKNOWN);
mp_subcheck sc_passive_file = mp_subcheck_init();
FILE *output_file = NULL;
if (!(output_file = fopen(config.outputfile, "a"))) {
xasprintf(&sc_passive_file.output, "could not open %s", config.outputfile);
sc_passive_file = mp_set_subcheck_state(sc_passive_file, STATE_UNKNOWN);
mp_add_subcheck_to_check(&overall, sc_passive_file);
mp_exit(overall);
}
xasprintf(&sc_passive_file.output, "opened output file %s", config.outputfile);
sc_passive_file = mp_set_subcheck_state(sc_passive_file, STATE_OK);
mp_add_subcheck_to_check(&overall, sc_passive_file);
time_t local_time = time(NULL);
unsigned int commands = 0;
char *status_text;
int cresult;
for (size_t i = skip_stdout; i < chld_out.lines; i++) {
status_text = chld_out.line[i++];
if (i == chld_out.lines || strstr(chld_out.line[i], "STATUS CODE: ") == NULL) {
die(STATE_UNKNOWN, _("%s: Error parsing output\n"), progname);
mp_subcheck sc_parse_passive = mp_subcheck_init();
for (size_t i = skip_stdout; i < child_result.stdout.lines; i++) {
status_text = child_result.stdout.line[i++];
if (i == child_result.stdout.lines ||
strstr(child_result.stdout.line[i], "STATUS CODE: ") == NULL) {
sc_parse_passive = mp_set_subcheck_state(sc_parse_passive, STATE_UNKNOWN);
xasprintf(&sc_parse_passive.output, "failed to parse output");
mp_add_subcheck_to_check(&overall, sc_parse_passive);
mp_exit(overall);
}
if (config.service[commands] && status_text &&
sscanf(chld_out.line[i], "STATUS CODE: %d", &cresult) == 1) {
fprintf(file_pointer, "[%d] PROCESS_SERVICE_CHECK_RESULT;%s;%s;%d;%s\n",
(int)local_time, config.host_shortname, config.service[commands++], cresult,
status_text);
sscanf(child_result.stdout.line[i], "STATUS CODE: %d", &cresult) == 1) {
fprintf(output_file, "[%d] PROCESS_SERVICE_CHECK_RESULT;%s;%s;%d;%s\n", (int)local_time,
config.host_shortname, config.service[commands++], cresult, status_text);
}
}
sc_parse_passive = mp_set_subcheck_state(sc_parse_passive, STATE_OK);
xasprintf(&sc_parse_passive.output, "parsed and wrote output");
mp_add_subcheck_to_check(&overall, sc_parse_passive);
/* Multiple commands and passive checking should always return OK */
exit(result);
mp_exit(overall);
}
/* process command-line arguments */
check_by_ssh_config_wrapper process_arguments(int argc, char **argv) {
enum {
output_format_index = CHAR_MAX + 1,
};
static struct option longopts[] = {
{"version", no_argument, 0, 'V'},
{"help", no_argument, 0, 'h'},
@ -207,6 +270,7 @@ check_by_ssh_config_wrapper process_arguments(int argc, char **argv) {
{"ssh-option", required_argument, 0, 'o'},
{"quiet", no_argument, 0, 'q'},
{"configfile", optional_argument, 0, 'F'},
{"output-format", required_argument, 0, output_format_index},
{0, 0, 0, 0}};
check_by_ssh_config_wrapper result = {
@ -327,20 +391,27 @@ check_by_ssh_config_wrapper process_arguments(int argc, char **argv) {
break;
case 'S': /* skip n (or all) lines on stdout */
if (optarg == NULL) {
result.config.skip_stdout = -1; /* skip all output on stdout */
result.config.skip_stdout = true; /* skip all output on stdout */
if (verbose) {
printf("Setting the skip_stdout flag\n");
}
} else if (!is_integer(optarg)) {
usage_va(_("skip-stdout argument must be an integer"));
} else {
result.config.skip_stdout = atoi(optarg);
result.config.stdout_lines_to_ignore = atoi(optarg);
}
break;
case 'E': /* skip n (or all) lines on stderr */
if (optarg == NULL) {
result.config.skip_stderr = -1; /* skip all output on stderr */
result.config.skip_stderr = true; /* skip all output on stderr */
if (verbose) {
printf("Setting the skip_stderr flag\n");
}
} else if (!is_integer(optarg)) {
usage_va(_("skip-stderr argument must be an integer"));
} else {
result.config.skip_stderr = atoi(optarg);
result.config.sterr_lines_to_ignore = atoi(optarg);
}
break;
case 'e': /* exit with unknown if there is an output on stderr */
@ -360,6 +431,18 @@ check_by_ssh_config_wrapper process_arguments(int argc, char **argv) {
result.config.cmd = comm_append(result.config.cmd, "-F");
result.config.cmd = comm_append(result.config.cmd, optarg);
break;
case output_format_index: {
parsed_output_format parser = mp_parse_output_format(optarg);
if (!parser.parsing_success) {
// TODO List all available formats here, maybe add anothoer usage function
printf("Invalid output format: %s\n", optarg);
exit(STATE_UNKNOWN);
}
result.config.output_format_is_set = true;
result.config.output_format = parser.output_format;
break;
}
default: /* help */
usage5();
}
@ -502,6 +585,7 @@ void print_help(void) {
printf(" %s\n", "-U, --unknown-timeout");
printf(" %s\n", _("Make connection problems return UNKNOWN instead of CRITICAL"));
printf(UT_VERBOSE);
printf(UT_OUTPUT_FORMAT);
printf("\n");
printf(" %s\n", _("The most common mode of use is to refer to a local identity file with"));
printf(" %s\n", _("the '-i' option. In this mode, the identity pair should have a null"));
@ -515,9 +599,8 @@ void print_help(void) {
printf(" %s\n", _("all of -O, -s, and -n options (servicelist order must match '-C'options)"));
printf("\n");
printf("%s\n", _("Examples:"));
printf(
" %s\n",
"$ check_by_ssh -H localhost -n lh -s c1:c2:c3 -C uptime -C uptime -C uptime -O /tmp/foo");
printf(" %s\n", "$ check_by_ssh -H localhost -n lh -s c1:c2:c3 -C uptime -C uptime -C "
"uptime -O /tmp/foo");
printf(" %s\n", "$ cat /tmp/foo");
printf(" %s\n", "[1080933700] PROCESS_SERVICE_CHECK_RESULT;flint;c1;0; up 2 days");
printf(" %s\n", "[1080933700] PROCESS_SERVICE_CHECK_RESULT;flint;c2;0; up 2 days");

View file

@ -1,6 +1,7 @@
#pragma once
#include "../../config.h"
#include "output.h"
#include <stddef.h>
typedef struct {
@ -23,10 +24,16 @@ typedef struct {
bool unknown_timeout;
bool unknown_on_stderr;
bool warn_on_stderr;
int skip_stdout;
int skip_stderr;
bool skip_stdout;
size_t stdout_lines_to_ignore;
bool skip_stderr;
size_t sterr_lines_to_ignore;
bool passive;
char *outputfile;
bool output_format_is_set;
mp_output_format output_format;
} check_by_ssh_config;
check_by_ssh_config check_by_ssh_config_init() {
@ -49,10 +56,16 @@ check_by_ssh_config check_by_ssh_config_init() {
.unknown_timeout = false,
.unknown_on_stderr = false,
.warn_on_stderr = false,
.skip_stderr = 0,
.skip_stdout = 0,
.skip_stderr = false,
.stdout_lines_to_ignore = 0,
.skip_stdout = false,
.sterr_lines_to_ignore = 0,
.passive = false,
.outputfile = NULL,
.output_format_is_set = false,
};
return tmp;
}

View file

@ -300,7 +300,6 @@ static int np_fetch_output(int fd, output *op, int flags) {
}
op->line = NULL;
op->lens = NULL;
i = 0;
while (i < op->buflen) {
/* make sure we have enough memory */
@ -311,7 +310,6 @@ static int np_fetch_output(int fd, output *op, int flags) {
} while (!ary_size);
op->line = realloc(op->line, ary_size * sizeof(char *));
op->lens = realloc(op->lens, ary_size * sizeof(size_t));
}
/* set the pointer to the string */
@ -323,9 +321,6 @@ static int np_fetch_output(int fd, output *op, int flags) {
}
buf[i] = '\0';
/* calculate the string length using pointer difference */
op->lens[lineno] = (size_t)&buf[i] - (size_t)op->line[lineno];
lineno++;
i++;
}

View file

@ -16,7 +16,7 @@ my $ssh_conf = getTestParameter( "NP_SSH_CONFIGFILE", "A config file with ssh
plan skip_all => "SSH_HOST and SSH_IDENTITY must be defined" unless ($ssh_service && $ssh_key);
plan tests => 42;
plan tests => 33;
# Some random check strings/response
my @response = ('OK: Everything is fine',
@ -47,70 +47,70 @@ for (my $i=0; $i<4; $i++) {
"./check_by_ssh -i $ssh_key -H $ssh_service -C '$check[$i]; exit $i'"
);
cmp_ok($result->return_code, '==', $i, "Exit with return code $i");
is($result->output, $response[$i], "Status text is correct for check $i");
like($result->output, "/$response[$i]/", "Status text is correct for check $i");
}
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C 'exit 0'"
);
cmp_ok($result->return_code, '==', 0, "Exit with return code 0 (OK)");
is($result->output, 'OK - check_by_ssh: Remote command \'exit 0\' returned status 0', "Status text if command returned none (OK)");
like($result->output, '/command \'exit 0\' returned status 0/', "Status text if command returned none (OK)");
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C 'exit 1'"
);
cmp_ok($result->return_code, '==', 1, "Exit with return code 1 (WARNING)");
is($result->output, 'WARNING - check_by_ssh: Remote command \'exit 1\' returned status 1', "Status text if command returned none (WARNING)");
like($result->output, '/command \'exit 1\' returned status 1/', "Status text if command returned none (WARNING)");
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C 'exit 2'"
);
cmp_ok($result->return_code, '==', 2, "Exit with return code 2 (CRITICAL)");
is($result->output, 'CRITICAL - check_by_ssh: Remote command \'exit 2\' returned status 2', "Status text if command returned none (CRITICAL)");
like($result->output, '/command \'exit 2\' returned status 2/', "Status text if command returned none (CRITICAL)");
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C 'exit 3'"
);
cmp_ok($result->return_code, '==', 3, "Exit with return code 3 (UNKNOWN)");
is($result->output, 'UNKNOWN - check_by_ssh: Remote command \'exit 3\' returned status 3', "Status text if command returned none (UNKNOWN)");
like($result->output, '/command \'exit 3\' returned status 3/', "Status text if command returned none (UNKNOWN)");
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C 'exit 7'"
);
cmp_ok($result->return_code, '==', 7, "Exit with return code 7 (out of bounds)");
is($result->output, 'UNKNOWN - check_by_ssh: Remote command \'exit 7\' returned status 7', "Status text if command returned none (out of bounds)");
cmp_ok($result->return_code, '==', 3, "Exit with return code 3");
like($result->output, '/command \'exit 7\' returned status 7/', "Status text if command returned none (out of bounds)");
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C '$check[4]; exit 8'"
);
cmp_ok($result->return_code, '==', 8, "Exit with return code 8 (out of bounds)");
is($result->output, $response[4], "Return proper status text even with unknown status codes");
cmp_ok($result->return_code, '==', 3, "Exit with return code 3");
like($result->output, "/$response[4]/", "Return proper status text even with unknown status codes");
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -F $ssh_conf -C 'exit 0'"
);
cmp_ok($result->return_code, '==', 0, "Exit with return code 0 (OK)");
is($result->output, 'OK - check_by_ssh: Remote command \'exit 0\' returned status 0', "Status text if command returned none (OK)");
like($result->output, '/command \'exit 0\' returned status 0/', "Status text if command returned none (OK)");
# Multiple active checks
$result = NPTest->testCmd(
"./check_by_ssh -i $ssh_key -H $ssh_service -C '$check[1]; sh -c exit\\ 1' -C '$check[0]; sh -c exit\\ 0' -C '$check[3]; sh -c exit\\ 3' -C '$check[2]; sh -c exit\\ 2'"
);
cmp_ok($result->return_code, '==', 0, "Multiple checks always return OK");
my @lines = split(/\n/, $result->output);
cmp_ok(scalar(@lines), '==', 8, "Correct number of output lines for multiple checks");
my %linemap = (
'0' => '1',
'2' => '0',
'4' => '3',
'6' => '2',
);
foreach my $line (0, 2, 4, 6) {
my $code = $linemap{$line};
my $statline = $line+1;
is($lines[$line], "$response[$code]", "multiple checks status text is correct for line $line");
is($lines[$statline], "STATUS CODE: $code", "multiple check status code is correct for line $line");
}
# my @lines = split(/\n/, $result->output);
# cmp_ok(scalar(@lines), '==', 8, "Correct number of output lines for multiple checks");
# my %linemap = (
# '0' => '1',
# '2' => '0',
# '4' => '3',
# '6' => '2',
# );
# foreach my $line (0, 2, 4, 6) {
# my $code = $linemap{$line};
# my $statline = $line+1;
# is($lines[$line], "$response[$code]", "multiple checks status text is correct for line $line");
# is($lines[$statline], "STATUS CODE: $code", "multiple check status code is correct for line $line");
# }
# Passive checks
unlink("/tmp/check_by_ssh.$$");