1
//! Building support for the network document meta-format
2
//!
3
//! Implements building documents according to
4
//! [dir-spec.txt](https://spec.torproject.org/dir-spec).
5
//! section 1.2 and 1.3.
6
//!
7
//! This facility processes output that complies with the meta-document format,
8
//! (`dir-spec.txt` section 1.2) -
9
//! unless `raw` methods are called with improper input.
10
//!
11
//! However, no checks are done on keyword presence/absence, multiplicity, or ordering,
12
//! so the output may not necessarily conform to the format of the particular intended document.
13
//! It is the caller's responsibility to call `.item()` in the right order,
14
//! with the right keywords and arguments.
15

            
16
use std::fmt::{Display, Write};
17

            
18
use base64ct::{Base64, Encoding};
19
use rand::{CryptoRng, RngCore};
20
use tor_bytes::EncodeError;
21
use tor_error::{internal, Bug};
22

            
23
use crate::parse::keyword::Keyword;
24
use crate::parse::tokenize::tag_keywords_ok;
25
use crate::types::misc::{Iso8601TimeNoSp, Iso8601TimeSp};
26

            
27
/// Encoder, representing a partially-built document.
28
///
29
/// For example usage, see the tests in this module, or a descriptor building
30
/// function in tor-netdoc (such as `hsdesc::build::inner::HsDescInner::build_sign`).
31
#[derive(Debug, Clone)]
32
pub(crate) struct NetdocEncoder {
33
    /// The being-built document, with everything accumulated so far
34
    ///
35
    /// If an [`ItemEncoder`] exists, it will add a newline when it's dropped.
36
    ///
37
    /// `Err` means bad values passed to some builder function
38
    built: Result<String, Bug>,
39
}
40

            
41
/// Encoder for an individual item within a being-built document
42
///
43
/// Returned by [`NetdocEncoder::item()`].
44
#[derive(Debug)]
45
pub(crate) struct ItemEncoder<'n> {
46
    /// The document including the partial item that we're building
47
    ///
48
    /// We will always add a newline when we're dropped
49
    doc: &'n mut NetdocEncoder,
50
}
51

            
52
/// Position within a (perhaps partially-) built document
53
///
54
/// This is provided mainly to allow the caller to perform signature operations
55
/// on the part of the document that is to be signed.
56
/// (Sometimes this is only part of it.)
57
///
58
/// There is no enforced linkage between this and the document it refers to.
59
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
60
pub(crate) struct Cursor {
61
    /// The offset (in bytes, as for `&str`)
62
    ///
63
    /// Can be out of range if the corresponding `NetdocEncoder` is contains an `Err`.
64
    offset: usize,
65
}
66

            
67
/// Types that can be added as argument(s) to item keyword lines
68
///
69
/// Implemented for strings, and various other types.
70
///
71
/// This is a separate trait so we can control the formatting of (eg) [`Iso8601TimeSp`],
72
/// without having a method on `ItemEncoder` for each argument type.
73
pub(crate) trait ItemArgument {
74
    /// Format as a string suitable for including as a netdoc keyword line argument
75
    ///
76
    /// The implementation is responsible for checking that the syntax is legal.
77
    /// For example, if `self` is a string, it must check that the string is
78
    /// in legal as a single argument.
79
    ///
80
    /// Some netdoc values (eg times) turn into several arguments; in that case,
81
    /// one `ItemArgument` may format into multiple arguments, and this method
82
    /// is responsible for writing them all, with the necessary spaces.
83
    fn write_onto(&self, out: &mut ItemEncoder<'_>) -> Result<(), Bug>;
84
}
85

            
86
impl NetdocEncoder {
87
    /// Start encoding a document
88
9118
    pub(crate) fn new() -> Self {
89
9118
        NetdocEncoder {
90
9118
            built: Ok(String::new()),
91
9118
        }
92
9118
    }
93

            
94
    /// Adds an item to the being-built document
95
    ///
96
    /// The item can be further extended with arguments or an object,
97
    /// using the returned `ItemEncoder`.
98
3984
    pub(crate) fn item(&mut self, keyword: impl Keyword) -> ItemEncoder {
99
3984
        self.raw(&keyword.to_str());
100
3984
        ItemEncoder { doc: self }
101
3984
    }
102

            
103
    /// Internal name for `push_raw_string()`
104
233564
    fn raw(&mut self, s: &dyn Display) {
105
239466
        self.write_with(|b| {
106
233564
            write!(b, "{}", s).expect("write! failed on String");
107
233564
            Ok(())
108
239466
        });
109
233564
    }
110

            
111
    /// Extend the being-built document with a fallible function `f`
112
    ///
113
    /// Doesn't call `f` if the building has already failed,
114
    /// and handles the error if `f` fails.
115
234932
    fn write_with(&mut self, f: impl FnOnce(&mut String) -> Result<(), Bug>) {
116
234932
        let Ok(build) = &mut self.built else {
117
            return;
118
        };
119
234932
        match f(build) {
120
234932
            Ok(()) => (),
121
            Err(e) => {
122
                self.built = Err(e);
123
            }
124
        }
125
234932
    }
126

            
127
    /// Adds raw text to the being-built document
128
    ///
129
    /// `s` is added as raw text, after the newline ending the previous item.
130
    /// If `item` is subsequently called, the start of that item
131
    /// will immediately follow `s`.
132
    ///
133
    /// It is the responsibility of the caller to obey the metadocument syntax.
134
    /// In particular, `s` should end with a newline.
135
    /// No checks are performed.
136
    /// Incorrect use might lead to malformed documents, or later errors.
137
    #[allow(dead_code)] // TODO: We should remove this if it never used.
138
    pub(crate) fn push_raw_string(&mut self, s: &dyn Display) {
139
        self.raw(s);
140
    }
141

            
142
    /// Return a cursor, pointing to just after the last item (if any)
143
6068
    pub(crate) fn cursor(&self) -> Cursor {
144
6068
        let offset = match &self.built {
145
6068
            Ok(b) => b.len(),
146
            Err(_) => usize::MAX,
147
        };
148
6068
        Cursor { offset }
149
6068
    }
150

            
151
    /// Obtain the text of a section of the document
152
    ///
153
    /// Useful for making a signature.
154
3034
    pub(crate) fn slice(&self, begin: Cursor, end: Cursor) -> Result<&str, Bug> {
155
3034
        self.built
156
3034
            .as_ref()
157
3034
            .map_err(Clone::clone)?
158
3034
            .get(begin.offset..end.offset)
159
3034
            .ok_or_else(|| internal!("NetdocEncoder::slice out of bounds, Cursor mismanaged"))
160
3034
    }
161

            
162
    /// Build the document into textual form
163
9114
    pub(crate) fn finish(self) -> Result<String, Bug> {
164
9114
        self.built
165
9114
    }
166
}
167

            
168
impl ItemArgument for str {
169
75830
    fn write_onto(&self, out: &mut ItemEncoder<'_>) -> Result<(), Bug> {
170
75830
        // Implements this
171
75830
        // https://gitlab.torproject.org/tpo/core/torspec/-/merge_requests/106
172
2305513
        if self.is_empty() || self.chars().any(|c| !c.is_ascii_graphic()) {
173
            return Err(internal!("invalid keyword argument syntax {:?}", self));
174
75830
        }
175
75830
        out.args_raw_nonempty(&self);
176
75830
        Ok(())
177
75830
    }
178
}
179

            
180
impl ItemArgument for String {
181
45486
    fn write_onto(&self, out: &mut ItemEncoder<'_>) -> Result<(), Bug> {
182
45486
        ItemArgument::write_onto(&self.as_str(), out)
183
45486
    }
184
}
185

            
186
impl<T: ItemArgument + ?Sized> ItemArgument for &'_ T {
187
75830
    fn write_onto(&self, out: &mut ItemEncoder<'_>) -> Result<(), Bug> {
188
75830
        <T as ItemArgument>::write_onto(self, out)
189
75830
    }
190
}
191

            
192
/// Implement [`ItemArgument`] for `$ty` in terms of `<$ty as Display>`
193
///
194
/// Checks that the syntax is acceptable.
195
macro_rules! impl_item_argument_as_display { { $( $ty:ty $(,)? )* } => { $(
196
    impl ItemArgument for $ty {
197
6092
        fn write_onto(&self, out: &mut ItemEncoder<'_>) -> Result<(), Bug> {
198
6092
            let arg = self.to_string();
199
6092
            out.add_arg(&arg.as_str());
200
6092
            Ok(())
201
6092
        }
202
    }
203
)* } }
204

            
205
impl_item_argument_as_display! { usize, u8, u16, u32, u64, u128 }
206
impl_item_argument_as_display! { isize, i8, i16, i32, i64, i128 }
207
// TODO: should we implement ItemArgument for, say, tor_llcrypto::pk::rsa::RsaIdentity ?
208
// It's not clear whether it's always formatted the same way in all parts of the spec.
209
// The Display impl of RsaIdentity adds a `$` which is not supposed to be present
210
// in (for example) an authority certificate (authcert)'s "fingerprint" line.
211

            
212
impl_item_argument_as_display! {Iso8601TimeNoSp}
213
impl ItemArgument for Iso8601TimeSp {
214
    // Unlike the macro'd formats, contains a space while still being one argument
215
6
    fn write_onto(&self, out: &mut ItemEncoder<'_>) -> Result<(), Bug> {
216
6
        let arg = self.to_string();
217
6
        out.args_raw_nonempty(&arg.as_str());
218
6
        Ok(())
219
6
    }
220
}
221

            
222
impl<'n> ItemEncoder<'n> {
223
    /// Add a single argument.
224
    ///
225
    /// If the argument is not in the correct syntax, a `Bug`
226
    /// error will be reported (later).
227
    //
228
    // This is not a hot path.  `dyn` for smaller code size.
229
75836
    pub(crate) fn arg(mut self, arg: &dyn ItemArgument) -> Self {
230
75836
        self.add_arg(arg);
231
75836
        self
232
75836
    }
233

            
234
    /// Add a single argument, to a borrowed `ItemEncoder`
235
    ///
236
    /// If the argument is not in the correct syntax, a `Bug`
237
    /// error will be reported (later).
238
    //
239
    // Needed for implementing `ItemArgument`
240
81928
    pub(crate) fn add_arg(&mut self, arg: &dyn ItemArgument) {
241
81928
        let () = arg
242
81928
            .write_onto(self)
243
81928
            .unwrap_or_else(|err| self.doc.built = Err(err));
244
81928
    }
245

            
246
    /// Add zero or more arguments, supplied as a single string.
247
    ///
248
    /// `args` should zero or more valid argument strings,
249
    /// separated by (single) spaces.
250
    /// This is not (properly) checked.
251
    /// Incorrect use might lead to malformed documents, or later errors.
252
    #[allow(unused)] // TODO: We should eventually remove this if nothing starts to use it.
253
    pub(crate) fn args_raw_string(mut self, args: &dyn Display) -> Self {
254
        let args = args.to_string();
255
        if !args.is_empty() {
256
            self.args_raw_nonempty(&args);
257
        }
258
        self
259
    }
260

            
261
    /// Add one or more arguments, supplied as a single string, without any checking
262
75836
    fn args_raw_nonempty(&mut self, args: &dyn Display) {
263
75836
        self.doc.raw(&format_args!(" {}", args));
264
75836
    }
265

            
266
    /// Add an object to the item
267
    ///
268
    /// Checks that `keywords` is in the correct syntax.
269
    /// Doesn't check that it makes semantic sense for the position of the document.
270
    /// `data` will be PEM (base64) encoded.
271
    //
272
    // If keyword is not in the correct syntax, a `Bug` is stored in self.doc.
273
1368
    pub(crate) fn object(
274
1368
        self,
275
1368
        keywords: &str,
276
1368
        // Writeable isn't dyn-compatible
277
1368
        data: impl tor_bytes::WriteableOnce,
278
1368
    ) {
279
1368
        use crate::parse::tokenize::object::*;
280
1368

            
281
1368
        self.doc.write_with(|out| {
282
1368
            if keywords.is_empty() || !tag_keywords_ok(keywords) {
283
                return Err(internal!("bad object keywords string {:?}", keywords));
284
1368
            }
285
1368
            let data = {
286
1368
                let mut bytes = vec![];
287
1368
                data.write_into(&mut bytes).map_err(EncodeError::from)?;
288
1368
                Base64::encode_string(&bytes)
289
1368
            };
290
1368
            let mut data = &data[..];
291
1368
            writeln!(out, "\n{BEGIN_STR}{keywords}{TAG_END}").expect("write!");
292
44256
            while !data.is_empty() {
293
42888
                let (l, r) = if data.len() > BASE64_PEM_MAX_LINE {
294
41524
                    data.split_at(BASE64_PEM_MAX_LINE)
295
                } else {
296
1364
                    (data, "")
297
                };
298
42888
                writeln!(out, "{l}").expect("write!");
299
42888
                data = r;
300
            }
301
            // final newline will be written by Drop impl
302
1368
            write!(out, "{END_STR}{keywords}{TAG_END}").expect("write!");
303
1368
            Ok(())
304
1368
        });
305
1368
    }
306
}
307

            
308
impl Drop for ItemEncoder<'_> {
309
78864
    fn drop(&mut self) {
310
78864
        self.doc.raw(&'\n');
311
78864
    }
312
}
313

            
314
/// A trait for building and signing netdocs.
315
pub trait NetdocBuilder {
316
    /// Build the document into textual form.
317
    fn build_sign<R: RngCore + CryptoRng>(self, rng: &mut R) -> Result<String, EncodeError>;
318
}
319

            
320
#[cfg(test)]
321
mod test {
322
    // @@ begin test lint list maintained by maint/add_warning @@
323
    #![allow(clippy::bool_assert_comparison)]
324
    #![allow(clippy::clone_on_copy)]
325
    #![allow(clippy::dbg_macro)]
326
    #![allow(clippy::mixed_attributes_style)]
327
    #![allow(clippy::print_stderr)]
328
    #![allow(clippy::print_stdout)]
329
    #![allow(clippy::single_char_pattern)]
330
    #![allow(clippy::unwrap_used)]
331
    #![allow(clippy::unchecked_duration_subtraction)]
332
    #![allow(clippy::useless_vec)]
333
    #![allow(clippy::needless_pass_by_value)]
334
    //! <!-- @@ end test lint list maintained by maint/add_warning @@ -->
335
    use super::*;
336
    use std::str::FromStr;
337

            
338
    use base64ct::{Base64Unpadded, Encoding};
339

            
340
    #[test]
341
    fn time_formats_as_args() {
342
        use crate::doc::authcert::AuthCertKwd as ACK;
343
        use crate::doc::netstatus::NetstatusKwd as NK;
344

            
345
        let t_sp = Iso8601TimeSp::from_str("2020-04-18 08:36:57").unwrap();
346
        let t_no_sp = Iso8601TimeNoSp::from_str("2021-04-18T08:36:57").unwrap();
347

            
348
        let mut encode = NetdocEncoder::new();
349
        encode.item(ACK::DIR_KEY_EXPIRES).arg(&t_sp);
350
        encode
351
            .item(NK::SHARED_RAND_PREVIOUS_VALUE)
352
            .arg(&"3")
353
            .arg(&"bMZR5Q6kBadzApPjd5dZ1tyLt1ckv1LfNCP/oyGhCXs=")
354
            .arg(&t_no_sp);
355

            
356
        let doc = encode.finish().unwrap();
357
        println!("{}", doc);
358
        assert_eq!(
359
            doc,
360
            r"dir-key-expires 2020-04-18 08:36:57
361
shared-rand-previous-value 3 bMZR5Q6kBadzApPjd5dZ1tyLt1ckv1LfNCP/oyGhCXs= 2021-04-18T08:36:57
362
"
363
        );
364
    }
365

            
366
    #[test]
367
    fn authcert() {
368
        use crate::doc::authcert::AuthCertKwd as ACK;
369
        use crate::doc::authcert::{AuthCert, UncheckedAuthCert};
370

            
371
        // c&p from crates/tor-llcrypto/tests/testvec.rs
372
        let pk_rsa = {
373
            let pem = "
374
MIGJAoGBANUntsY9boHTnDKKlM4VfczcBE6xrYwhDJyeIkh7TPrebUBBvRBGmmV+
375
PYK8AM9irDtqmSR+VztUwQxH9dyEmwrM2gMeym9uXchWd/dt7En/JNL8srWIf7El
376
qiBHRBGbtkF/Re5pb438HC/CGyuujp43oZ3CUYosJOfY/X+sD0aVAgMBAAE";
377
            Base64Unpadded::decode_vec(&pem.replace('\n', "")).unwrap()
378
        };
379

            
380
        let mut encode = NetdocEncoder::new();
381
        encode.item(ACK::DIR_KEY_CERTIFICATE_VERSION).arg(&3);
382
        encode
383
            .item(ACK::FINGERPRINT)
384
            .arg(&"9367f9781da8eabbf96b691175f0e701b43c602e");
385
        encode
386
            .item(ACK::DIR_KEY_PUBLISHED)
387
            .arg(&Iso8601TimeSp::from_str("2020-04-18 08:36:57").unwrap());
388
        encode
389
            .item(ACK::DIR_KEY_EXPIRES)
390
            .arg(&Iso8601TimeSp::from_str("2021-04-18 08:36:57").unwrap());
391
        encode
392
            .item(ACK::DIR_IDENTITY_KEY)
393
            .object("RSA PUBLIC KEY", &*pk_rsa);
394
        encode
395
            .item(ACK::DIR_SIGNING_KEY)
396
            .object("RSA PUBLIC KEY", &*pk_rsa);
397
        encode
398
            .item(ACK::DIR_KEY_CROSSCERT)
399
            .object("ID SIGNATURE", []);
400
        encode
401
            .item(ACK::DIR_KEY_CERTIFICATION)
402
            .object("SIGNATURE", []);
403

            
404
        let doc = encode.finish().unwrap();
405
        eprintln!("{}", doc);
406
        assert_eq!(
407
            doc,
408
            r"dir-key-certificate-version 3
409
fingerprint 9367f9781da8eabbf96b691175f0e701b43c602e
410
dir-key-published 2020-04-18 08:36:57
411
dir-key-expires 2021-04-18 08:36:57
412
dir-identity-key
413
-----BEGIN RSA PUBLIC KEY-----
414
MIGJAoGBANUntsY9boHTnDKKlM4VfczcBE6xrYwhDJyeIkh7TPrebUBBvRBGmmV+
415
PYK8AM9irDtqmSR+VztUwQxH9dyEmwrM2gMeym9uXchWd/dt7En/JNL8srWIf7El
416
qiBHRBGbtkF/Re5pb438HC/CGyuujp43oZ3CUYosJOfY/X+sD0aVAgMBAAE=
417
-----END RSA PUBLIC KEY-----
418
dir-signing-key
419
-----BEGIN RSA PUBLIC KEY-----
420
MIGJAoGBANUntsY9boHTnDKKlM4VfczcBE6xrYwhDJyeIkh7TPrebUBBvRBGmmV+
421
PYK8AM9irDtqmSR+VztUwQxH9dyEmwrM2gMeym9uXchWd/dt7En/JNL8srWIf7El
422
qiBHRBGbtkF/Re5pb438HC/CGyuujp43oZ3CUYosJOfY/X+sD0aVAgMBAAE=
423
-----END RSA PUBLIC KEY-----
424
dir-key-crosscert
425
-----BEGIN ID SIGNATURE-----
426
-----END ID SIGNATURE-----
427
dir-key-certification
428
-----BEGIN SIGNATURE-----
429
-----END SIGNATURE-----
430
"
431
        );
432

            
433
        let _: UncheckedAuthCert = AuthCert::parse(&doc).unwrap();
434
    }
435
}