/* * pkd_util.c -- pkd utilities * * (c) 2014, 2018 Jon Simons */ #include #include #include #include #include #include #include #include "pkd_client.h" #include "pkd_util.h" /** * @brief runs system(3); exits if that is interrupted with SIGINT/QUIT * @returns 0 upon success, non-zero otherwise */ int system_checked(const char *cmd) { int rc = system(cmd); if (WIFSIGNALED(rc) && ((WTERMSIG(rc) == SIGINT) || (WTERMSIG(rc) == SIGQUIT))) { exit(1); } if (rc == -1) { return -1; } return WEXITSTATUS(rc); } static int bin_exists(const char *binary) { char bin[1024] = { 0 }; snprintf(&bin[0], sizeof(bin), "type %s 1>/dev/null 2>/dev/null", binary); return (system_checked(bin) == 0); } static int is_openssh_client_new_enough(void) { int rc = -1; FILE *fp = NULL; char version[1024] = { 0 }; int version_ok = 0; unsigned long int major = 0; char *tmp = NULL; fp = popen("ssh -V 2>&1", "r"); if (fp == NULL) { fprintf(stderr, "failed to get OpenSSH client version\n"); goto done; } if (fgets(&version[0], sizeof(version), fp) == NULL) { fprintf(stderr, "failed to get OpenSSH client version string\n"); goto errfgets; } /* "OpenSSH_...." */ if (strlen(version) < 11) { goto errversion; } /* Extract major. */ major = strtoul(version + 8, &tmp, 10); if ((tmp == (version + 8)) || ((errno = ERANGE) && (major == ULONG_MAX)) || ((errno != 0) && (major == 0)) || ((major < 1) || (major > 100))) { fprintf(stderr, "failed to parse OpenSSH client version, " "errno %d\n", errno); goto errversion; } if (major < 7) { fprintf(stderr, "error: minimum OpenSSH client version " "required is 7, found: %ld\n", major); goto errversion; } version_ok = 1; errversion: errfgets: rc = pclose(fp); if (rc != 0) { fprintf(stderr, "failed to get OpenSSH client version: %d\n", rc); } done: return version_ok; } int is_openssh_client_enabled(void) { return (bin_exists(OPENSSH_BINARY) && bin_exists(OPENSSH_KEYGEN) && is_openssh_client_new_enough()); } int is_dropbear_client_enabled(void) { return (bin_exists(DROPBEAR_BINARY) && bin_exists(DROPBEAR_KEYGEN)); }