1use std::sync::Arc;
4
5use thiserror::Error;
6use tor_error::{Bug, ErrorKind, HasKind};
7use tor_linkspec::OwnedChanTarget;
8use tor_rtcompat::TimeoutError;
9
10use crate::SourceInfo;
11
12#[derive(Error, Debug, Clone)]
14#[non_exhaustive]
15#[allow(clippy::large_enum_variant)] pub enum Error {
17 #[error("Error while getting a circuit")]
19 CircMgr(#[from] tor_circmgr::Error),
20
21 #[error("Error fetching directory information")]
23 RequestFailed(#[from] RequestFailedError),
24
25 #[error("Internal error")]
27 Bug(#[from] Bug),
28}
29
30#[derive(Error, Debug, Clone)]
32#[allow(clippy::exhaustive_structs)] #[error("Request failed{}", FromSource(.source))]
34pub struct RequestFailedError {
35 pub source: Option<SourceInfo>,
37
38 #[source]
40 pub error: RequestError,
41}
42
43struct FromSource<'a>(&'a Option<SourceInfo>);
45
46impl std::fmt::Display for FromSource<'_> {
47 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
48 if let Some(si) = self.0 {
49 write!(f, " from {}", si)
50 } else {
51 Ok(())
52 }
53 }
54}
55
56#[derive(Error, Debug, Clone)]
58#[non_exhaustive]
59pub enum RequestError {
60 #[error("directory timed out")]
62 DirTimeout,
63
64 #[error("truncated HTTP headers")]
66 TruncatedHeaders,
67
68 #[error("response too long; gave up after {0} bytes")]
70 ResponseTooLong(usize),
71
72 #[error("headers too long; gave up after {0} bytes")]
74 HeadersTooLong(usize),
75
76 #[error("Couldn't decode data as UTF-8.")]
78 Utf8Encoding(#[from] std::string::FromUtf8Error),
79
80 #[error("IO error")]
82 IoError(#[source] Arc<std::io::Error>),
83
84 #[error("Protocol error while launching a stream")]
86 Proto(#[from] tor_proto::Error),
87
88 #[error("Couldn't parse HTTP headers")]
90 HttparseError(#[from] httparse::Error),
91
92 #[error("Couldn't create HTTP request")]
98 HttpError(#[source] Arc<http::Error>),
99
100 #[error("Unrecognized content encoding: {0:?}")]
102 ContentEncoding(String),
103
104 #[error("Too much clock skew with directory cache")]
109 TooMuchClockSkew,
110
111 #[error("We didn't have any objects to request")]
116 EmptyRequest,
117
118 #[error("HTTP status code {0}: {1:?}")]
120 HttpStatus(u16, String),
121}
122
123impl From<TimeoutError> for RequestError {
124 fn from(_: TimeoutError) -> Self {
125 RequestError::DirTimeout
126 }
127}
128
129impl From<std::io::Error> for RequestError {
130 fn from(err: std::io::Error) -> Self {
131 Self::IoError(Arc::new(err))
132 }
133}
134
135impl From<http::Error> for RequestError {
136 fn from(err: http::Error) -> Self {
137 Self::HttpError(Arc::new(err))
138 }
139}
140
141impl Error {
142 pub fn should_retire_circ(&self) -> bool {
145 match self {
148 Error::CircMgr(_) => true, Error::RequestFailed(RequestFailedError { error, .. }) => error.should_retire_circ(),
150 Error::Bug(_) => true,
151 }
152 }
153
154 pub fn cache_ids(&self) -> Vec<&OwnedChanTarget> {
159 match &self {
160 Error::CircMgr(e) => e.peers(),
161 Error::RequestFailed(RequestFailedError {
162 source: Some(source),
163 ..
164 }) => vec![source.cache_id()],
165 _ => Vec::new(),
166 }
167 }
168}
169
170impl RequestError {
171 pub fn should_retire_circ(&self) -> bool {
174 true
177 }
178}
179
180impl HasKind for RequestError {
181 fn kind(&self) -> ErrorKind {
182 use ErrorKind as EK;
183 use RequestError as E;
184 match self {
185 E::DirTimeout => EK::TorNetworkTimeout,
186 E::TruncatedHeaders => EK::TorProtocolViolation,
187 E::ResponseTooLong(_) => EK::TorProtocolViolation,
188 E::HeadersTooLong(_) => EK::TorProtocolViolation,
189 E::Utf8Encoding(_) => EK::TorProtocolViolation,
190 E::IoError(_) => EK::TorDirectoryError,
194 E::Proto(e) => e.kind(),
195 E::HttparseError(_) => EK::TorProtocolViolation,
196 E::HttpError(_) => EK::Internal,
197 E::ContentEncoding(_) => EK::TorProtocolViolation,
198 E::TooMuchClockSkew => EK::TorDirectoryError,
199 E::EmptyRequest => EK::Internal,
200 E::HttpStatus(_, _) => EK::TorDirectoryError,
201 }
202 }
203}
204
205impl HasKind for RequestFailedError {
206 fn kind(&self) -> ErrorKind {
207 self.error.kind()
208 }
209}
210
211impl HasKind for Error {
212 fn kind(&self) -> ErrorKind {
213 use Error as E;
214 match self {
215 E::CircMgr(e) => e.kind(),
216 E::RequestFailed(e) => e.kind(),
217 E::Bug(e) => e.kind(),
218 }
219 }
220}
221
222#[cfg(any(feature = "hs-client", feature = "hs-service"))]
223impl Error {
224 pub fn should_report_as_suspicious_if_anon(&self) -> bool {
228 use Error as E;
229 match self {
230 E::CircMgr(_) => false,
231 E::RequestFailed(e) => e.error.should_report_as_suspicious_if_anon(),
232 E::Bug(_) => false,
233 }
234 }
235}
236#[cfg(any(feature = "hs-client", feature = "hs-service"))]
237impl RequestError {
238 pub fn should_report_as_suspicious_if_anon(&self) -> bool {
242 use tor_proto::Error as PE;
243 match self {
244 RequestError::ResponseTooLong(_) => true,
245 RequestError::HeadersTooLong(_) => true,
246 RequestError::Proto(PE::ExcessInboundCells) => true,
247 RequestError::Proto(_) => false,
248 RequestError::DirTimeout => false,
249 RequestError::TruncatedHeaders => false,
250 RequestError::Utf8Encoding(_) => false,
251 RequestError::IoError(_) => false,
252 RequestError::HttparseError(_) => false,
253 RequestError::HttpError(_) => false,
254 RequestError::ContentEncoding(_) => false,
255 RequestError::TooMuchClockSkew => false,
256 RequestError::EmptyRequest => false,
257 RequestError::HttpStatus(_, _) => false,
258 }
259 }
260}