Rework signature verification output.

- Signature verification output is confusing.  The main problem is
    the terminology.  It talks about "good signatures", "good
    checksums", and "bad checksums," but it is unclear what good or
    bad means, and what a checksum is.  Instead, talk about
    "authenticated signatures," "unauthenticated signatures," and
    completely drop the term "checksum" and just say that the
    certificate for the alleged signer is missing.

  - Fixes #4.
This commit is contained in:
Neal H. Walfield 2024-10-30 09:41:04 +01:00
parent daebb8f0c5
commit 973b249f88
No known key found for this signature in database
GPG Key ID: 6863C9AD5B4D22D3
4 changed files with 72 additions and 68 deletions

View File

@ -1371,7 +1371,7 @@ when I run sq sign --signer-file alice.pgp hello.txt --output signed1.txt
when I run sq sign --signer-file bob.pgp --append signed1.txt --output signed2.txt
when I run sq verify signed2.txt --signer-file alice-cert.pgp --signer-file bob-cert.pgp
then stdout contains "hello, world"
then stderr matches regex 2.good signatures
then stderr matches regex 2.authenticated signatures
~~~
## Merge signed files
@ -1392,7 +1392,7 @@ when I run sq sign --signer-file bob.pgp hello.txt --output signed2.txt
when I run sq sign --merge=signed2.txt signed1.txt --output merged.txt
when I run sq verify merged.txt --signer-file alice-cert.pgp --signer-file bob-cert.pgp
then stdout contains "hello, world"
then stderr matches regex 2.good signatures
then stderr matches regex 2.authenticated signatures
~~~

View File

@ -215,11 +215,11 @@ pub struct VHelper<'c, 'store, 'rstore>
aead_algo: Option<AEADAlgorithm>,
// Tracks the signatures encountered.
good_signatures: usize,
good_checksums: usize,
unknown_checksums: usize,
authenticated_signatures: usize,
unauthenticated_signatures: usize,
uncheckable_signatures: usize,
bad_signatures: usize,
bad_checksums: usize,
broken_keys: usize,
broken_signatures: usize,
quiet: bool,
}
@ -236,11 +236,11 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
trusted: HashSet::new(),
sym_algo: None,
aead_algo: None,
good_signatures: 0,
good_checksums: 0,
unknown_checksums: 0,
authenticated_signatures: 0,
unauthenticated_signatures: 0,
uncheckable_signatures: 0,
broken_keys: 0,
bad_signatures: 0,
bad_checksums: 0,
broken_signatures: 0,
quiet: sq.quiet,
}
@ -254,8 +254,8 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
}
fn print_status(&self) {
fn p(s: &mut String, what: &str, quantity: usize) {
if quantity > 0 {
fn p(s: &mut String, what: &str, threshold: usize, quantity: usize) {
if quantity >= threshold {
use std::fmt::Write;
use crate::output::pluralize::Pluralize;
let dirty = ! s.is_empty();
@ -267,12 +267,12 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
}
let mut status = String::new();
p(&mut status, "good signature", self.good_signatures);
p(&mut status, "unauthenticated checksum", self.good_checksums);
p(&mut status, "unknown checksum", self.unknown_checksums);
p(&mut status, "bad signature", self.bad_signatures);
p(&mut status, "bad checksum", self.bad_checksums);
p(&mut status, "broken signatures", self.broken_signatures);
p(&mut status, "authenticated signature", 0, self.authenticated_signatures);
p(&mut status, "unauthenticated signature", 1, self.unauthenticated_signatures);
p(&mut status, "uncheckable signature", 1, self.uncheckable_signatures);
p(&mut status, "bad signature", 1, self.bad_signatures);
p(&mut status, "bad key", 1, self.broken_keys);
p(&mut status, "broken signatures", 1, self.broken_signatures);
if ! status.is_empty() {
wprintln!("{}.", status);
}
@ -300,45 +300,46 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
.expect("missing key checksum has an issuer")
.to_string();
let what = match sig.level() {
0 => "checksum".into(),
n => format!("level {} notarizing checksum", n),
0 => "signature".into(),
n => format!("level {} notarization", n),
};
wprintln!("No cert to check {} from {}", what, issuer);
wprintln!("Can't authenticate {} allegedly made by {}: \
missing certificate.",
what, issuer);
self.sq.hint(format_args!(
"Consider trying to retrieve the key from the network \
using:"))
"Consider searching for the certificate using:"))
.sq().arg("network").arg("search")
.arg(issuer)
.done();
self.unknown_checksums += 1;
self.uncheckable_signatures += 1;
continue;
},
Err(UnboundKey { cert, error, .. }) => {
wprintln!("Signing key on {} is not bound:",
cert.fingerprint());
print_error_chain(error);
self.bad_checksums += 1;
self.broken_keys += 1;
continue;
},
Err(BadKey { ka, error, .. }) => {
wprintln!("Signing key on {} is bad:",
ka.cert().fingerprint());
print_error_chain(error);
self.bad_checksums += 1;
self.broken_keys += 1;
continue;
},
Err(BadSignature { sig, ka, error }) => {
let issuer = ka.fingerprint().to_string();
let what = match sig.level() {
0 => "checksum".into(),
n => format!("level {} notarizing checksum", n),
0 => "signature".into(),
n => format!("level {} notarizing signature", n),
};
wprintln!("Error verifying {} from {}:",
wprintln!("Error verifying {} made by {}:",
what, issuer);
print_error_chain(error);
self.bad_checksums += 1;
self.bad_signatures += 1;
continue;
}
};
@ -351,10 +352,10 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
.unwrap_or_else(|_| "<unknown>".to_string());
// Direct trust.
let mut trusted = self.trusted.contains(&issuer);
let mut authenticated = self.trusted.contains(&issuer);
let mut prefix = "";
let trust_roots = self.sq.trust_roots();
if ! trusted && ! trust_roots.is_empty() {
if ! authenticated && ! trust_roots.is_empty() {
prefix = " ";
// Web of trust.
@ -439,9 +440,9 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
.collect::<Vec<UserID>>();
if authenticated_userids.is_empty() {
trusted = false;
authenticated = false;
} else {
trusted = true;
authenticated = true;
// If we managed to authenticate the
// signers user ID, prefer that one.
@ -474,25 +475,27 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
});
let level = sig.level();
match (level == 0, trusted) {
match (level == 0, authenticated) {
(true, true) => {
wprintln!(indent=prefix,
"Good signature from {} ({:?})",
"Authenticated signature made by {} ({:?})",
label, signer_userid);
}
(false, true) => {
wprintln!(indent=prefix,
"Good level {} notarization from {} ({:?})",
"Authenticated level {} notarization \
made by {} ({:?})",
level, label, signer_userid);
}
(true, false) => {
wprintln!(indent=prefix,
"Unauthenticated checksum from {} ({:?})",
"Can't authenticate signature made by {} ({:?}): \
the certificate can't be authenticated.",
label, signer_userid);
self.sq.hint(format_args!(
"After checking that {} belongs to {:?}, \
you can authenticate the binding using:",
you can mark it as authenticated using:",
cert_fpr, signer_userid))
.sq().arg("pki").arg("link").arg("add")
.arg("--cert").arg(cert_fpr)
@ -501,13 +504,14 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
}
(false, false) => {
wprintln!(indent=prefix,
"Unauthenticated level {} notarizing \
checksum from {} ({:?})",
"Can't authenticate level {} notarization \
made by {} ({:?}): the certificate \
can't be authenticated.",
level, label, signer_userid);
self.sq.hint(format_args!(
"After checking that {} belongs to {:?}, \
you can authenticate the binding using:",
you can mark it as authenticated using:",
cert_fpr, signer_userid))
.sq().arg("pki").arg("link").arg("add")
.arg("--cert").arg(cert_fpr)
@ -516,10 +520,10 @@ impl<'c, 'store, 'rstore> VHelper<'c, 'store, 'rstore> {
}
};
if trusted {
self.good_signatures += 1;
if authenticated {
self.authenticated_signatures += 1;
} else {
self.good_checksums += 1;
self.unauthenticated_signatures += 1;
}
qprintln!("");
@ -604,13 +608,13 @@ impl<'c, 'store, 'rstore> VerificationHelper for VHelper<'c, 'store, 'rstore>
}
}
if self.good_signatures >= self.signatures {
if self.authenticated_signatures >= self.signatures {
Ok(())
} else {
if ! self.quiet {
self.print_status();
}
Err(anyhow::anyhow!("Verification failed: could not fully \
Err(anyhow::anyhow!("Verification failed: could not \
authenticate any signatures"))
}
}

View File

@ -42,7 +42,7 @@ fn sq_verify(sq: &Sq,
trust_roots: &[&str],
signer_files: &[&str],
msg_pgp: &str,
good_sigs: usize, good_checksums: usize)
authenticated_sigs: usize, unauthenticated_sigs: usize)
{
let mut cmd = sq.command();
for trust_root in trust_roots {
@ -65,12 +65,12 @@ fn sq_verify(sq: &Sq,
let stdout = String::from_utf8_lossy(&output.stdout).to_string();
let stderr = String::from_utf8_lossy(&output.stderr).to_string();
if good_sigs > 0 {
if authenticated_sigs > 0 {
assert!(status.success(),
"\nstdout:\n{}\nstderr:\n{}",
stdout, stderr);
assert!(stderr.contains(&format!("{} good signature",
good_sigs)),
assert!(stderr.contains(&format!("{} authenticated signature",
authenticated_sigs)),
"stdout:\n{}\nstderr:\n{}",
stdout, stderr);
} else {
@ -79,9 +79,9 @@ fn sq_verify(sq: &Sq,
stdout, stderr);
}
if good_checksums > 0 {
assert!(stderr.contains(&format!("{} unauthenticated checksum",
good_checksums)),
if unauthenticated_sigs > 0 {
assert!(stderr.contains(&format!("{} unauthenticated signature",
unauthenticated_sigs)),
"stdout:\n{}\nstderr:\n{}", stdout, stderr);
}
}

View File

@ -948,7 +948,7 @@ fn sq_sign_using_cert_store() -> Result<()> {
assert!(! output.status.success(),
"stdout:\n{}\nstderr: {}", stdout, stderr);
assert!(stderr.contains("No cert to check checksum from "),
assert!(stderr.contains("missing certificate."),
"stdout:\n{}\nstderr: {}", stdout, stderr);
assert!(stderr.contains("Error: Verification failed"),
"stdout:\n{}\nstderr: {}", stdout, stderr);
@ -966,7 +966,7 @@ fn sq_sign_using_cert_store() -> Result<()> {
// The default trust model says that certificates from the
// certificate store are not authenticated.
assert!(stderr.contains("Unauthenticated checksum from "),
assert!(stderr.contains("the certificate can't be authenticated."),
"stdout:\n{}\nstderr: {}", stdout, stderr);
assert!(stderr.contains("Error: Verification failed"),
"stdout:\n{}\nstderr: {}", stdout, stderr);
@ -985,9 +985,9 @@ fn sq_sign_using_cert_store() -> Result<()> {
// The default trust model says that certificates from the
// certificate store are not authenticated.
assert!(stderr.contains("Good signature from "),
assert!(stderr.contains("Authenticated signature made by "),
"stdout:\n{}\nstderr: {}", stdout, stderr);
assert!(stderr.contains("1 good signature."),
assert!(stderr.contains("1 authenticated signature."),
"stdout:\n{}\nstderr: {}", stdout, stderr);
Ok(())
@ -1108,7 +1108,7 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[], &[&alice_pgp], &msg_pgp);
assert!(! output.0.success());
// But, one good signature is enough.
// But, one authenticated signature is enough.
let output = sq_verify(&sq, &[], &[&alice_pgp, &bob_pgp], &msg_pgp);
assert!(output.0.success());
}
@ -1120,7 +1120,7 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[], &[], &msg_pgp);
assert!(! output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Unauthenticated checksum from "),
assert!(output.2.contains("the certificate can't be authenticated."),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
@ -1129,7 +1129,7 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[&alice_fpr], &[], &msg_pgp);
assert!(! output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Unauthenticated checksum from "),
assert!(output.2.contains("the certificate can't be authenticated."),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
}
@ -1139,7 +1139,7 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[&bob_fpr], &[], &msg_pgp);
assert!(output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Good signature from "),
assert!(output.2.contains("Authenticated signature made by "),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
@ -1147,7 +1147,7 @@ fn sq_verify_wot() -> Result<()> {
&sq, &[&alice_fpr, &bob_fpr], &[], &msg_pgp);
assert!(output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Good signature from "),
assert!(output.2.contains("Authenticated signature made by "),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
}
@ -1161,7 +1161,7 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[&alice_fpr], &[], &msg_pgp);
assert!(! output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Unauthenticated checksum from "),
assert!(output.2.contains("the certificate can't be authenticated."),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
}
@ -1176,10 +1176,10 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[&alice_fpr], &[], &msg_pgp);
assert!(! output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Unauthenticated checksum from "),
assert!(output.2.contains("the certificate can't be authenticated."),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
assert!(output.2.contains("3 unauthenticated checksums"),
assert!(output.2.contains("3 unauthenticated signatures"),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
}
@ -1193,10 +1193,10 @@ fn sq_verify_wot() -> Result<()> {
let output = sq_verify(&sq, &[&alice_fpr], &[], &msg_pgp);
assert!(output.0.success(),
"stdout:\n{}\nstderr:\n{}", output.1, output.2);
assert!(output.2.contains("Good signature from "),
assert!(output.2.contains("Authenticated signature made by "),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
assert!(output.2.contains("1 good signature, 2 unauthenticated checksums"),
assert!(output.2.contains("1 authenticated signature, 2 unauthenticated signatures"),
"stdout:\n{}\nstderr:\n{}",
output.1, output.2);
}