1use std::cmp::Ordering;
2use std::hash::{Hash, Hasher};
3use std::net::{AddrParseError, IpAddr, SocketAddr, SocketAddrV4, SocketAddrV6};
4use std::str::FromStr;
5
6#[derive(Clone, Debug)]
7pub enum Address {
8 Ori(String),
9 Addr(SocketAddr),
10}
11
12impl Address {
13 pub fn hash_code(&self) -> u64 {
14 match self {
15 Address::Ori(ori) => crate::utils::bytes_to_hash_code(ori.as_bytes()),
16 Address::Addr(addr) => crate::utils::socketaddr_to_hash_code(addr),
17 }
18 }
19}
20
21impl From<String> for Address {
22 fn from(ori: String) -> Self {
23 Address::Ori(ori)
24 }
25}
26
27impl From<&str> for Address {
28 fn from(ori: &str) -> Self {
29 Address::Ori(ori.to_string())
30 }
31}
32
33impl From<SocketAddr> for Address {
34 fn from(addr: SocketAddr) -> Self {
35 Address::Addr(addr)
36 }
37}
38
39impl<I: Into<IpAddr>> From<(I, u16)> for Address {
40 fn from(value: (I, u16)) -> Self {
41 Address::from(SocketAddr::from(value))
42 }
43}
44
45impl From<SocketAddrV4> for Address {
46 fn from(value: SocketAddrV4) -> Self {
47 Address::from(SocketAddr::from(value))
48 }
49}
50
51impl From<SocketAddrV6> for Address {
52 fn from(value: SocketAddrV6) -> Self {
53 Address::from(SocketAddr::from(value))
54 }
55}
56
57impl FromStr for Address {
58 type Err = AddrParseError;
59
60 fn from_str(s: &str) -> Result<Self, Self::Err> {
61 let a = SocketAddr::from_str(s)?;
62 Ok(Address::from(a))
63 }
64}
65
66#[derive(Clone, Debug)]
67pub struct BackendState {
68 hash_code: u64,
69 valid: bool,
70 address: Address,
71}
72
73impl BackendState {
74 pub fn new(addr: Address) -> BackendState {
75 let hash_code = addr.hash_code();
76 BackendState {
77 hash_code,
78 valid: true,
79 address: addr,
80 }
81 }
82
83 pub fn hash_code(&self) -> u64 {
84 self.hash_code
85 }
86
87 pub fn set_valid(&mut self, valid: bool) {
88 self.valid = valid;
89 }
90
91 pub fn get_valid(&self) -> bool {
92 self.valid
93 }
94
95 pub fn get_address(&self) -> &Address {
96 &self.address
97 }
98}
99
100impl Hash for BackendState {
101 fn hash<H: Hasher>(&self, state: &mut H) {
102 state.write_u64(self.hash_code)
103 }
104}
105
106impl Eq for BackendState {}
107
108impl PartialEq for BackendState {
109 fn eq(&self, other: &Self) -> bool {
110 self.hash_code.eq(&other.hash_code)
111 }
112}
113
114impl PartialOrd for BackendState {
115 fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
116 self.hash_code.partial_cmp(&other.hash_code)
117 }
118}
119
120impl Ord for BackendState {
121 fn cmp(&self, other: &Self) -> Ordering {
122 self.hash_code.cmp(&other.hash_code)
123 }
124}
125
126impl From<Address> for BackendState {
127 fn from(addr: Address) -> Self {
128 BackendState::new(addr)
129 }
130}
131
132#[derive(Default)]
133pub struct BackendStates(Vec<BackendState>);
134
135impl BackendStates {
136 pub fn new() -> Self {
137 BackendStates(Vec::new())
138 }
139
140 pub fn contain(&self, hash_code: u64) -> bool {
141 self.0
142 .iter()
143 .find(|bs| bs.hash_code() == hash_code)
144 .is_some()
145 }
146
147 pub fn add_backend(&mut self, addr: Address) {
148 let hash_code = addr.hash_code();
149 if self.0.iter().any(|bs| bs.hash_code == hash_code) {
150 return;
151 }
152
153 self.0.push(BackendState::new(addr));
154 self.sort_backends();
155 }
156
157 pub fn remove_backend(&mut self, addr: &Address) -> bool {
158 let c = self.0.len();
159 let hash_code = addr.hash_code();
160 self.0.retain(|bs| bs.hash_code != hash_code);
161 self.0.len() != c
162 }
163
164 pub fn set_backend_valid(&mut self, addr: &Address, valid: bool) {
165 if let Some(bs) = self.get_backend_mut(addr) {
166 bs.valid = valid;
167 }
168 }
169
170 pub fn get_backend_valid(&self, addr: &Address) -> Option<bool> {
171 self.get_backend(addr).map(|bs| bs.valid)
172 }
173
174 pub fn get_backend(&self, addr: &Address) -> Option<&BackendState> {
175 let hash_code = addr.hash_code();
176 self.0.iter().find(|bs| bs.hash_code == hash_code)
177 }
178
179 pub fn get_backend_mut(&mut self, addr: &Address) -> Option<&mut BackendState> {
180 let hash_code = addr.hash_code();
181 self.0.iter_mut().find(|bs| bs.hash_code == hash_code)
182 }
183
184 pub fn get_backends(&self) -> &[BackendState] {
185 self.0.as_slice()
186 }
187
188 pub fn get_backends_mut(&mut self) -> &mut [BackendState] {
189 self.0.as_mut_slice()
190 }
191
192 fn sort_backends(&mut self) {
193 self.0.sort_by(|bs1, bs2| bs1.hash_code.cmp(&bs2.hash_code));
194 }
195}