wreq-rb 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/Cargo.lock +2688 -0
- data/Cargo.toml +6 -0
- data/README.md +179 -0
- data/ext/wreq_rb/Cargo.toml +39 -0
- data/ext/wreq_rb/extconf.rb +22 -0
- data/ext/wreq_rb/src/client.rs +565 -0
- data/ext/wreq_rb/src/error.rs +25 -0
- data/ext/wreq_rb/src/lib.rs +20 -0
- data/ext/wreq_rb/src/response.rs +132 -0
- data/lib/wreq-rb/version.rb +5 -0
- data/lib/wreq-rb.rb +17 -0
- data/patches/0001-add-transfer-size-tracking.patch +292 -0
- data/vendor/wreq/Cargo.toml +306 -0
- data/vendor/wreq/LICENSE +202 -0
- data/vendor/wreq/README.md +122 -0
- data/vendor/wreq/examples/cert_store.rs +77 -0
- data/vendor/wreq/examples/connect_via_lower_priority_tokio_runtime.rs +258 -0
- data/vendor/wreq/examples/emulation.rs +118 -0
- data/vendor/wreq/examples/form.rs +14 -0
- data/vendor/wreq/examples/http1_websocket.rs +37 -0
- data/vendor/wreq/examples/http2_websocket.rs +45 -0
- data/vendor/wreq/examples/json_dynamic.rs +41 -0
- data/vendor/wreq/examples/json_typed.rs +47 -0
- data/vendor/wreq/examples/keylog.rs +16 -0
- data/vendor/wreq/examples/request_with_emulation.rs +115 -0
- data/vendor/wreq/examples/request_with_interface.rs +37 -0
- data/vendor/wreq/examples/request_with_local_address.rs +16 -0
- data/vendor/wreq/examples/request_with_proxy.rs +13 -0
- data/vendor/wreq/examples/request_with_redirect.rs +22 -0
- data/vendor/wreq/examples/request_with_version.rs +15 -0
- data/vendor/wreq/examples/tor_socks.rs +24 -0
- data/vendor/wreq/examples/unix_socket.rs +33 -0
- data/vendor/wreq/src/client/body.rs +304 -0
- data/vendor/wreq/src/client/conn/conn.rs +231 -0
- data/vendor/wreq/src/client/conn/connector.rs +549 -0
- data/vendor/wreq/src/client/conn/http.rs +1023 -0
- data/vendor/wreq/src/client/conn/proxy/socks.rs +233 -0
- data/vendor/wreq/src/client/conn/proxy/tunnel.rs +260 -0
- data/vendor/wreq/src/client/conn/proxy.rs +39 -0
- data/vendor/wreq/src/client/conn/tls_info.rs +98 -0
- data/vendor/wreq/src/client/conn/uds.rs +44 -0
- data/vendor/wreq/src/client/conn/verbose.rs +149 -0
- data/vendor/wreq/src/client/conn.rs +323 -0
- data/vendor/wreq/src/client/core/body/incoming.rs +485 -0
- data/vendor/wreq/src/client/core/body/length.rs +118 -0
- data/vendor/wreq/src/client/core/body.rs +34 -0
- data/vendor/wreq/src/client/core/common/buf.rs +149 -0
- data/vendor/wreq/src/client/core/common/rewind.rs +141 -0
- data/vendor/wreq/src/client/core/common/watch.rs +76 -0
- data/vendor/wreq/src/client/core/common.rs +3 -0
- data/vendor/wreq/src/client/core/conn/http1.rs +342 -0
- data/vendor/wreq/src/client/core/conn/http2.rs +307 -0
- data/vendor/wreq/src/client/core/conn.rs +11 -0
- data/vendor/wreq/src/client/core/dispatch.rs +299 -0
- data/vendor/wreq/src/client/core/error.rs +435 -0
- data/vendor/wreq/src/client/core/ext.rs +201 -0
- data/vendor/wreq/src/client/core/http1.rs +178 -0
- data/vendor/wreq/src/client/core/http2.rs +483 -0
- data/vendor/wreq/src/client/core/proto/h1/conn.rs +988 -0
- data/vendor/wreq/src/client/core/proto/h1/decode.rs +1170 -0
- data/vendor/wreq/src/client/core/proto/h1/dispatch.rs +684 -0
- data/vendor/wreq/src/client/core/proto/h1/encode.rs +580 -0
- data/vendor/wreq/src/client/core/proto/h1/io.rs +879 -0
- data/vendor/wreq/src/client/core/proto/h1/role.rs +694 -0
- data/vendor/wreq/src/client/core/proto/h1.rs +104 -0
- data/vendor/wreq/src/client/core/proto/h2/client.rs +650 -0
- data/vendor/wreq/src/client/core/proto/h2/ping.rs +539 -0
- data/vendor/wreq/src/client/core/proto/h2.rs +379 -0
- data/vendor/wreq/src/client/core/proto/headers.rs +138 -0
- data/vendor/wreq/src/client/core/proto.rs +58 -0
- data/vendor/wreq/src/client/core/rt/bounds.rs +57 -0
- data/vendor/wreq/src/client/core/rt/timer.rs +150 -0
- data/vendor/wreq/src/client/core/rt/tokio.rs +99 -0
- data/vendor/wreq/src/client/core/rt.rs +25 -0
- data/vendor/wreq/src/client/core/upgrade.rs +267 -0
- data/vendor/wreq/src/client/core.rs +16 -0
- data/vendor/wreq/src/client/emulation.rs +161 -0
- data/vendor/wreq/src/client/http/client/error.rs +142 -0
- data/vendor/wreq/src/client/http/client/exec.rs +29 -0
- data/vendor/wreq/src/client/http/client/extra.rs +77 -0
- data/vendor/wreq/src/client/http/client/lazy.rs +79 -0
- data/vendor/wreq/src/client/http/client/pool.rs +1105 -0
- data/vendor/wreq/src/client/http/client/util.rs +104 -0
- data/vendor/wreq/src/client/http/client.rs +1003 -0
- data/vendor/wreq/src/client/http/future.rs +99 -0
- data/vendor/wreq/src/client/http.rs +1629 -0
- data/vendor/wreq/src/client/layer/config/options.rs +156 -0
- data/vendor/wreq/src/client/layer/config.rs +116 -0
- data/vendor/wreq/src/client/layer/cookie.rs +161 -0
- data/vendor/wreq/src/client/layer/decoder.rs +139 -0
- data/vendor/wreq/src/client/layer/redirect/future.rs +270 -0
- data/vendor/wreq/src/client/layer/redirect/policy.rs +63 -0
- data/vendor/wreq/src/client/layer/redirect.rs +145 -0
- data/vendor/wreq/src/client/layer/retry/classify.rs +105 -0
- data/vendor/wreq/src/client/layer/retry/scope.rs +51 -0
- data/vendor/wreq/src/client/layer/retry.rs +151 -0
- data/vendor/wreq/src/client/layer/timeout/body.rs +233 -0
- data/vendor/wreq/src/client/layer/timeout/future.rs +90 -0
- data/vendor/wreq/src/client/layer/timeout.rs +177 -0
- data/vendor/wreq/src/client/layer.rs +15 -0
- data/vendor/wreq/src/client/multipart.rs +717 -0
- data/vendor/wreq/src/client/request.rs +818 -0
- data/vendor/wreq/src/client/response.rs +534 -0
- data/vendor/wreq/src/client/ws/json.rs +99 -0
- data/vendor/wreq/src/client/ws/message.rs +453 -0
- data/vendor/wreq/src/client/ws.rs +714 -0
- data/vendor/wreq/src/client.rs +27 -0
- data/vendor/wreq/src/config.rs +140 -0
- data/vendor/wreq/src/cookie.rs +579 -0
- data/vendor/wreq/src/dns/gai.rs +249 -0
- data/vendor/wreq/src/dns/hickory.rs +78 -0
- data/vendor/wreq/src/dns/resolve.rs +180 -0
- data/vendor/wreq/src/dns.rs +69 -0
- data/vendor/wreq/src/error.rs +502 -0
- data/vendor/wreq/src/ext.rs +398 -0
- data/vendor/wreq/src/hash.rs +143 -0
- data/vendor/wreq/src/header.rs +506 -0
- data/vendor/wreq/src/into_uri.rs +187 -0
- data/vendor/wreq/src/lib.rs +586 -0
- data/vendor/wreq/src/proxy/mac.rs +82 -0
- data/vendor/wreq/src/proxy/matcher.rs +806 -0
- data/vendor/wreq/src/proxy/uds.rs +66 -0
- data/vendor/wreq/src/proxy/win.rs +31 -0
- data/vendor/wreq/src/proxy.rs +569 -0
- data/vendor/wreq/src/redirect.rs +575 -0
- data/vendor/wreq/src/retry.rs +198 -0
- data/vendor/wreq/src/sync.rs +129 -0
- data/vendor/wreq/src/tls/conn/cache.rs +123 -0
- data/vendor/wreq/src/tls/conn/cert_compression.rs +125 -0
- data/vendor/wreq/src/tls/conn/ext.rs +82 -0
- data/vendor/wreq/src/tls/conn/macros.rs +34 -0
- data/vendor/wreq/src/tls/conn/service.rs +138 -0
- data/vendor/wreq/src/tls/conn.rs +681 -0
- data/vendor/wreq/src/tls/keylog/handle.rs +64 -0
- data/vendor/wreq/src/tls/keylog.rs +99 -0
- data/vendor/wreq/src/tls/options.rs +464 -0
- data/vendor/wreq/src/tls/x509/identity.rs +122 -0
- data/vendor/wreq/src/tls/x509/parser.rs +71 -0
- data/vendor/wreq/src/tls/x509/store.rs +228 -0
- data/vendor/wreq/src/tls/x509.rs +68 -0
- data/vendor/wreq/src/tls.rs +154 -0
- data/vendor/wreq/src/trace.rs +55 -0
- data/vendor/wreq/src/util.rs +122 -0
- data/vendor/wreq/tests/badssl.rs +228 -0
- data/vendor/wreq/tests/brotli.rs +350 -0
- data/vendor/wreq/tests/client.rs +1098 -0
- data/vendor/wreq/tests/connector_layers.rs +227 -0
- data/vendor/wreq/tests/cookie.rs +306 -0
- data/vendor/wreq/tests/deflate.rs +347 -0
- data/vendor/wreq/tests/emulation.rs +260 -0
- data/vendor/wreq/tests/gzip.rs +347 -0
- data/vendor/wreq/tests/layers.rs +261 -0
- data/vendor/wreq/tests/multipart.rs +165 -0
- data/vendor/wreq/tests/proxy.rs +438 -0
- data/vendor/wreq/tests/redirect.rs +629 -0
- data/vendor/wreq/tests/retry.rs +135 -0
- data/vendor/wreq/tests/support/delay_server.rs +117 -0
- data/vendor/wreq/tests/support/error.rs +16 -0
- data/vendor/wreq/tests/support/layer.rs +183 -0
- data/vendor/wreq/tests/support/mod.rs +9 -0
- data/vendor/wreq/tests/support/server.rs +232 -0
- data/vendor/wreq/tests/timeouts.rs +281 -0
- data/vendor/wreq/tests/unix_socket.rs +135 -0
- data/vendor/wreq/tests/upgrade.rs +98 -0
- data/vendor/wreq/tests/zstd.rs +559 -0
- metadata +225 -0
|
@@ -0,0 +1,347 @@
|
|
|
1
|
+
mod support;
|
|
2
|
+
use std::io::Write;
|
|
3
|
+
|
|
4
|
+
use flate2::{Compression, write::ZlibEncoder};
|
|
5
|
+
use support::server;
|
|
6
|
+
use tokio::io::AsyncWriteExt;
|
|
7
|
+
|
|
8
|
+
#[tokio::test]
|
|
9
|
+
async fn deflate_response() {
|
|
10
|
+
deflate_case(10_000, 4096).await;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
#[tokio::test]
|
|
14
|
+
async fn deflate_single_byte_chunks() {
|
|
15
|
+
deflate_case(10, 1).await;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
#[tokio::test]
|
|
19
|
+
async fn test_deflate_empty_body() {
|
|
20
|
+
let server = server::http(move |req| async move {
|
|
21
|
+
assert_eq!(req.method(), "HEAD");
|
|
22
|
+
|
|
23
|
+
http::Response::builder()
|
|
24
|
+
.header("content-encoding", "deflate")
|
|
25
|
+
.body(Default::default())
|
|
26
|
+
.unwrap()
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
let res = wreq::head(format!("http://{}/deflate", server.addr()))
|
|
30
|
+
.send()
|
|
31
|
+
.await
|
|
32
|
+
.unwrap();
|
|
33
|
+
|
|
34
|
+
let body = res.text().await.unwrap();
|
|
35
|
+
|
|
36
|
+
assert_eq!(body, "");
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
#[tokio::test]
|
|
40
|
+
async fn test_accept_header_is_not_changed_if_set() {
|
|
41
|
+
let server = server::http(move |req| async move {
|
|
42
|
+
assert_eq!(req.headers()["accept"], "application/json");
|
|
43
|
+
assert!(
|
|
44
|
+
req.headers()["accept-encoding"]
|
|
45
|
+
.to_str()
|
|
46
|
+
.unwrap()
|
|
47
|
+
.contains("deflate")
|
|
48
|
+
);
|
|
49
|
+
http::Response::default()
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
let res = wreq::get(format!("http://{}/accept", server.addr()))
|
|
53
|
+
.header(
|
|
54
|
+
wreq::header::ACCEPT,
|
|
55
|
+
wreq::header::HeaderValue::from_static("application/json"),
|
|
56
|
+
)
|
|
57
|
+
.send()
|
|
58
|
+
.await
|
|
59
|
+
.unwrap();
|
|
60
|
+
|
|
61
|
+
assert_eq!(res.status(), wreq::StatusCode::OK);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
#[tokio::test]
|
|
65
|
+
async fn test_accept_encoding_header_is_not_changed_if_set() {
|
|
66
|
+
let server = server::http(move |req| async move {
|
|
67
|
+
assert_eq!(req.headers()["accept"], "*/*");
|
|
68
|
+
assert_eq!(req.headers()["accept-encoding"], "identity");
|
|
69
|
+
http::Response::default()
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
let res = wreq::get(format!("http://{}/accept-encoding", server.addr()))
|
|
73
|
+
.header(wreq::header::ACCEPT, "*/*")
|
|
74
|
+
.header(
|
|
75
|
+
wreq::header::ACCEPT_ENCODING,
|
|
76
|
+
wreq::header::HeaderValue::from_static("identity"),
|
|
77
|
+
)
|
|
78
|
+
.send()
|
|
79
|
+
.await
|
|
80
|
+
.unwrap();
|
|
81
|
+
|
|
82
|
+
assert_eq!(res.status(), wreq::StatusCode::OK);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
async fn deflate_case(response_size: usize, chunk_size: usize) {
|
|
86
|
+
use futures_util::stream::StreamExt;
|
|
87
|
+
|
|
88
|
+
let content: String = (0..response_size).fold(String::new(), |mut acc, i| {
|
|
89
|
+
acc.push_str(&format!("test {i}"));
|
|
90
|
+
acc
|
|
91
|
+
});
|
|
92
|
+
let mut encoder = ZlibEncoder::new(Vec::new(), Compression::default());
|
|
93
|
+
encoder.write_all(content.as_bytes()).unwrap();
|
|
94
|
+
let deflated_content = encoder.finish().unwrap();
|
|
95
|
+
|
|
96
|
+
let mut response = format!(
|
|
97
|
+
"\
|
|
98
|
+
HTTP/1.1 200 OK\r\n\
|
|
99
|
+
Server: test-accept\r\n\
|
|
100
|
+
Content-Encoding: deflate\r\n\
|
|
101
|
+
Content-Length: {}\r\n\
|
|
102
|
+
\r\n",
|
|
103
|
+
&deflated_content.len()
|
|
104
|
+
)
|
|
105
|
+
.into_bytes();
|
|
106
|
+
response.extend(&deflated_content);
|
|
107
|
+
|
|
108
|
+
let server = server::http(move |req| {
|
|
109
|
+
assert!(
|
|
110
|
+
req.headers()["accept-encoding"]
|
|
111
|
+
.to_str()
|
|
112
|
+
.unwrap()
|
|
113
|
+
.contains("deflate")
|
|
114
|
+
);
|
|
115
|
+
|
|
116
|
+
let deflated = deflated_content.clone();
|
|
117
|
+
async move {
|
|
118
|
+
let len = deflated.len();
|
|
119
|
+
let stream =
|
|
120
|
+
futures_util::stream::unfold((deflated, 0), move |(deflated, pos)| async move {
|
|
121
|
+
let chunk = deflated.chunks(chunk_size).nth(pos)?.to_vec();
|
|
122
|
+
|
|
123
|
+
Some((chunk, (deflated, pos + 1)))
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
let body = wreq::Body::wrap_stream(stream.map(Ok::<_, std::convert::Infallible>));
|
|
127
|
+
|
|
128
|
+
http::Response::builder()
|
|
129
|
+
.header("content-encoding", "deflate")
|
|
130
|
+
.header("content-length", len)
|
|
131
|
+
.body(body)
|
|
132
|
+
.unwrap()
|
|
133
|
+
}
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
let res = wreq::get(format!("http://{}/deflate", server.addr()))
|
|
137
|
+
.send()
|
|
138
|
+
.await
|
|
139
|
+
.expect("response");
|
|
140
|
+
|
|
141
|
+
let body = res.text().await.expect("text");
|
|
142
|
+
assert_eq!(body, content);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
const COMPRESSED_RESPONSE_HEADERS: &[u8] = b"HTTP/1.1 200 OK\x0d\x0a\
|
|
146
|
+
Content-Type: text/plain\x0d\x0a\
|
|
147
|
+
Connection: keep-alive\x0d\x0a\
|
|
148
|
+
Content-Encoding: deflate\x0d\x0a";
|
|
149
|
+
|
|
150
|
+
const RESPONSE_CONTENT: &str = "some message here";
|
|
151
|
+
|
|
152
|
+
fn deflate_compress(input: &[u8]) -> Vec<u8> {
|
|
153
|
+
let mut encoder = ZlibEncoder::new(Vec::new(), Compression::default());
|
|
154
|
+
encoder.write_all(input).unwrap();
|
|
155
|
+
encoder.finish().unwrap()
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
#[tokio::test]
|
|
159
|
+
async fn test_non_chunked_non_fragmented_response() {
|
|
160
|
+
let server = server::low_level_with_response(|_raw_request, client_socket| {
|
|
161
|
+
Box::new(async move {
|
|
162
|
+
let deflated_content = deflate_compress(RESPONSE_CONTENT.as_bytes());
|
|
163
|
+
let content_length_header =
|
|
164
|
+
format!("Content-Length: {}\r\n\r\n", deflated_content.len()).into_bytes();
|
|
165
|
+
let response = [
|
|
166
|
+
COMPRESSED_RESPONSE_HEADERS,
|
|
167
|
+
&content_length_header,
|
|
168
|
+
&deflated_content,
|
|
169
|
+
]
|
|
170
|
+
.concat();
|
|
171
|
+
|
|
172
|
+
client_socket
|
|
173
|
+
.write_all(response.as_slice())
|
|
174
|
+
.await
|
|
175
|
+
.expect("response write_all failed");
|
|
176
|
+
client_socket.flush().await.expect("response flush failed");
|
|
177
|
+
})
|
|
178
|
+
});
|
|
179
|
+
|
|
180
|
+
let res = wreq::get(format!("http://{}/", server.addr()))
|
|
181
|
+
.send()
|
|
182
|
+
.await
|
|
183
|
+
.expect("response");
|
|
184
|
+
|
|
185
|
+
assert_eq!(res.text().await.expect("text"), RESPONSE_CONTENT);
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
#[tokio::test]
|
|
189
|
+
async fn test_chunked_fragmented_response_1() {
|
|
190
|
+
const DELAY_BETWEEN_RESPONSE_PARTS: tokio::time::Duration =
|
|
191
|
+
tokio::time::Duration::from_millis(1000);
|
|
192
|
+
const DELAY_MARGIN: tokio::time::Duration = tokio::time::Duration::from_millis(50);
|
|
193
|
+
|
|
194
|
+
let server = server::low_level_with_response(|_raw_request, client_socket| {
|
|
195
|
+
Box::new(async move {
|
|
196
|
+
let deflated_content = deflate_compress(RESPONSE_CONTENT.as_bytes());
|
|
197
|
+
let response_first_part = [
|
|
198
|
+
COMPRESSED_RESPONSE_HEADERS,
|
|
199
|
+
format!(
|
|
200
|
+
"Transfer-Encoding: chunked\r\n\r\n{:x}\r\n",
|
|
201
|
+
deflated_content.len()
|
|
202
|
+
)
|
|
203
|
+
.as_bytes(),
|
|
204
|
+
&deflated_content,
|
|
205
|
+
]
|
|
206
|
+
.concat();
|
|
207
|
+
let response_second_part = b"\r\n0\r\n\r\n";
|
|
208
|
+
|
|
209
|
+
client_socket
|
|
210
|
+
.write_all(response_first_part.as_slice())
|
|
211
|
+
.await
|
|
212
|
+
.expect("response_first_part write_all failed");
|
|
213
|
+
client_socket
|
|
214
|
+
.flush()
|
|
215
|
+
.await
|
|
216
|
+
.expect("response_first_part flush failed");
|
|
217
|
+
|
|
218
|
+
tokio::time::sleep(DELAY_BETWEEN_RESPONSE_PARTS).await;
|
|
219
|
+
|
|
220
|
+
client_socket
|
|
221
|
+
.write_all(response_second_part)
|
|
222
|
+
.await
|
|
223
|
+
.expect("response_second_part write_all failed");
|
|
224
|
+
client_socket
|
|
225
|
+
.flush()
|
|
226
|
+
.await
|
|
227
|
+
.expect("response_second_part flush failed");
|
|
228
|
+
})
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
let start = tokio::time::Instant::now();
|
|
232
|
+
let res = wreq::get(format!("http://{}/", server.addr()))
|
|
233
|
+
.send()
|
|
234
|
+
.await
|
|
235
|
+
.expect("response");
|
|
236
|
+
|
|
237
|
+
assert_eq!(res.text().await.expect("text"), RESPONSE_CONTENT);
|
|
238
|
+
assert!(start.elapsed() >= DELAY_BETWEEN_RESPONSE_PARTS - DELAY_MARGIN);
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
#[tokio::test]
|
|
242
|
+
async fn test_chunked_fragmented_response_2() {
|
|
243
|
+
const DELAY_BETWEEN_RESPONSE_PARTS: tokio::time::Duration =
|
|
244
|
+
tokio::time::Duration::from_millis(1000);
|
|
245
|
+
const DELAY_MARGIN: tokio::time::Duration = tokio::time::Duration::from_millis(50);
|
|
246
|
+
|
|
247
|
+
let server = server::low_level_with_response(|_raw_request, client_socket| {
|
|
248
|
+
Box::new(async move {
|
|
249
|
+
let deflated_content = deflate_compress(RESPONSE_CONTENT.as_bytes());
|
|
250
|
+
let response_first_part = [
|
|
251
|
+
COMPRESSED_RESPONSE_HEADERS,
|
|
252
|
+
format!(
|
|
253
|
+
"Transfer-Encoding: chunked\r\n\r\n{:x}\r\n",
|
|
254
|
+
deflated_content.len()
|
|
255
|
+
)
|
|
256
|
+
.as_bytes(),
|
|
257
|
+
&deflated_content,
|
|
258
|
+
b"\r\n",
|
|
259
|
+
]
|
|
260
|
+
.concat();
|
|
261
|
+
let response_second_part = b"0\r\n\r\n";
|
|
262
|
+
|
|
263
|
+
client_socket
|
|
264
|
+
.write_all(response_first_part.as_slice())
|
|
265
|
+
.await
|
|
266
|
+
.expect("response_first_part write_all failed");
|
|
267
|
+
client_socket
|
|
268
|
+
.flush()
|
|
269
|
+
.await
|
|
270
|
+
.expect("response_first_part flush failed");
|
|
271
|
+
|
|
272
|
+
tokio::time::sleep(DELAY_BETWEEN_RESPONSE_PARTS).await;
|
|
273
|
+
|
|
274
|
+
client_socket
|
|
275
|
+
.write_all(response_second_part)
|
|
276
|
+
.await
|
|
277
|
+
.expect("response_second_part write_all failed");
|
|
278
|
+
client_socket
|
|
279
|
+
.flush()
|
|
280
|
+
.await
|
|
281
|
+
.expect("response_second_part flush failed");
|
|
282
|
+
})
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
let start = tokio::time::Instant::now();
|
|
286
|
+
let res = wreq::get(format!("http://{}/", server.addr()))
|
|
287
|
+
.send()
|
|
288
|
+
.await
|
|
289
|
+
.expect("response");
|
|
290
|
+
|
|
291
|
+
assert_eq!(res.text().await.expect("text"), RESPONSE_CONTENT);
|
|
292
|
+
assert!(start.elapsed() >= DELAY_BETWEEN_RESPONSE_PARTS - DELAY_MARGIN);
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
#[tokio::test]
|
|
296
|
+
async fn test_chunked_fragmented_response_with_extra_bytes() {
|
|
297
|
+
const DELAY_BETWEEN_RESPONSE_PARTS: tokio::time::Duration =
|
|
298
|
+
tokio::time::Duration::from_millis(1000);
|
|
299
|
+
const DELAY_MARGIN: tokio::time::Duration = tokio::time::Duration::from_millis(50);
|
|
300
|
+
|
|
301
|
+
let server = server::low_level_with_response(|_raw_request, client_socket| {
|
|
302
|
+
Box::new(async move {
|
|
303
|
+
let deflated_content = deflate_compress(RESPONSE_CONTENT.as_bytes());
|
|
304
|
+
let response_first_part = [
|
|
305
|
+
COMPRESSED_RESPONSE_HEADERS,
|
|
306
|
+
format!(
|
|
307
|
+
"Transfer-Encoding: chunked\r\n\r\n{:x}\r\n",
|
|
308
|
+
deflated_content.len()
|
|
309
|
+
)
|
|
310
|
+
.as_bytes(),
|
|
311
|
+
&deflated_content,
|
|
312
|
+
]
|
|
313
|
+
.concat();
|
|
314
|
+
let response_second_part = b"\r\n2ab\r\n0\r\n\r\n";
|
|
315
|
+
|
|
316
|
+
client_socket
|
|
317
|
+
.write_all(response_first_part.as_slice())
|
|
318
|
+
.await
|
|
319
|
+
.expect("response_first_part write_all failed");
|
|
320
|
+
client_socket
|
|
321
|
+
.flush()
|
|
322
|
+
.await
|
|
323
|
+
.expect("response_first_part flush failed");
|
|
324
|
+
|
|
325
|
+
tokio::time::sleep(DELAY_BETWEEN_RESPONSE_PARTS).await;
|
|
326
|
+
|
|
327
|
+
client_socket
|
|
328
|
+
.write_all(response_second_part)
|
|
329
|
+
.await
|
|
330
|
+
.expect("response_second_part write_all failed");
|
|
331
|
+
client_socket
|
|
332
|
+
.flush()
|
|
333
|
+
.await
|
|
334
|
+
.expect("response_second_part flush failed");
|
|
335
|
+
})
|
|
336
|
+
});
|
|
337
|
+
|
|
338
|
+
let start = tokio::time::Instant::now();
|
|
339
|
+
let res = wreq::get(format!("http://{}/", server.addr()))
|
|
340
|
+
.send()
|
|
341
|
+
.await
|
|
342
|
+
.expect("response");
|
|
343
|
+
|
|
344
|
+
let err = res.text().await.expect_err("there must be an error");
|
|
345
|
+
assert!(err.is_decode());
|
|
346
|
+
assert!(start.elapsed() >= DELAY_BETWEEN_RESPONSE_PARTS - DELAY_MARGIN);
|
|
347
|
+
}
|
|
@@ -0,0 +1,260 @@
|
|
|
1
|
+
use std::time::Duration;
|
|
2
|
+
|
|
3
|
+
use wreq::{
|
|
4
|
+
Client, Emulation,
|
|
5
|
+
http1::Http1Options,
|
|
6
|
+
http2::{
|
|
7
|
+
Http2Options, PseudoId, PseudoOrder, SettingId, SettingsOrder, StreamDependency, StreamId,
|
|
8
|
+
},
|
|
9
|
+
tls::{AlpnProtocol, CertificateCompressionAlgorithm, ExtensionType, TlsOptions, TlsVersion},
|
|
10
|
+
};
|
|
11
|
+
|
|
12
|
+
macro_rules! join {
|
|
13
|
+
($sep:expr, $first:expr $(, $rest:expr)*) => {
|
|
14
|
+
concat!($first $(, $sep, $rest)*)
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
fn tls_options_template() -> TlsOptions {
|
|
19
|
+
// TLS options config
|
|
20
|
+
TlsOptions::builder()
|
|
21
|
+
.curves_list(join!(
|
|
22
|
+
":",
|
|
23
|
+
"X25519MLKEM768",
|
|
24
|
+
"X25519",
|
|
25
|
+
"P-256",
|
|
26
|
+
"P-384",
|
|
27
|
+
"P-521",
|
|
28
|
+
"ffdhe2048",
|
|
29
|
+
"ffdhe3072"
|
|
30
|
+
))
|
|
31
|
+
.cipher_list(join!(
|
|
32
|
+
":",
|
|
33
|
+
"TLS_AES_128_GCM_SHA256",
|
|
34
|
+
"TLS_CHACHA20_POLY1305_SHA256",
|
|
35
|
+
"TLS_AES_256_GCM_SHA384",
|
|
36
|
+
"TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256",
|
|
37
|
+
"TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
|
|
38
|
+
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256",
|
|
39
|
+
"TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256",
|
|
40
|
+
"TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384",
|
|
41
|
+
"TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384",
|
|
42
|
+
"TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA",
|
|
43
|
+
"TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA",
|
|
44
|
+
"TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA",
|
|
45
|
+
"TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA",
|
|
46
|
+
"TLS_RSA_WITH_AES_128_GCM_SHA256",
|
|
47
|
+
"TLS_RSA_WITH_AES_256_GCM_SHA384",
|
|
48
|
+
"TLS_RSA_WITH_AES_128_CBC_SHA",
|
|
49
|
+
"TLS_RSA_WITH_AES_256_CBC_SHA"
|
|
50
|
+
))
|
|
51
|
+
.sigalgs_list(join!(
|
|
52
|
+
":",
|
|
53
|
+
"ecdsa_secp256r1_sha256",
|
|
54
|
+
"ecdsa_secp384r1_sha384",
|
|
55
|
+
"ecdsa_secp521r1_sha512",
|
|
56
|
+
"rsa_pss_rsae_sha256",
|
|
57
|
+
"rsa_pss_rsae_sha384",
|
|
58
|
+
"rsa_pss_rsae_sha512",
|
|
59
|
+
"rsa_pkcs1_sha256",
|
|
60
|
+
"rsa_pkcs1_sha384",
|
|
61
|
+
"rsa_pkcs1_sha512",
|
|
62
|
+
"ecdsa_sha1",
|
|
63
|
+
"rsa_pkcs1_sha1"
|
|
64
|
+
))
|
|
65
|
+
.delegated_credentials(join!(
|
|
66
|
+
":",
|
|
67
|
+
"ecdsa_secp256r1_sha256",
|
|
68
|
+
"ecdsa_secp384r1_sha384",
|
|
69
|
+
"ecdsa_secp521r1_sha512",
|
|
70
|
+
"ecdsa_sha1"
|
|
71
|
+
))
|
|
72
|
+
.certificate_compression_algorithms(&[
|
|
73
|
+
CertificateCompressionAlgorithm::ZLIB,
|
|
74
|
+
CertificateCompressionAlgorithm::BROTLI,
|
|
75
|
+
CertificateCompressionAlgorithm::ZSTD,
|
|
76
|
+
])
|
|
77
|
+
.alpn_protocols([AlpnProtocol::HTTP2, AlpnProtocol::HTTP1])
|
|
78
|
+
.record_size_limit(0x4001)
|
|
79
|
+
.pre_shared_key(true)
|
|
80
|
+
.enable_ech_grease(true)
|
|
81
|
+
.enable_ocsp_stapling(true)
|
|
82
|
+
.enable_signed_cert_timestamps(true)
|
|
83
|
+
.min_tls_version(TlsVersion::TLS_1_2)
|
|
84
|
+
.max_tls_version(TlsVersion::TLS_1_3)
|
|
85
|
+
.key_shares_limit(3)
|
|
86
|
+
.preserve_tls13_cipher_list(true)
|
|
87
|
+
.aes_hw_override(true)
|
|
88
|
+
.random_aes_hw_override(true)
|
|
89
|
+
.extension_permutation(&[
|
|
90
|
+
ExtensionType::SERVER_NAME,
|
|
91
|
+
ExtensionType::EXTENDED_MASTER_SECRET,
|
|
92
|
+
ExtensionType::RENEGOTIATE,
|
|
93
|
+
ExtensionType::SUPPORTED_GROUPS,
|
|
94
|
+
ExtensionType::EC_POINT_FORMATS,
|
|
95
|
+
ExtensionType::SESSION_TICKET,
|
|
96
|
+
ExtensionType::APPLICATION_LAYER_PROTOCOL_NEGOTIATION,
|
|
97
|
+
ExtensionType::STATUS_REQUEST,
|
|
98
|
+
ExtensionType::DELEGATED_CREDENTIAL,
|
|
99
|
+
ExtensionType::KEY_SHARE,
|
|
100
|
+
ExtensionType::SUPPORTED_VERSIONS,
|
|
101
|
+
ExtensionType::SIGNATURE_ALGORITHMS,
|
|
102
|
+
ExtensionType::PSK_KEY_EXCHANGE_MODES,
|
|
103
|
+
ExtensionType::RECORD_SIZE_LIMIT,
|
|
104
|
+
ExtensionType::CERT_COMPRESSION,
|
|
105
|
+
ExtensionType::ENCRYPTED_CLIENT_HELLO,
|
|
106
|
+
])
|
|
107
|
+
.build()
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
fn http2_options_template() -> Http2Options {
|
|
111
|
+
// HTTP/2 headers frame pseudo-header order
|
|
112
|
+
let headers_pseudo_order = PseudoOrder::builder()
|
|
113
|
+
.extend([
|
|
114
|
+
PseudoId::Method,
|
|
115
|
+
PseudoId::Path,
|
|
116
|
+
PseudoId::Authority,
|
|
117
|
+
PseudoId::Scheme,
|
|
118
|
+
])
|
|
119
|
+
.build();
|
|
120
|
+
|
|
121
|
+
// HTTP/2 settings frame order
|
|
122
|
+
let settings_order = SettingsOrder::builder()
|
|
123
|
+
.extend([
|
|
124
|
+
SettingId::HeaderTableSize,
|
|
125
|
+
SettingId::EnablePush,
|
|
126
|
+
SettingId::MaxConcurrentStreams,
|
|
127
|
+
SettingId::InitialWindowSize,
|
|
128
|
+
SettingId::MaxFrameSize,
|
|
129
|
+
SettingId::MaxHeaderListSize,
|
|
130
|
+
SettingId::EnableConnectProtocol,
|
|
131
|
+
SettingId::NoRfc7540Priorities,
|
|
132
|
+
])
|
|
133
|
+
.build();
|
|
134
|
+
|
|
135
|
+
Http2Options::builder()
|
|
136
|
+
.header_table_size(65536)
|
|
137
|
+
.enable_push(false)
|
|
138
|
+
.initial_window_size(131072)
|
|
139
|
+
.max_frame_size(16384)
|
|
140
|
+
.initial_connection_window_size(12517377 + 65535)
|
|
141
|
+
.headers_stream_dependency(StreamDependency::new(StreamId::ZERO, 41, false))
|
|
142
|
+
.headers_pseudo_order(headers_pseudo_order)
|
|
143
|
+
.settings_order(settings_order)
|
|
144
|
+
.build()
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
fn emulation_template() -> Emulation {
|
|
148
|
+
// HTTP/1 options config
|
|
149
|
+
let http1 = Http1Options::builder()
|
|
150
|
+
.allow_obsolete_multiline_headers_in_responses(true)
|
|
151
|
+
.max_headers(100)
|
|
152
|
+
.build();
|
|
153
|
+
|
|
154
|
+
// This provider encapsulates TLS, HTTP/1, HTTP/2, default headers, and original headers
|
|
155
|
+
Emulation::builder()
|
|
156
|
+
.tls_options(tls_options_template())
|
|
157
|
+
.http1_options(http1)
|
|
158
|
+
.http2_options(http2_options_template())
|
|
159
|
+
.build()
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
#[tokio::test]
|
|
163
|
+
async fn test_emulation() -> wreq::Result<()> {
|
|
164
|
+
let client = Client::builder()
|
|
165
|
+
.emulation(emulation_template())
|
|
166
|
+
.connect_timeout(Duration::from_secs(10))
|
|
167
|
+
.cert_verification(false)
|
|
168
|
+
.build()?;
|
|
169
|
+
|
|
170
|
+
let text = client
|
|
171
|
+
.get("https://tls.browserleaks.com/")
|
|
172
|
+
.send()
|
|
173
|
+
.await?
|
|
174
|
+
.text()
|
|
175
|
+
.await?;
|
|
176
|
+
|
|
177
|
+
assert!(
|
|
178
|
+
text.contains("t13d1717h2_5b57614c22b0_3cbfd9057e0d"),
|
|
179
|
+
"Response ja4_hash fingerprint not found: {text}"
|
|
180
|
+
);
|
|
181
|
+
assert!(
|
|
182
|
+
text.contains("6ea73faa8fc5aac76bded7bd238f6433"),
|
|
183
|
+
"Response akamai_hash fingerprint not found: {text}"
|
|
184
|
+
);
|
|
185
|
+
|
|
186
|
+
Ok(())
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
#[tokio::test]
|
|
190
|
+
async fn test_request_with_emulation() -> wreq::Result<()> {
|
|
191
|
+
let client = Client::builder()
|
|
192
|
+
.connect_timeout(Duration::from_secs(10))
|
|
193
|
+
.cert_verification(false)
|
|
194
|
+
.build()?;
|
|
195
|
+
|
|
196
|
+
let text = client
|
|
197
|
+
.get("https://tls.browserleaks.com/")
|
|
198
|
+
.emulation(emulation_template())
|
|
199
|
+
.send()
|
|
200
|
+
.await?
|
|
201
|
+
.text()
|
|
202
|
+
.await?;
|
|
203
|
+
|
|
204
|
+
assert!(
|
|
205
|
+
text.contains("t13d1717h2_5b57614c22b0_3cbfd9057e0d"),
|
|
206
|
+
"Response ja4_hash fingerprint not found: {text}"
|
|
207
|
+
);
|
|
208
|
+
assert!(
|
|
209
|
+
text.contains("6ea73faa8fc5aac76bded7bd238f6433"),
|
|
210
|
+
"Response akamai_hash fingerprint not found: {text}"
|
|
211
|
+
);
|
|
212
|
+
|
|
213
|
+
Ok(())
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
#[tokio::test]
|
|
217
|
+
async fn test_request_with_emulation_tls() -> wreq::Result<()> {
|
|
218
|
+
let client = Client::builder()
|
|
219
|
+
.connect_timeout(Duration::from_secs(10))
|
|
220
|
+
.cert_verification(false)
|
|
221
|
+
.build()?;
|
|
222
|
+
|
|
223
|
+
let text = client
|
|
224
|
+
.get("https://tls.browserleaks.com/")
|
|
225
|
+
.emulation(tls_options_template())
|
|
226
|
+
.send()
|
|
227
|
+
.await?
|
|
228
|
+
.text()
|
|
229
|
+
.await?;
|
|
230
|
+
|
|
231
|
+
assert!(
|
|
232
|
+
text.contains("t13d1717h2_5b57614c22b0_3cbfd9057e0d"),
|
|
233
|
+
"Response ja4_hash fingerprint not found: {text}"
|
|
234
|
+
);
|
|
235
|
+
|
|
236
|
+
Ok(())
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
#[tokio::test]
|
|
240
|
+
async fn test_request_with_emulation_http2() -> wreq::Result<()> {
|
|
241
|
+
let client = Client::builder()
|
|
242
|
+
.connect_timeout(Duration::from_secs(10))
|
|
243
|
+
.cert_verification(false)
|
|
244
|
+
.build()?;
|
|
245
|
+
|
|
246
|
+
let text = client
|
|
247
|
+
.get("https://tls.browserleaks.com/")
|
|
248
|
+
.emulation(http2_options_template())
|
|
249
|
+
.send()
|
|
250
|
+
.await?
|
|
251
|
+
.text()
|
|
252
|
+
.await?;
|
|
253
|
+
|
|
254
|
+
assert!(
|
|
255
|
+
text.contains("6ea73faa8fc5aac76bded7bd238f6433"),
|
|
256
|
+
"Response akamai_hash fingerprint not found: {text}"
|
|
257
|
+
);
|
|
258
|
+
|
|
259
|
+
Ok(())
|
|
260
|
+
}
|