tools: support validating user/custom PKI certs

The virt-pki-validate command can validate the system certificate
directories. The remote driver, however, also supports a standard
per-user certs location, as well as a runtime custom path. This
extends the validation tool to be able to cope with these alternate
locations too.

Reviewed-by: Michal Privoznik <mprivozn@redhat.com>
Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
This commit is contained in:
Daniel P. Berrangé 2024-06-07 11:30:35 +01:00
parent 8e97fd4181
commit acb26f22a1
2 changed files with 197 additions and 79 deletions

View File

@ -15,7 +15,7 @@ SYNOPSIS
======== ========
``virt-pki-validate`` [*OPTION*] ``virt-pki-validate`` [*OPTION*] [trust|server|client]
DESCRIPTION DESCRIPTION
@ -26,6 +26,9 @@ a secure libvirt server or client using the TLS encryption protocol.
It will report any missing certificate or key files on the host. It It will report any missing certificate or key files on the host. It
should be run as root to ensure it can read all the necessary files should be run as root to ensure it can read all the necessary files
With no arguments it will check the trusted CA config, the server
config and the client config. The optional positional argument can
be used to restrict the checks to just one of these three sets.
OPTIONS OPTIONS
======= =======

View File

@ -60,40 +60,77 @@ virPKIValidateFile(const char *file,
} while (0) } while (0)
static bool static bool
virPKIValidateTrust(void) virPKIValidateTrust(bool system, const char *path)
{ {
g_autofree char *cacert = NULL, *cacrl = NULL; g_autofree char *cacert = NULL, *cacrl = NULL;
bool ok = true; bool ok = true;
virNetTLSConfigSystemTrust(&cacert, if (system) {
&cacrl); virNetTLSConfigSystemTrust(&cacert,
&cacrl);
FILE_REQUIRE_EXISTS("TRUST", FILE_REQUIRE_EXISTS("TRUST",
LIBVIRT_PKI_DIR, LIBVIRT_PKI_DIR,
_("Checking if system PKI dir exists"), _("Checking if system PKI dir exists"),
_("The system PKI dir %1$s is usually installed as part of the base filesystem or openssl packages"), _("The system PKI dir %1$s is usually installed as part of the base filesystem or openssl packages"),
LIBVIRT_PKI_DIR); LIBVIRT_PKI_DIR);
FILE_REQUIRE_ACCESS("TRUST", FILE_REQUIRE_ACCESS("TRUST",
LIBVIRT_PKI_DIR, LIBVIRT_PKI_DIR,
_("Checking system PKI dir access"), _("Checking system PKI dir access"),
0, 0, 0755, 0, 0, 0755,
_("The system PKI dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"), _("The system PKI dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"),
LIBVIRT_PKI_DIR, LIBVIRT_PKI_DIR); LIBVIRT_PKI_DIR, LIBVIRT_PKI_DIR);
FILE_REQUIRE_EXISTS("TRUST", FILE_REQUIRE_EXISTS("TRUST",
LIBVIRT_CACERT_DIR, LIBVIRT_CACERT_DIR,
_("Checking if system CA dir exists"), _("Checking if system CA dir exists"),
_("The system CA dir %1$s is usually installed as part of the base filesystem or openssl packages"), _("The system CA dir %1$s is usually installed as part of the base filesystem or openssl packages"),
LIBVIRT_CACERT_DIR); LIBVIRT_CACERT_DIR);
FILE_REQUIRE_ACCESS("TRUST", FILE_REQUIRE_ACCESS("TRUST",
LIBVIRT_CACERT_DIR, LIBVIRT_CACERT_DIR,
_("Checking system CA dir access"), _("Checking system CA dir access"),
0, 0, 0755, 0, 0, 0755,
_("The system CA dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"), _("The system CA dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"),
LIBVIRT_CACERT_DIR, LIBVIRT_CACERT_DIR); LIBVIRT_CACERT_DIR, LIBVIRT_CACERT_DIR);
} else if (path) {
virNetTLSConfigCustomTrust(path,
&cacert,
&cacrl);
FILE_REQUIRE_EXISTS("TRUST",
path,
_("Checking if custom PKI base dir exists"),
_("Create the dir %1$s"),
path);
FILE_REQUIRE_ACCESS("TRUST",
path,
_("Checking custom PKI base dir access"),
getuid(), getgid(), 0700,
_("The PKI base dir %1$s must not be accessible to other users. Run: chown %2$d.%3$d %4$s; chmod 0700 %5$s"),
path, getuid(), getgid(), path, path);
} else {
g_autofree char *pkipath = virNetTLSConfigUserPKIBaseDir();
virNetTLSConfigUserTrust(&cacert,
&cacrl);
FILE_REQUIRE_EXISTS("TRUST",
pkipath,
_("Checking if user PKI base dir exists"),
_("Create the dir %1$s"),
pkipath);
FILE_REQUIRE_ACCESS("TRUST",
pkipath,
_("Checking user PKI base dir access"),
getuid(), getgid(), 0700,
_("The PKI base dir %1$s must not be accessible to other users. Run: chown %2$d.%3$d %4$s; chmod 0700 %5$s"),
pkipath, getuid(), getgid(), pkipath, pkipath);
}
FILE_REQUIRE_EXISTS("TRUST", FILE_REQUIRE_EXISTS("TRUST",
cacert, cacert,
@ -101,56 +138,81 @@ virPKIValidateTrust(void)
_("The machine cannot act as a client or server. See https://libvirt.org/kbase/tlscerts.html#setting-up-a-certificate-authority-ca on how to install %1$s"), _("The machine cannot act as a client or server. See https://libvirt.org/kbase/tlscerts.html#setting-up-a-certificate-authority-ca on how to install %1$s"),
cacert); cacert);
FILE_REQUIRE_ACCESS("TRUST", if (system) {
cacert, FILE_REQUIRE_ACCESS("TRUST",
_("Checking CA cert access"), cacert,
0, 0, 0644, _("Checking CA cert access"),
_("The CA certificate %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s"), 0, 0, 0644,
cacert, cacert, cacert); _("The CA certificate %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s"),
cacert, cacert, cacert);
} else {
FILE_REQUIRE_ACCESS("TRUST",
cacert,
_("Checking CA cert access"),
getuid(), getgid(), 0600,
_("The CA certificate %1$s must not be accessible to other users. As this user, run: chown %2$d.%3$d %4$s; chmod 0600 %5$s"),
cacert, getuid(), getgid(), cacert, cacert);
}
done: done:
return ok; return ok;
} }
static bool static bool
virPKIValidateIdentity(bool isServer) virPKIValidateIdentity(bool isServer, bool system, const char *path)
{ {
g_autofree char *cacert = NULL, *cacrl = NULL; g_autofree char *cacert = NULL, *cacrl = NULL;
g_autofree char *cert = NULL, *key = NULL; g_autofree char *cert = NULL, *key = NULL;
bool ok = true; bool ok = true;
const char *scope = isServer ? "SERVER" : "CLIENT"; const char *scope = isServer ? "SERVER" : "CLIENT";
virNetTLSConfigSystemTrust(&cacert, if (system) {
&cacrl); virNetTLSConfigSystemTrust(&cacert,
virNetTLSConfigSystemIdentity(isServer, &cacrl);
&cert, virNetTLSConfigSystemIdentity(isServer,
&key); &cert,
&key);
FILE_REQUIRE_EXISTS(scope, FILE_REQUIRE_EXISTS(scope,
LIBVIRT_CERT_DIR, LIBVIRT_CERT_DIR,
_("Checking if system cert dir exists"), _("Checking if system cert dir exists"),
_("The system cert dir %1$s is usually installed as part of the libvirt package"), _("The system cert dir %1$s is usually installed as part of the libvirt package"),
LIBVIRT_CERT_DIR); LIBVIRT_CERT_DIR);
FILE_REQUIRE_ACCESS(scope, FILE_REQUIRE_ACCESS(scope,
LIBVIRT_CERT_DIR, LIBVIRT_CERT_DIR,
_("Checking system cert dir access"), _("Checking system cert dir access"),
0, 0, 0755, 0, 0, 0755,
_("The system cert dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"), _("The system cert dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"),
LIBVIRT_PKI_DIR, LIBVIRT_PKI_DIR); LIBVIRT_PKI_DIR, LIBVIRT_PKI_DIR);
FILE_REQUIRE_EXISTS(scope, FILE_REQUIRE_EXISTS(scope,
LIBVIRT_KEY_DIR, LIBVIRT_KEY_DIR,
_("Checking if system key dir exists"), _("Checking if system key dir exists"),
_("The system key dir %1$s is usually installed as part of the libvirt package"), _("The system key dir %1$s is usually installed as part of the libvirt package"),
LIBVIRT_KEY_DIR); LIBVIRT_KEY_DIR);
FILE_REQUIRE_ACCESS(scope, FILE_REQUIRE_ACCESS(scope,
LIBVIRT_KEY_DIR, LIBVIRT_KEY_DIR,
_("Checking system key dir access"), _("Checking system key dir access"),
0, 0, 0755, 0, 0, 0755,
_("The system key dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"), _("The system key dir %1$s must be accessible to all users. As root, run: chown root.root; chmod 0755 %2$s"),
LIBVIRT_KEY_DIR, LIBVIRT_PKI_DIR); LIBVIRT_KEY_DIR, LIBVIRT_PKI_DIR);
} else if (path) {
virNetTLSConfigCustomTrust(path,
&cacert,
&cacrl);
virNetTLSConfigCustomIdentity(path,
isServer,
&cert,
&key);
} else {
virNetTLSConfigUserTrust(&cacert,
&cacrl);
virNetTLSConfigUserIdentity(isServer,
&cert,
&key);
}
FILE_REQUIRE_EXISTS(scope, FILE_REQUIRE_EXISTS(scope,
key, key,
@ -160,14 +222,25 @@ virPKIValidateIdentity(bool isServer)
_("The machine cannot act as a client. See https://libvirt.org/kbase/tlscerts.html#issuing-client-certificates on how to regenerate %1$s"), _("The machine cannot act as a client. See https://libvirt.org/kbase/tlscerts.html#issuing-client-certificates on how to regenerate %1$s"),
key); key);
FILE_REQUIRE_ACCESS(scope, if (system) {
key, FILE_REQUIRE_ACCESS(scope,
_("Checking key access"), key,
0, 0, isServer ? 0600 : 0644, _("Checking key access"),
isServer ? 0, 0, isServer ? 0600 : 0644,
_("The server key %1$s must not be accessible to unprivileged users. As root run: chown root.root %2$s; chmod 0600 %3$s") : isServer ?
_("The client key %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s"), _("The server key %1$s must not be accessible to unprivileged users. As root run: chown root.root %2$s; chmod 0600 %3$s") :
key, key, key); _("The client key %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s"),
key, key, key);
} else {
FILE_REQUIRE_ACCESS(scope,
key,
_("Checking key access"),
getuid(), getgid(), 0600,
isServer ?
_("The server key %1$s must be not be accessible to other users. As this user, run: chown %2$d.%3$d %4$s; chmod 0600 %5$s") :
_("The client key %1$s must be not be accessible to other users. As this user, run: chown %2$d.%3$d %4$s; chmod 0600 %5$s"),
key, getuid(), getgid(), key, key);
}
FILE_REQUIRE_EXISTS(scope, FILE_REQUIRE_EXISTS(scope,
cert, cert,
@ -177,14 +250,25 @@ virPKIValidateIdentity(bool isServer)
_("The machine cannot act as a client. See https://libvirt.org/kbase/tlscerts.html#issuing-client-certificates on how to regenerate %1$s"), _("The machine cannot act as a client. See https://libvirt.org/kbase/tlscerts.html#issuing-client-certificates on how to regenerate %1$s"),
cert); cert);
FILE_REQUIRE_ACCESS(scope, if (system) {
cert, FILE_REQUIRE_ACCESS(scope,
_("Checking cert access"), cert,
0, 0, 0644, _("Checking cert access"),
isServer ? 0, 0, 0644,
_("The server cert %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s") : isServer ?
_("The client cert %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s"), _("The server cert %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s") :
cert, cert, cert); _("The client cert %1$s must be accessible to all users. As root run: chown root.root %2$s; chmod 0644 %3$s"),
cert, cert, cert);
} else {
FILE_REQUIRE_ACCESS(scope,
cert,
_("Checking cert access"),
getuid(), getgid(), 0600,
isServer ?
_("The server cert %1$s must be restricted to this user. As this user, run: chown %2$d.%3$d %4$s; chmod 0600 %5$s") :
_("The client cert %1$s must be restricted to this user. As this user, run: chown %2$d.%3$d %4$s; chmod 0600 %5$s"),
cert, getuid(), getgid(), cert, cert);
}
virValidateCheck(scope, "%s", _("Checking cert properties")); virValidateCheck(scope, "%s", _("Checking cert properties"));
@ -239,6 +323,9 @@ print_usage(const char *progname,
"Validate TLS certificate configuration\n" "Validate TLS certificate configuration\n"
"\n" "\n"
"options:\n" "options:\n"
" -s | --system validate system certificates (default)\n"
" -u | --user validate user certificates\n"
" -p DIR | --path DIR validate custom certificate path\n"
" -h | --help display this help and exit\n" " -h | --help display this help and exit\n"
" -v | --version output version information and exit\n"), " -v | --version output version information and exit\n"),
progname); progname);
@ -247,6 +334,9 @@ print_usage(const char *progname,
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
const char *scope = NULL; const char *scope = NULL;
bool system = false;
bool user = false;
const char *path = NULL;
bool quiet = false; bool quiet = false;
int arg = 0; int arg = 0;
bool ok = true; bool ok = true;
@ -254,6 +344,9 @@ int main(int argc, char **argv)
struct option opt[] = { struct option opt[] = {
{ "help", no_argument, NULL, 'h' }, { "help", no_argument, NULL, 'h' },
{ "version", no_argument, NULL, 'v' }, { "version", no_argument, NULL, 'v' },
{ "system", no_argument, NULL, 's' },
{ "user", no_argument, NULL, 'u' },
{ "path", required_argument, NULL, 'p' },
{ NULL, 0, NULL, 0 }, { NULL, 0, NULL, 0 },
}; };
@ -262,6 +355,18 @@ int main(int argc, char **argv)
while ((arg = getopt_long(argc, argv, "hvsup:", opt, NULL)) != -1) { while ((arg = getopt_long(argc, argv, "hvsup:", opt, NULL)) != -1) {
switch (arg) { switch (arg) {
case 's':
system = true;
break;
case 'u':
user = true;
break;
case 'p':
path = optarg;
break;
case 'v': case 'v':
printf("%s\n", PACKAGE_VERSION); printf("%s\n", PACKAGE_VERSION);
return EXIT_SUCCESS; return EXIT_SUCCESS;
@ -292,14 +397,24 @@ int main(int argc, char **argv)
virValidateSetQuiet(quiet); virValidateSetQuiet(quiet);
if ((system && user) ||
(system && path) ||
(user && path)) {
g_printerr("--system, --user & --path are mutually exclusive\n");
return EXIT_FAILURE;
}
if (!system && !user && !path)
system = true;
if ((!scope || g_str_equal(scope, "trust")) && if ((!scope || g_str_equal(scope, "trust")) &&
!virPKIValidateTrust()) !virPKIValidateTrust(system, path))
ok = false; ok = false;
if ((!scope || g_str_equal(scope, "server")) && if ((!scope || g_str_equal(scope, "server")) &&
!virPKIValidateIdentity(true)) !virPKIValidateIdentity(true, system, path))
ok = false; ok = false;
if ((!scope || g_str_equal(scope, "client")) && if ((!scope || g_str_equal(scope, "client")) &&
!virPKIValidateIdentity(false)) !virPKIValidateIdentity(false, system, path))
ok = false; ok = false;
if (!ok) if (!ok)