torrust_tracker/servers/http/v1/extractors/
announce_request.rs1use std::panic::Location;
31
32use axum::extract::FromRequestParts;
33use axum::http::request::Parts;
34use axum::response::{IntoResponse, Response};
35use futures::future::BoxFuture;
36use futures::FutureExt;
37
38use crate::servers::http::v1::query::Query;
39use crate::servers::http::v1::requests::announce::{Announce, ParseAnnounceQueryError};
40use crate::servers::http::v1::responses;
41
42pub struct ExtractRequest(pub Announce);
45
46impl<S> FromRequestParts<S> for ExtractRequest
47where
48 S: Send + Sync,
49{
50 type Rejection = Response;
51
52 #[must_use]
53 fn from_request_parts<'life0, 'life1, 'async_trait>(
54 parts: &'life0 mut Parts,
55 _state: &'life1 S,
56 ) -> BoxFuture<'async_trait, Result<Self, Self::Rejection>>
57 where
58 'life0: 'async_trait,
59 'life1: 'async_trait,
60 Self: 'async_trait,
61 {
62 async {
63 match extract_announce_from(parts.uri.query()) {
64 Ok(announce_request) => Ok(ExtractRequest(announce_request)),
65 Err(error) => Err(error.into_response()),
66 }
67 }
68 .boxed()
69 }
70}
71
72fn extract_announce_from(maybe_raw_query: Option<&str>) -> Result<Announce, responses::error::Error> {
73 if maybe_raw_query.is_none() {
74 return Err(responses::error::Error::from(ParseAnnounceQueryError::MissingParams {
75 location: Location::caller(),
76 }));
77 }
78
79 let query = maybe_raw_query.unwrap().parse::<Query>();
80
81 if let Err(error) = query {
82 return Err(responses::error::Error::from(error));
83 }
84
85 let announce_request = Announce::try_from(query.unwrap());
86
87 if let Err(error) = announce_request {
88 return Err(responses::error::Error::from(error));
89 }
90
91 Ok(announce_request.unwrap())
92}
93
94#[cfg(test)]
95mod tests {
96 use std::str::FromStr;
97
98 use aquatic_udp_protocol::{NumberOfBytes, PeerId};
99 use torrust_tracker_primitives::info_hash::InfoHash;
100
101 use super::extract_announce_from;
102 use crate::servers::http::v1::requests::announce::{Announce, Compact, Event};
103 use crate::servers::http::v1::responses::error::Error;
104
105 fn assert_error_response(error: &Error, error_message: &str) {
106 assert!(
107 error.failure_reason.contains(error_message),
108 "Error response does not contain message: '{error_message}'. Error: {error:?}"
109 );
110 }
111
112 #[test]
113 fn it_should_extract_the_announce_request_from_the_url_query_params() {
114 let raw_query = "info_hash=%3B%24U%04%CF%5F%11%BB%DB%E1%20%1C%EAjk%F4Z%EE%1B%C0&peer_addr=2.137.87.41&downloaded=0&uploaded=0&peer_id=-qB00000000000000001&port=17548&left=0&event=completed&compact=0&numwant=50";
115
116 let announce = extract_announce_from(Some(raw_query)).unwrap();
117
118 assert_eq!(
119 announce,
120 Announce {
121 info_hash: InfoHash::from_str("3b245504cf5f11bbdbe1201cea6a6bf45aee1bc0").unwrap(),
122 peer_id: PeerId(*b"-qB00000000000000001"),
123 port: 17548,
124 downloaded: Some(NumberOfBytes::new(0)),
125 uploaded: Some(NumberOfBytes::new(0)),
126 left: Some(NumberOfBytes::new(0)),
127 event: Some(Event::Completed),
128 compact: Some(Compact::NotAccepted),
129 numwant: Some(50),
130 }
131 );
132 }
133
134 #[test]
135 fn it_should_reject_a_request_without_query_params() {
136 let response = extract_announce_from(None).unwrap_err();
137
138 assert_error_response(
139 &response,
140 "Cannot parse query params for announce request: missing query params for announce request",
141 );
142 }
143
144 #[test]
145 fn it_should_reject_a_request_with_a_query_that_cannot_be_parsed() {
146 let invalid_query = "param1=value1=value2";
147 let response = extract_announce_from(Some(invalid_query)).unwrap_err();
148
149 assert_error_response(&response, "Cannot parse query params");
150 }
151
152 #[test]
153 fn it_should_reject_a_request_with_a_query_that_cannot_be_parsed_into_an_announce_request() {
154 let response = extract_announce_from(Some("param1=value1")).unwrap_err();
155
156 assert_error_response(&response, "Cannot parse query params for announce request");
157 }
158}