1use subslice::SubsliceExt as _;
23#[derive(Debug)]
4pub(super) enum CompatibleDocument<'a> {
5 WellFormed(&'a [u8]),
6 CleanedFromRing(Vec<u8>),
7}
89impl<'a> CompatibleDocument<'a> {
10pub(super) fn from_bytes(bytes: &'a [u8]) -> Self {
11if is_ring(bytes) {
12Self::CleanedFromRing(fix_ring_doc(bytes.to_vec()))
13 } else {
14Self::WellFormed(bytes)
15 }
16 }
17}
1819// Ring uses a very specific template to generate its documents,
20// and so this is essentially a sentinel value of that.
21//
22// It corresponds to CONTEXT-SPECIFIC[1](35) { BIT-STRING(32) {...} } in ASN.1
23//
24// A well-formed bit would look like just CONTEXT-SPECIFIC[1](32) { ... }
25//
26// Note: this is purely a sentinel value, don't take these bytes out of context
27// to detect or fiddle with the document.
28const RING_TEMPLATE_CONTEXT_SPECIFIC: &[u8] = &[0xA1, 0x23, 0x03, 0x21];
2930// A checked well-formed context-specific[1] prefix.
31const WELL_FORMED_CONTEXT_ONE_PREFIX: &[u8] = &[0x81, 0x21];
3233// If present, removes a malfunctioning pubkey suffix and adjusts the length at the start.
34fn fix_ring_doc(mut doc: Vec<u8>) -> Vec<u8> {
35assert!(!doc.is_empty());
36// Check if first tag is ASN.1 SEQUENCE
37assert_eq!(doc[0], 0x30);
38// Second byte asserts the length for the rest of the document
39assert_eq!(doc[1] as usize, doc.len() - 2);
4041let idx = doc
42 .find(RING_TEMPLATE_CONTEXT_SPECIFIC)
43 .expect("Expected to find ring template in doc, but found none.");
4445// Snip off the malformed bit.
46let suffix = doc.split_off(idx);
4748// Feed back an actual well-formed prefix.
49doc.extend(WELL_FORMED_CONTEXT_ONE_PREFIX);
5051// Then give it the actual public key.
52doc.extend(&suffix[4..]);
5354 doc[1] = doc.len() as u8 - 2;
5556 doc
57}
5859fn is_ring(bytes: &[u8]) -> bool {
60 bytes.find(RING_TEMPLATE_CONTEXT_SPECIFIC).is_some()
61}