1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
use std::{
	cmp,
	hash::{self, Hash},
};

use static_regular_grammar::RegularGrammar;

mod host;
mod port;
mod userinfo;

use crate::common::AuthorityImpl;

pub use host::*;
pub use port::*;
pub use userinfo::*;

#[derive(RegularGrammar)]
#[grammar(
	file = "src/uri/grammar.abnf",
	entry_point = "authority",
	name = "URI authority",
	ascii,
	cache = "automata/uri/authority.aut.cbor"
)]
#[grammar(sized(
	AuthorityBuf,
	derive(Debug, Display, PartialEq, Eq, PartialOrd, Ord, Hash)
))]
#[cfg_attr(feature = "serde", grammar(serde))]
#[cfg_attr(feature = "ignore-grammars", grammar(disable))]
pub struct Authority([u8]);

impl AuthorityImpl for Authority {
	type UserInfo = UserInfo;
	type Host = Host;

	unsafe fn new_unchecked(bytes: &[u8]) -> &Self {
		Self::new_unchecked(bytes)
	}

	fn as_bytes(&self) -> &[u8] {
		&self.0
	}
}

#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct AuthorityParts<'a> {
	pub user_info: Option<&'a UserInfo>,
	pub host: &'a Host,
	pub port: Option<&'a Port>,
}

impl Authority {
	pub fn user_info(&self) -> Option<&UserInfo> {
		AuthorityImpl::user_info(self)
	}

	pub fn host(&self) -> &Host {
		AuthorityImpl::host(self)
	}

	pub fn port(&self) -> Option<&Port> {
		AuthorityImpl::port(self)
	}

	pub fn parts(&self) -> AuthorityParts {
		let ranges = AuthorityImpl::parts(self);

		AuthorityParts {
			user_info: ranges
				.user_info
				.map(|r| unsafe { UserInfo::new_unchecked(&self.0[r]) }),
			host: unsafe { Host::new_unchecked(&self.0[ranges.host]) },
			port: ranges
				.port
				.map(|r| unsafe { Port::new_unchecked(&self.0[r]) }),
		}
	}
}

impl cmp::PartialEq for Authority {
	#[inline]
	fn eq(&self, other: &Authority) -> bool {
		self.parts() == other.parts()
	}
}

impl Eq for Authority {}

impl<'a> PartialEq<&'a str> for Authority {
	#[inline]
	fn eq(&self, other: &&'a str) -> bool {
		self.as_str() == *other
	}
}

impl PartialOrd for Authority {
	#[inline]
	fn partial_cmp(&self, other: &Authority) -> Option<cmp::Ordering> {
		Some(self.cmp(other))
	}
}

impl Ord for Authority {
	#[inline]
	fn cmp(&self, other: &Authority) -> cmp::Ordering {
		self.parts().cmp(&other.parts())
	}
}

impl Hash for Authority {
	#[inline]
	fn hash<H: hash::Hasher>(&self, hasher: &mut H) {
		self.parts().hash(hasher)
	}
}