1
//! Routerstatus-specific parts of networkstatus parsing.
2
//!
3
//! This is a private module; relevant pieces are re-exported by its
4
//! parent.
5

            
6
#[cfg(feature = "build_docs")]
7
pub(crate) mod build;
8
pub(crate) mod md;
9
pub(crate) mod plain;
10
#[cfg(feature = "incomplete")]
11
pub(crate) mod vote;
12

            
13
use super::{ConsensusFlavor, ConsensusMethods, consensus_methods_comma_separated};
14
use crate::doc::netstatus::{
15
    IgnoredPublicationTimeSp, NetParams, NetstatusKwd, Protocols, RelayWeight, RelayWeightsItem,
16
};
17
use crate::encode::{EncodeOrd, ItemEncoder};
18
use crate::parse::parser::Section;
19
use crate::parse2::ItemArgumentParseable;
20
use crate::types::misc::*;
21
use crate::types::policy::PortPolicy;
22
use crate::types::relay_flags::{self, DocRelayFlags, RelayFlag, RelayFlags};
23
use crate::types::version::TorVersion;
24
use crate::{Error, NetdocErrorKind as EK, Result};
25
use derive_deftly::Deftly;
26
use itertools::chain;
27
use std::cmp::Ordering;
28
use std::sync::Arc;
29
use std::{net, time};
30
use tor_basic_utils::intern::InternCache;
31
use tor_error::{Bug, internal};
32
use tor_llcrypto::pk::rsa::RsaIdentity;
33

            
34
/// A version as presented in a router status.
35
///
36
/// This can either be a parsed Tor version, or an unparsed string.
37
//
38
// TODO: This might want to merge, at some point, with routerdesc::RelayPlatform.
39
#[derive(Clone, Debug, Eq, PartialEq, Hash, derive_more::Display)]
40
#[non_exhaustive]
41
pub enum SoftwareVersion {
42
    /// A Tor version
43
    #[display("Tor {_0}")]
44
    CTor(TorVersion),
45
    /// A string we couldn't parse.
46
    Other(Arc<str>),
47
}
48

            
49
/// A cache of unparsable version strings.
50
///
51
/// We use this because we expect there not to be very many distinct versions of
52
/// relay software in existence.
53
static OTHER_VERSION_CACHE: InternCache<str> = InternCache::new();
54

            
55
/// `m` item in votes
56
///
57
/// <https://spec.torproject.org/dir-spec/consensus-formats.html#item:m>
58
///
59
/// This is different to the `m` line in microdesc consensuses.
60
/// Plain consensuses don't have `m` lines at all.
61
///
62
/// ### Non-invariants
63
///
64
///  * There may be overlapping or even contradictory information.
65
///  * It might not be sorted.
66
///    Users of the structure who need to emit reproducible document encodings.
67
///    must sort it.
68
///  * These non-invariants apply both within one instance of this struct,
69
///    and across multiple instances of it within a `RouterStatus`.
70
#[derive(Debug, Clone, Default, Eq, PartialEq, Ord, PartialOrd, Deftly)]
71
#[derive_deftly(ItemValueParseable)]
72
#[cfg_attr(feature = "incomplete", derive_deftly(ItemValueEncodable))] // untested
73
#[non_exhaustive]
74
pub struct RouterStatusMdDigestsVote {
75
    /// The methods for which this document is applicable.
76
    #[deftly(netdoc(with = consensus_methods_comma_separated))]
77
    pub consensus_methods: ConsensusMethods,
78

            
79
    /// The various hashes of this document.
80
    pub digests: Vec<IdentifiedDigest>,
81
}
82

            
83
impl std::str::FromStr for SoftwareVersion {
84
    type Err = Error;
85

            
86
2592
    fn from_str(s: &str) -> Result<Self> {
87
2592
        let mut elts = s.splitn(3, ' ');
88
2592
        if elts.next() == Some("Tor") {
89
2588
            if let Some(Ok(v)) = elts.next().map(str::parse) {
90
2588
                return Ok(SoftwareVersion::CTor(v));
91
            }
92
4
        }
93

            
94
4
        Ok(SoftwareVersion::Other(OTHER_VERSION_CACHE.intern_ref(s)))
95
2592
    }
96
}
97

            
98
/// Helper to decode a document digest in the format in which it
99
/// appears in a given kind of routerstatus.
100
trait FromRsString: Sized {
101
    /// Try to decode the given object.
102
    fn decode(s: &str) -> Result<Self>;
103
}