Struct sequoia_openpgp::cert::CertParser
source · pub struct CertParser<'a> { /* private fields */ }
Expand description
An iterator over a sequence of certificates, i.e., an OpenPGP keyring.
The source of packets is a fallible iterator over Packet
s. In
this way, it is possible to propagate parse errors.
A CertParser
returns each TPK
or TSK
that it encounters.
Note: if you don’t actually need all of the certificates, it is
usually faster to use a RawCertParser
and only fully parse and
canonicalize those certificates that are relevant.
A CertParser
’s behavior can be modeled using a simple state
machine.
In the first and initial state, it looks for the start of a
certificate, a Public Key
packet or a Secret Key
packet.
When it encounters such a packet it buffers it, and transitions to
the second state. Any other packet or an error causes it to emit
an error and stay in the same state. When the source of packets
is exhausted, it enters the End
state.
In the second state, it looks for packets that belong to a
certificate’s body. If it encounters a valid body packet, then it
buffers it and stays in the same state. If it encounters the
start of a certificate, then it emits the buffered certificate,
buffers the packet, and stays in the same state. If it encounters
an invalid packet (e.g., a Literal Data
packet), it emits two
items, the buffered certificate, and an error, and then it
transitions back to the initial state. When the source of packets
is exhausted, it emits the buffered certificate and enters the end
state.
In the end state, it emits None
.
Invalid Packet / Error
,------------------------.
v |
Not a +---------+ +---------+
Start .-> | Looking | -------------> | Looking | <-. Cert
of Cert | | for | Start | for | | Body
Packet | | Start | of Cert | Cert | | Packet
/ Error `-- | of Cert | Packet | Body | --'
+---------+ .-> +---------+
| | | |
| `------' |
| Start of Cert Packet |
| |
EOF | +-----+ | EOF
`------> | End | <---------'
+-----+
| ^
`--'
The parser does not recurse into containers, thus when it
encounters a container like a Compressed Data
Packet, it will
return an error even if the container contains a valid
certificate.
The parser considers unknown packets to be valid body packets.
(In a Cert
, these show up as Unknown
components.) The
goal is to provide some future compatibility.
§Examples
Print information about all certificates in a keyring:
use sequoia_openpgp as openpgp;
use openpgp::parse::Parse;
use openpgp::parse::PacketParser;
use openpgp::cert::prelude::*;
let ppr = PacketParser::from_bytes(&keyring)?;
for certo in CertParser::from(ppr) {
match certo {
Ok(cert) => {
println!("Key: {}", cert.fingerprint());
for ua in cert.userids() {
println!(" User ID: {}", ua.userid());
}
}
Err(err) => {
eprintln!("Error reading keyring: {}", err);
}
}
}
When an invalid packet is encountered, an error is returned and parsing continues:
use sequoia_openpgp as openpgp;
use openpgp::cert::prelude::*;
use openpgp::packet::prelude::*;
use openpgp::types::DataFormat;
let mut lit = Literal::new(DataFormat::Text);
lit.set_body(b"test".to_vec());
let (alice, _) =
CertBuilder::general_purpose(None, Some("alice@example.org"))
.generate()?;
let (bob, _) =
CertBuilder::general_purpose(None, Some("bob@example.org"))
.generate()?;
let mut packets : Vec<Packet> = Vec::new();
packets.extend(alice.clone());
packets.push(lit.clone().into());
packets.push(lit.clone().into());
packets.extend(bob.clone());
let r : Vec<Result<Cert>> = CertParser::from(packets).collect();
assert_eq!(r.len(), 4);
assert_eq!(r[0].as_ref().unwrap().fingerprint(), alice.fingerprint());
assert!(r[1].is_err());
assert!(r[2].is_err());
assert_eq!(r[3].as_ref().unwrap().fingerprint(), bob.fingerprint());
Implementations§
source§impl<'a> CertParser<'a>
impl<'a> CertParser<'a>
sourcepub fn from_iter<I, J>(iter: I) -> Selfwhere
I: 'a + IntoIterator<Item = J>,
J: 'a + Into<Result<Packet>>,
<I as IntoIterator>::IntoIter: Send + Sync,
pub fn from_iter<I, J>(iter: I) -> Selfwhere
I: 'a + IntoIterator<Item = J>,
J: 'a + Into<Result<Packet>>,
<I as IntoIterator>::IntoIter: Send + Sync,
Creates a CertParser
from a Result<Packet>
iterator.
Note: because we implement From<Packet>
for
Result<Packet>
, it is possible to pass in an iterator over
Packet
s.
§Examples
From a Vec<Packet>
:
use sequoia_openpgp as openpgp;
use openpgp::cert::prelude::*;
use openpgp::packet::prelude::*;
for certo in CertParser::from_iter(packets) {
match certo {
Ok(cert) => {
println!("Key: {}", cert.fingerprint());
for ua in cert.userids() {
println!(" User ID: {}", ua.userid());
}
}
Err(err) => {
eprintln!("Error reading keyring: {}", err);
}
}
}
sourcepub fn unvalidated_cert_filter<F>(self, filter: F) -> Self
pub fn unvalidated_cert_filter<F>(self, filter: F) -> Self
Filters the Certs prior to validation.
By default, the CertParser
only returns valdiated Cert
s.
Checking that a certificate’s self-signatures are valid,
however, is computationally expensive, and not always
necessary. For example, when looking for a small number of
certificates in a large keyring, most certificates can be
immediately discarded. That is, it is more efficient to
filter, validate, and double check, than to validate and
filter. (It is necessary to double check, because the check
might have been on an invalid part. For example, if searching
for a key with a particular Key ID, a matching key might not
have any self signatures.)
If the CertParser
gave out unvalidated Cert
s, and provided
an interface to validate them, then the caller could implement
this check-validate-double-check pattern. Giving out
unvalidated Cert
s, however, is dangerous: inevitably, a
Cert
will be used without having been validated in a context
where it should have been.
This function avoids this class of bugs while still providing
a mechanism to filter Cert
s prior to validation: the caller
provides a callback that is invoked on the unvalidated
Cert
. If the callback returns true
, then the parser
validates the Cert
, and invokes the callback a second time
to make sure the Cert
is really wanted. If the callback
returns false, then the Cert
is skipped.
Note: calling this function multiple times on a single
CertParser
will not replace the existing filter, but install
multiple filters.
§Examples
use sequoia_openpgp as openpgp;
use openpgp::cert::prelude::*;
for certr in CertParser::from(ppr)
.unvalidated_cert_filter(|cert, _| {
for component in cert.keys() {
if component.key().keyid() == some_keyid {
return true;
}
}
false
})
{
match certr {
Ok(cert) => {
// The Cert contains the subkey.
}
Err(err) => {
eprintln!("Error reading keyring: {}", err);
}
}
}
Trait Implementations§
source§impl<'a> Default for CertParser<'a>
impl<'a> Default for CertParser<'a>
source§impl<'a> From<PacketParserResult<'a>> for CertParser<'a>
impl<'a> From<PacketParserResult<'a>> for CertParser<'a>
source§fn from(ppr: PacketParserResult<'a>) -> Self
fn from(ppr: PacketParserResult<'a>) -> Self
Initializes a CertParser
from a PacketParser
.
source§impl<'a> Iterator for CertParser<'a>
impl<'a> Iterator for CertParser<'a>
source§fn next(&mut self) -> Option<Self::Item>
fn next(&mut self) -> Option<Self::Item>
source§fn next_chunk<const N: usize>(
&mut self,
) -> Result<[Self::Item; N], IntoIter<Self::Item, N>>where
Self: Sized,
fn next_chunk<const N: usize>(
&mut self,
) -> Result<[Self::Item; N], IntoIter<Self::Item, N>>where
Self: Sized,
iter_next_chunk
)N
values. Read more1.0.0 · source§fn size_hint(&self) -> (usize, Option<usize>)
fn size_hint(&self) -> (usize, Option<usize>)
1.0.0 · source§fn count(self) -> usizewhere
Self: Sized,
fn count(self) -> usizewhere
Self: Sized,
1.0.0 · source§fn last(self) -> Option<Self::Item>where
Self: Sized,
fn last(self) -> Option<Self::Item>where
Self: Sized,
source§fn advance_by(&mut self, n: usize) -> Result<(), NonZero<usize>>
fn advance_by(&mut self, n: usize) -> Result<(), NonZero<usize>>
iter_advance_by
)n
elements. Read more1.0.0 · source§fn nth(&mut self, n: usize) -> Option<Self::Item>
fn nth(&mut self, n: usize) -> Option<Self::Item>
n
th element of the iterator. Read more1.28.0 · source§fn step_by(self, step: usize) -> StepBy<Self>where
Self: Sized,
fn step_by(self, step: usize) -> StepBy<Self>where
Self: Sized,
1.0.0 · source§fn chain<U>(self, other: U) -> Chain<Self, <U as IntoIterator>::IntoIter>
fn chain<U>(self, other: U) -> Chain<Self, <U as IntoIterator>::IntoIter>
1.0.0 · source§fn zip<U>(self, other: U) -> Zip<Self, <U as IntoIterator>::IntoIter>where
Self: Sized,
U: IntoIterator,
fn zip<U>(self, other: U) -> Zip<Self, <U as IntoIterator>::IntoIter>where
Self: Sized,
U: IntoIterator,
source§fn intersperse_with<G>(self, separator: G) -> IntersperseWith<Self, G>
fn intersperse_with<G>(self, separator: G) -> IntersperseWith<Self, G>
iter_intersperse
)separator
between adjacent items of the original iterator. Read more1.0.0 · source§fn map<B, F>(self, f: F) -> Map<Self, F>
fn map<B, F>(self, f: F) -> Map<Self, F>
1.0.0 · source§fn filter<P>(self, predicate: P) -> Filter<Self, P>
fn filter<P>(self, predicate: P) -> Filter<Self, P>
1.0.0 · source§fn filter_map<B, F>(self, f: F) -> FilterMap<Self, F>
fn filter_map<B, F>(self, f: F) -> FilterMap<Self, F>
1.0.0 · source§fn enumerate(self) -> Enumerate<Self>where
Self: Sized,
fn enumerate(self) -> Enumerate<Self>where
Self: Sized,
1.0.0 · source§fn skip_while<P>(self, predicate: P) -> SkipWhile<Self, P>
fn skip_while<P>(self, predicate: P) -> SkipWhile<Self, P>
1.0.0 · source§fn take_while<P>(self, predicate: P) -> TakeWhile<Self, P>
fn take_while<P>(self, predicate: P) -> TakeWhile<Self, P>
1.57.0 · source§fn map_while<B, P>(self, predicate: P) -> MapWhile<Self, P>
fn map_while<B, P>(self, predicate: P) -> MapWhile<Self, P>
1.0.0 · source§fn skip(self, n: usize) -> Skip<Self>where
Self: Sized,
fn skip(self, n: usize) -> Skip<Self>where
Self: Sized,
n
elements. Read more1.0.0 · source§fn take(self, n: usize) -> Take<Self>where
Self: Sized,
fn take(self, n: usize) -> Take<Self>where
Self: Sized,
n
elements, or fewer
if the underlying iterator ends sooner. Read more1.0.0 · source§fn flat_map<U, F>(self, f: F) -> FlatMap<Self, U, F>
fn flat_map<U, F>(self, f: F) -> FlatMap<Self, U, F>
source§fn map_windows<F, R, const N: usize>(self, f: F) -> MapWindows<Self, F, N>
fn map_windows<F, R, const N: usize>(self, f: F) -> MapWindows<Self, F, N>
iter_map_windows
)f
for each contiguous window of size N
over
self
and returns an iterator over the outputs of f
. Like slice::windows()
,
the windows during mapping overlap as well. Read more1.0.0 · source§fn inspect<F>(self, f: F) -> Inspect<Self, F>
fn inspect<F>(self, f: F) -> Inspect<Self, F>
1.0.0 · source§fn by_ref(&mut self) -> &mut Selfwhere
Self: Sized,
fn by_ref(&mut self) -> &mut Selfwhere
Self: Sized,
source§fn collect_into<E>(self, collection: &mut E) -> &mut E
fn collect_into<E>(self, collection: &mut E) -> &mut E
iter_collect_into
)1.0.0 · source§fn partition<B, F>(self, f: F) -> (B, B)
fn partition<B, F>(self, f: F) -> (B, B)
source§fn is_partitioned<P>(self, predicate: P) -> bool
fn is_partitioned<P>(self, predicate: P) -> bool
iter_is_partitioned
)true
precede all those that return false
. Read more1.27.0 · source§fn try_fold<B, F, R>(&mut self, init: B, f: F) -> R
fn try_fold<B, F, R>(&mut self, init: B, f: F) -> R
1.27.0 · source§fn try_for_each<F, R>(&mut self, f: F) -> R
fn try_for_each<F, R>(&mut self, f: F) -> R
1.0.0 · source§fn fold<B, F>(self, init: B, f: F) -> B
fn fold<B, F>(self, init: B, f: F) -> B
1.51.0 · source§fn reduce<F>(self, f: F) -> Option<Self::Item>
fn reduce<F>(self, f: F) -> Option<Self::Item>
source§fn try_reduce<R>(
&mut self,
f: impl FnMut(Self::Item, Self::Item) -> R,
) -> <<R as Try>::Residual as Residual<Option<<R as Try>::Output>>>::TryType
fn try_reduce<R>( &mut self, f: impl FnMut(Self::Item, Self::Item) -> R, ) -> <<R as Try>::Residual as Residual<Option<<R as Try>::Output>>>::TryType
iterator_try_reduce
)1.0.0 · source§fn all<F>(&mut self, f: F) -> bool
fn all<F>(&mut self, f: F) -> bool
1.0.0 · source§fn any<F>(&mut self, f: F) -> bool
fn any<F>(&mut self, f: F) -> bool
1.0.0 · source§fn find<P>(&mut self, predicate: P) -> Option<Self::Item>
fn find<P>(&mut self, predicate: P) -> Option<Self::Item>
1.30.0 · source§fn find_map<B, F>(&mut self, f: F) -> Option<B>
fn find_map<B, F>(&mut self, f: F) -> Option<B>
source§fn try_find<R>(
&mut self,
f: impl FnMut(&Self::Item) -> R,
) -> <<R as Try>::Residual as Residual<Option<Self::Item>>>::TryType
fn try_find<R>( &mut self, f: impl FnMut(&Self::Item) -> R, ) -> <<R as Try>::Residual as Residual<Option<Self::Item>>>::TryType
try_find
)1.0.0 · source§fn position<P>(&mut self, predicate: P) -> Option<usize>
fn position<P>(&mut self, predicate: P) -> Option<usize>
1.6.0 · source§fn max_by_key<B, F>(self, f: F) -> Option<Self::Item>
fn max_by_key<B, F>(self, f: F) -> Option<Self::Item>
1.15.0 · source§fn max_by<F>(self, compare: F) -> Option<Self::Item>
fn max_by<F>(self, compare: F) -> Option<Self::Item>
1.6.0 · source§fn min_by_key<B, F>(self, f: F) -> Option<Self::Item>
fn min_by_key<B, F>(self, f: F) -> Option<Self::Item>
1.15.0 · source§fn min_by<F>(self, compare: F) -> Option<Self::Item>
fn min_by<F>(self, compare: F) -> Option<Self::Item>
1.0.0 · source§fn unzip<A, B, FromA, FromB>(self) -> (FromA, FromB)
fn unzip<A, B, FromA, FromB>(self) -> (FromA, FromB)
1.36.0 · source§fn copied<'a, T>(self) -> Copied<Self>
fn copied<'a, T>(self) -> Copied<Self>
source§fn array_chunks<const N: usize>(self) -> ArrayChunks<Self, N>where
Self: Sized,
fn array_chunks<const N: usize>(self) -> ArrayChunks<Self, N>where
Self: Sized,
iter_array_chunks
)N
elements of the iterator at a time. Read more1.11.0 · source§fn product<P>(self) -> P
fn product<P>(self) -> P
source§fn cmp_by<I, F>(self, other: I, cmp: F) -> Ordering
fn cmp_by<I, F>(self, other: I, cmp: F) -> Ordering
iter_order_by
)Iterator
with those
of another with respect to the specified comparison function. Read more1.5.0 · source§fn partial_cmp<I>(self, other: I) -> Option<Ordering>
fn partial_cmp<I>(self, other: I) -> Option<Ordering>
PartialOrd
elements of
this Iterator
with those of another. The comparison works like short-circuit
evaluation, returning a result without comparing the remaining elements.
As soon as an order can be determined, the evaluation stops and a result is returned. Read moresource§fn partial_cmp_by<I, F>(self, other: I, partial_cmp: F) -> Option<Ordering>where
Self: Sized,
I: IntoIterator,
F: FnMut(Self::Item, <I as IntoIterator>::Item) -> Option<Ordering>,
fn partial_cmp_by<I, F>(self, other: I, partial_cmp: F) -> Option<Ordering>where
Self: Sized,
I: IntoIterator,
F: FnMut(Self::Item, <I as IntoIterator>::Item) -> Option<Ordering>,
iter_order_by
)Iterator
with those
of another with respect to the specified comparison function. Read moresource§fn eq_by<I, F>(self, other: I, eq: F) -> bool
fn eq_by<I, F>(self, other: I, eq: F) -> bool
iter_order_by
)1.5.0 · source§fn lt<I>(self, other: I) -> bool
fn lt<I>(self, other: I) -> bool
Iterator
are lexicographically
less than those of another. Read more1.5.0 · source§fn le<I>(self, other: I) -> bool
fn le<I>(self, other: I) -> bool
Iterator
are lexicographically
less or equal to those of another. Read more1.5.0 · source§fn gt<I>(self, other: I) -> bool
fn gt<I>(self, other: I) -> bool
Iterator
are lexicographically
greater than those of another. Read more1.5.0 · source§fn ge<I>(self, other: I) -> bool
fn ge<I>(self, other: I) -> bool
Iterator
are lexicographically
greater than or equal to those of another. Read moresource§fn is_sorted_by<F>(self, compare: F) -> bool
fn is_sorted_by<F>(self, compare: F) -> bool
is_sorted
)source§fn is_sorted_by_key<F, K>(self, f: F) -> bool
fn is_sorted_by_key<F, K>(self, f: F) -> bool
is_sorted
)source§impl<'a> Parse<'a, CertParser<'a>> for CertParser<'a>
impl<'a> Parse<'a, CertParser<'a>> for CertParser<'a>
source§fn from_buffered_reader<R>(reader: R) -> Result<CertParser<'a>>where
R: BufferedReader<Cookie> + 'a,
fn from_buffered_reader<R>(reader: R) -> Result<CertParser<'a>>where
R: BufferedReader<Cookie> + 'a,
Initializes a CertParser
from a BufferedReader
.