1
//! Connect to relays via a proxy.
2
//!
3
//! This code is here for two reasons:
4
//!   1. To connect via external pluggable transports (for which we use SOCKS to
5
//!      build our connections).
6
//!   2. To support users who are behind a firewall that requires them to use a
7
//!      SOCKS proxy to connect.
8
//!
9
//! Supports SOCKS4/4a/5 and HTTP CONNECT proxies.
10
//
11
// TODO: Maybe refactor this so that tor-ptmgr can exist in a more freestanding
12
// way, with fewer arti dependencies.
13
#![allow(dead_code)]
14

            
15
use std::{
16
    net::{IpAddr, SocketAddr},
17
    sync::Arc,
18
};
19

            
20
use base64ct::{Base64, Encoding};
21
use futures::io::{AsyncBufReadExt, BufReader};
22
use futures::{AsyncReadExt, AsyncWriteExt};
23
use httparse;
24
use safelog::Sensitive;
25
use tor_linkspec::PtTargetAddr;
26
use tor_rtcompat::NetStreamProvider;
27
use tor_socksproto::{
28
    Handshake as _, SocksAddr, SocksAuth, SocksClientHandshake, SocksCmd, SocksRequest,
29
    SocksStatus, SocksVersion,
30
};
31
use tracing::trace;
32

            
33
#[cfg(feature = "pt-client")]
34
use super::TransportImplHelper;
35
#[cfg(feature = "pt-client")]
36
use async_trait::async_trait;
37
#[cfg(feature = "pt-client")]
38
use safelog::sensitive as sv;
39
#[cfg(feature = "pt-client")]
40
use tor_error::bad_api_usage;
41
#[cfg(feature = "pt-client")]
42
use tor_linkspec::{ChannelMethod, HasChanMethod, OwnedChanTarget};
43
#[cfg(feature = "pt-client")]
44
use tor_proto::peer::PeerAddr;
45

            
46
/// Information about what proxy protocol to use, and how to use it.
47
#[derive(Clone, Debug, Eq, PartialEq)]
48
#[non_exhaustive]
49
pub enum Protocol {
50
    /// Connect via SOCKS 4, SOCKS 4a, or SOCKS 5.
51
    Socks(SocksVersion, SocksAuth),
52
    /// Connect via HTTP CONNECT proxy (RFC 7231).
53
    HttpConnect {
54
        /// Optional Basic auth credentials (username, password) for Proxy-Authorization header.
55
        auth: Option<(Sensitive<String>, Sensitive<String>)>,
56
    },
57
}
58

            
59
/// An address to use when told to connect to "no address."
60
const NO_ADDR: IpAddr = IpAddr::V4(std::net::Ipv4Addr::new(0, 0, 0, 1));
61
/// Maximum number of bytes allowed in HTTP response headers from proxy.
62
const MAX_HTTP_HEADER_BYTES: usize = 16 * 1024;
63

            
64
/// Open a connection to `target` via the proxy at `proxy`, using the protocol
65
/// at `protocol`.
66
///
67
/// # Limitations
68
///
69
/// We will give an error if the proxy sends us any data on the connection along
70
/// with its final handshake: due to our implementation, any such data will be
71
/// discarded, and so we give an error rather than fail silently.
72
///
73
/// This limitation doesn't matter when the underlying protocol is Tor, or
74
/// anything else where the initiator is expected to speak before the responder
75
/// says anything.  To lift it, we would have to make this function's return
76
/// type become something buffered.
77
//
78
// TODO: Perhaps we should refactor this someday so it can be a general-purpose
79
// proxy function, not only for Arti.
80
pub(crate) async fn connect_via_proxy<R: NetStreamProvider + Send + Sync>(
81
    runtime: &R,
82
    proxy: &SocketAddr,
83
    protocol: &Protocol,
84
    target: &PtTargetAddr,
85
) -> Result<R::Stream, ProxyError> {
86
    trace!(
87
        "Launching a proxied connection to {} via proxy at {} using {:?}",
88
        target, proxy, protocol
89
    );
90
    // We don't (yet) use any custom options on the socket.
91
    let connect_options = Default::default();
92
    let stream = runtime
93
        .connect(proxy, &connect_options)
94
        .await
95
        .map_err(|e| ProxyError::ProxyConnect(Arc::new(e)))?;
96

            
97
    match protocol {
98
        Protocol::Socks(version, auth) => {
99
            do_socks_handshake::<R>(stream, version, auth, target).await
100
        }
101
        Protocol::HttpConnect { auth } => {
102
            do_http_connect_handshake::<R>(stream, auth, target).await
103
        }
104
    }
105
}
106

            
107
/// Perform SOCKS proxy handshake.
108
async fn do_socks_handshake<R: NetStreamProvider + Send + Sync>(
109
    mut stream: R::Stream,
110
    version: &SocksVersion,
111
    auth: &SocksAuth,
112
    target: &PtTargetAddr,
113
) -> Result<R::Stream, ProxyError> {
114
    let (target_addr, target_port): (SocksAddr, u16) = match target {
115
        PtTargetAddr::IpPort(a) => (SocksAddr::Ip(a.ip()), a.port()),
116
        #[cfg(feature = "pt-client")]
117
        PtTargetAddr::HostPort(host, port) => (
118
            SocksAddr::Hostname(
119
                host.clone()
120
                    .try_into()
121
                    .map_err(ProxyError::InvalidSocksAddr)?,
122
            ),
123
            *port,
124
        ),
125
        #[cfg(feature = "pt-client")]
126
        PtTargetAddr::None => (SocksAddr::Ip(NO_ADDR), 1),
127
        _ => return Err(ProxyError::UnrecognizedAddr),
128
    };
129

            
130
    let request = SocksRequest::new(
131
        *version,
132
        SocksCmd::CONNECT,
133
        target_addr,
134
        target_port,
135
        auth.clone(),
136
    )
137
    .map_err(ProxyError::InvalidSocksRequest)?;
138
    let mut handshake = SocksClientHandshake::new(request);
139

            
140
    let mut buf = tor_socksproto::Buffer::new();
141
    let reply = loop {
142
        use tor_socksproto::NextStep as NS;
143
        match handshake.step(&mut buf).map_err(ProxyError::SocksProto)? {
144
            NS::Send(send) => {
145
                stream.write_all(&send).await?;
146
                stream.flush().await?;
147
            }
148
            NS::Finished(fin) => {
149
                break fin
150
                    .into_output_forbid_pipelining()
151
                    .map_err(ProxyError::SocksProto)?;
152
            }
153
            NS::Recv(mut recv) => {
154
                let n = stream.read(recv.buf()).await?;
155
                recv.note_received(n).map_err(ProxyError::SocksProto)?;
156
            }
157
        }
158
    };
159

            
160
    let status = reply.status();
161
    trace!("SOCKS handshake succeeded, status {:?}", status);
162

            
163
    if status != SocksStatus::SUCCEEDED {
164
        return Err(ProxyError::SocksError(status));
165
    }
166

            
167
    Ok(stream)
168
}
169

            
170
/// Format target address for HTTP CONNECT request line and Host header.
171
fn format_connect_target(target: &PtTargetAddr) -> Result<String, ProxyError> {
172
    match target {
173
        PtTargetAddr::IpPort(a) => {
174
            let host = match a.ip() {
175
                IpAddr::V4(ip) => ip.to_string(),
176
                IpAddr::V6(ip) => format!("[{}]", ip),
177
            };
178
            Ok(format!("{}:{}", host, a.port()))
179
        }
180
        #[cfg(feature = "pt-client")]
181
        PtTargetAddr::HostPort(host, port) => Ok(format!("{}:{}", host, port)),
182
        #[cfg(feature = "pt-client")]
183
        PtTargetAddr::None => Err(ProxyError::UnrecognizedAddr),
184
        _ => Err(ProxyError::UnrecognizedAddr),
185
    }
186
}
187

            
188
/// Build HTTP CONNECT request string with optional Basic auth.
189
fn build_http_connect_request(
190
    target_str: &str,
191
    auth: &Option<(Sensitive<String>, Sensitive<String>)>,
192
) -> String {
193
    // Build CONNECT request: CONNECT host:port HTTP/1.1\r\nHost: host:port\r\n[...]\r\n
194
    let mut request = format!(
195
        "CONNECT {} HTTP/1.1\r\nHost: {}\r\n",
196
        target_str, target_str
197
    );
198

            
199
    if let Some((user, pass)) = auth {
200
        // Proxy-Authorization: Basic base64(username:password) per RFC 7617
201
        let credentials = format!("{}:{}", user.as_ref(), pass.as_ref());
202
        let encoded = Base64::encode_string(credentials.as_bytes());
203
        request.push_str(&format!("Proxy-Authorization: Basic {}\r\n", encoded));
204
    }
205

            
206
    request.push_str("\r\n");
207
    request
208
}
209

            
210
/// Parse HTTP CONNECT response and extract status code.
211
///
212
/// Uses httparse for spec-compliant parsing. Rejects:
213
/// - Responses larger than 16KB (prevents header bomb attacks)
214
/// - Pipelined data after headers (connection is dedicated to tunnel)
215
12
fn parse_http_connect_response(response_bytes: &[u8]) -> Result<u16, ProxyError> {
216
12
    if response_bytes.len() > MAX_HTTP_HEADER_BYTES {
217
2
        return Err(ProxyError::HttpConnectMalformed);
218
10
    }
219

            
220
10
    let mut headers = [httparse::EMPTY_HEADER; 64];
221
10
    let mut resp = httparse::Response::new(&mut headers);
222

            
223
10
    match resp.parse(response_bytes) {
224
8
        Ok(httparse::Status::Complete(header_end)) => {
225
8
            let status = resp.code.ok_or(ProxyError::HttpConnectMalformed)?;
226

            
227
8
            if !(200..300).contains(&status) {
228
2
                return Err(ProxyError::HttpConnectError(status));
229
6
            }
230

            
231
            // Reject any pipelined data after headers
232
6
            if header_end < response_bytes.len() {
233
2
                return Err(ProxyError::UnexpectedData);
234
4
            }
235

            
236
4
            trace!("HTTP CONNECT successful, status {}", status);
237
4
            Ok(status)
238
        }
239
        Ok(httparse::Status::Partial) => Err(ProxyError::HttpConnectMalformed),
240
2
        Err(_) => Err(ProxyError::HttpConnectMalformed),
241
    }
242
12
}
243

            
244
/// Send HTTP CONNECT request to proxy.
245
async fn send_http_connect_request<R: NetStreamProvider + Send + Sync>(
246
    stream: &mut R::Stream,
247
    auth: &Option<(Sensitive<String>, Sensitive<String>)>,
248
    target_str: &str,
249
) -> Result<(), ProxyError> {
250
    let request = build_http_connect_request(target_str, auth);
251
    trace!("Sending HTTP CONNECT request for {}", target_str);
252
    stream.write_all(request.as_bytes()).await?;
253
    stream.flush().await?;
254
    Ok(())
255
}
256

            
257
/// Perform HTTP CONNECT proxy handshake (RFC 7231, RFC 7617 for Basic auth).
258
async fn do_http_connect_handshake<R: NetStreamProvider + Send + Sync>(
259
    mut stream: R::Stream,
260
    auth: &Option<(Sensitive<String>, Sensitive<String>)>,
261
    target: &PtTargetAddr,
262
) -> Result<R::Stream, ProxyError> {
263
    let target_str = format_connect_target(target)?;
264
    send_http_connect_request::<R>(&mut stream, auth, &target_str).await?;
265

            
266
    // Read response until we see the double CRLF that terminates headers
267
    let mut response_buffer = Vec::new();
268
    let mut reader = BufReader::new(stream);
269
    let mut line = String::new();
270

            
271
    loop {
272
        line.clear();
273
        let n = reader.read_line(&mut line).await?;
274
        if n == 0 {
275
            return Err(ProxyError::HttpConnectMalformed);
276
        }
277

            
278
        response_buffer.extend_from_slice(line.as_bytes());
279
        if response_buffer.len() > MAX_HTTP_HEADER_BYTES {
280
            return Err(ProxyError::HttpConnectMalformed);
281
        }
282

            
283
        // Check for blank line (end of headers)
284
        if line == "\r\n" || line == "\n" {
285
            break;
286
        }
287
    }
288

            
289
    // Parse and validate response
290
    let _status_code = parse_http_connect_response(&response_buffer)?;
291

            
292
    // Return the underlying stream (BufReader is dropped, stream continues)
293
    Ok(reader.into_inner())
294
}
295

            
296
/// An error that occurs while negotiating a connection with a proxy.
297
#[derive(Clone, Debug, thiserror::Error)]
298
#[non_exhaustive]
299
pub enum ProxyError {
300
    /// We had an IO error while trying to open a connection to the proxy.
301
    #[error("Problem while connecting to proxy")]
302
    ProxyConnect(#[source] Arc<std::io::Error>),
303

            
304
    /// We had an IO error while talking to the proxy.
305
    #[error("Problem while communicating with proxy")]
306
    ProxyIo(#[source] Arc<std::io::Error>),
307

            
308
    /// We tried to use an address which socks doesn't support.
309
    #[error("SOCKS proxy does not support target address")]
310
    InvalidSocksAddr(#[source] tor_socksproto::Error),
311

            
312
    /// We tried to use an address type which _we_ don't recognize.
313
    #[error("Got an address type we don't recognize")]
314
    UnrecognizedAddr,
315

            
316
    /// Our SOCKS implementation told us that this request cannot be encoded.
317
    #[error("Tried to make an invalid SOCKS request")]
318
    InvalidSocksRequest(#[source] tor_socksproto::Error),
319

            
320
    /// The peer refused our request, or spoke SOCKS incorrectly.
321
    #[error("Protocol error while communicating with SOCKS proxy")]
322
    SocksProto(#[source] tor_socksproto::Error),
323

            
324
    /// We encountered an internal programming error.
325
    #[error("Internal error")]
326
    Bug(#[from] tor_error::Bug),
327

            
328
    /// We got extra data immediately after our handshake, before we actually
329
    /// sent anything.
330
    ///
331
    /// This is not a bug in the calling code or in the peer protocol: it just
332
    /// means that the remote peer sent us data before we actually sent it any
333
    /// data. Unfortunately, there's a limitation in our code that makes it
334
    /// discard any such data, and therefore we have to give this error to
335
    /// prevent bugs.
336
    ///
337
    /// We could someday remove this limitation.
338
    #[error("Received unexpected early data from peer")]
339
    UnexpectedData,
340

            
341
    /// The proxy told us that our attempt failed.
342
    #[error("SOCKS proxy reported an error: {0}")]
343
    SocksError(SocksStatus),
344

            
345
    /// HTTP CONNECT proxy returned a non-2xx status code.
346
    #[error("HTTP CONNECT proxy returned status: {0}")]
347
    HttpConnectError(u16),
348

            
349
    /// HTTP CONNECT proxy returned a malformed response.
350
    #[error("HTTP CONNECT proxy returned invalid response")]
351
    HttpConnectMalformed,
352
}
353

            
354
impl From<std::io::Error> for ProxyError {
355
    fn from(e: std::io::Error) -> Self {
356
        ProxyError::ProxyIo(Arc::new(e))
357
    }
358
}
359

            
360
impl From<ProxyError> for std::io::Error {
361
    fn from(e: ProxyError) -> Self {
362
        std::io::Error::other(e)
363
    }
364
}
365

            
366
impl tor_error::HasKind for ProxyError {
367
    fn kind(&self) -> tor_error::ErrorKind {
368
        use ProxyError as E;
369
        use tor_error::ErrorKind as EK;
370
        match self {
371
            E::ProxyConnect(_) | E::ProxyIo(_) => EK::LocalNetworkError,
372
            E::InvalidSocksAddr(_) | E::InvalidSocksRequest(_) => EK::BadApiUsage,
373
            E::UnrecognizedAddr => EK::NotImplemented,
374
            E::SocksProto(_) => EK::LocalProtocolViolation,
375
            E::Bug(e) => e.kind(),
376
            E::UnexpectedData => EK::NotImplemented,
377
            E::SocksError(_) => EK::LocalProtocolViolation,
378
            E::HttpConnectError(_) | E::HttpConnectMalformed => EK::LocalProtocolViolation,
379
        }
380
    }
381
}
382

            
383
impl tor_error::HasRetryTime for ProxyError {
384
    fn retry_time(&self) -> tor_error::RetryTime {
385
        use ProxyError as E;
386
        use SocksStatus as S;
387
        use tor_error::RetryTime as RT;
388
        match self {
389
            E::ProxyConnect(_) | E::ProxyIo(_) => RT::AfterWaiting,
390
            E::InvalidSocksAddr(_) => RT::Never,
391
            E::UnrecognizedAddr => RT::Never,
392
            E::InvalidSocksRequest(_) => RT::Never,
393
            E::SocksProto(_) => RT::AfterWaiting,
394
            E::Bug(_) => RT::Never,
395
            E::UnexpectedData => RT::Never,
396
            E::SocksError(e) => match *e {
397
                S::CONNECTION_REFUSED
398
                | S::GENERAL_FAILURE
399
                | S::HOST_UNREACHABLE
400
                | S::NETWORK_UNREACHABLE
401
                | S::TTL_EXPIRED => RT::AfterWaiting,
402
                _ => RT::Never,
403
            },
404
            E::HttpConnectError(code) => {
405
                // 502/503/504 may be transient; auth and policy errors are not.
406
                if *code == 502 || *code == 503 || *code == 504 {
407
                    RT::AfterWaiting
408
                } else {
409
                    RT::Never
410
                }
411
            }
412
            E::HttpConnectMalformed => RT::Never,
413
        }
414
    }
415
}
416

            
417
#[cfg(feature = "pt-client")]
418
/// An object that connects to a Tor bridge via an external pluggable transport
419
/// that provides a proxy.
420
#[derive(Clone, Debug)]
421
pub struct ExternalProxyPlugin<R> {
422
    /// The runtime to use for connections.
423
    runtime: R,
424
    /// The location of the proxy.
425
    proxy_addr: SocketAddr,
426
    /// The SOCKS protocol version to use.
427
    proxy_version: SocksVersion,
428
}
429

            
430
#[cfg(feature = "pt-client")]
431
impl<R: NetStreamProvider + Send + Sync> ExternalProxyPlugin<R> {
432
    /// Make a new `ExternalProxyPlugin`.
433
    pub fn new(rt: R, proxy_addr: SocketAddr, proxy_version: SocksVersion) -> Self {
434
        Self {
435
            runtime: rt,
436
            proxy_addr,
437
            proxy_version,
438
        }
439
    }
440
}
441

            
442
#[cfg(feature = "pt-client")]
443
#[async_trait]
444
impl<R: NetStreamProvider + Send + Sync> TransportImplHelper for ExternalProxyPlugin<R> {
445
    type Stream = R::Stream;
446

            
447
    async fn connect(&self, target: &OwnedChanTarget) -> crate::Result<(PeerAddr, R::Stream)> {
448
        let pt_target = match target.chan_method() {
449
            ChannelMethod::Direct(_) => {
450
                return Err(crate::Error::UnusableTarget(bad_api_usage!(
451
                    "Used pluggable transport for a TCP connection."
452
                )));
453
            }
454
            ChannelMethod::Pluggable(target) => target,
455
            other => {
456
                return Err(crate::Error::UnusableTarget(bad_api_usage!(
457
                    "Used unknown, unsupported, transport {:?} for a TCP connection.",
458
                    other,
459
                )));
460
            }
461
        };
462

            
463
        let into_err = |e: ProxyError| crate::Error::Connect {
464
            addresses: vec![(sv(pt_target.to_string()), e.into())],
465
        };
466
        let protocol =
467
            settings_to_protocol(self.proxy_version, encode_settings(pt_target.settings()))
468
                .map_err(into_err)?;
469
        let stream =
470
            connect_via_proxy(&self.runtime, &self.proxy_addr, &protocol, pt_target.addr())
471
                .await
472
                .map_err(into_err)?;
473

            
474
        Ok((pt_target.into(), stream))
475
    }
476
}
477

            
478
/// Encode the PT settings from `IT` in a format that a pluggable transport can use.
479
#[cfg(feature = "pt-client")]
480
14
fn encode_settings<'a, IT>(settings: IT) -> String
481
14
where
482
14
    IT: Iterator<Item = (&'a str, &'a str)>,
483
{
484
    /// Escape a character in the way expected by pluggable transports.
485
    ///
486
    /// This escape machinery is a mirror of that in the standard library.
487
    enum EscChar {
488
        /// Return a backslash then a character.
489
        Backslash(char),
490
        /// Return a character.
491
        Literal(char),
492
        /// Return nothing.
493
        Done,
494
    }
495
    impl EscChar {
496
        /// Create an iterator to escape one character.
497
264
        fn new(ch: char, in_key: bool) -> Self {
498
6
            match ch {
499
8
                '\\' | ';' => EscChar::Backslash(ch),
500
2
                '=' if in_key => EscChar::Backslash(ch),
501
254
                _ => EscChar::Literal(ch),
502
            }
503
264
        }
504
    }
505
    impl Iterator for EscChar {
506
        type Item = char;
507

            
508
538
        fn next(&mut self) -> Option<Self::Item> {
509
538
            match *self {
510
10
                EscChar::Backslash(ch) => {
511
10
                    *self = EscChar::Literal(ch);
512
10
                    Some('\\')
513
                }
514
264
                EscChar::Literal(ch) => {
515
264
                    *self = EscChar::Done;
516
264
                    Some(ch)
517
                }
518
264
                EscChar::Done => None,
519
            }
520
538
        }
521
    }
522

            
523
    /// escape a key or value string.
524
40
    fn esc(s: &str, in_key: bool) -> impl Iterator<Item = char> + '_ {
525
284
        s.chars().flat_map(move |c| EscChar::new(c, in_key))
526
40
    }
527

            
528
14
    let mut result = String::new();
529
20
    for (k, v) in settings {
530
20
        result.extend(esc(k, true));
531
20
        result.push('=');
532
20
        result.extend(esc(v, false));
533
20
        result.push(';');
534
20
    }
535
14
    result.pop(); // remove the final ';' if any. Yes this is ugly.
536

            
537
14
    result
538
14
}
539

            
540
/// Transform a string into a representation that can be sent as SOCKS
541
/// authentication.
542
// NOTE(eta): I am very unsure of the logic in here.
543
#[cfg(feature = "pt-client")]
544
24
pub fn settings_to_protocol(vers: SocksVersion, s: String) -> Result<Protocol, ProxyError> {
545
24
    let mut bytes: Vec<_> = s.into();
546
24
    Ok(if bytes.is_empty() {
547
2
        Protocol::Socks(vers, SocksAuth::NoAuth)
548
22
    } else if vers == SocksVersion::V4 {
549
4
        if bytes.contains(&0) {
550
2
            return Err(ProxyError::InvalidSocksRequest(
551
2
                tor_socksproto::Error::NotImplemented(
552
2
                    "SOCKS 4 doesn't support internal NUL bytes (for PT settings list)".into(),
553
2
                ),
554
2
            ));
555
        } else {
556
2
            Protocol::Socks(SocksVersion::V4, SocksAuth::Socks4(bytes))
557
        }
558
18
    } else if bytes.len() <= 255 {
559
        // The [0] here is mandatory according to the pt-spec.
560
6
        Protocol::Socks(SocksVersion::V5, SocksAuth::Username(bytes, vec![0]))
561
12
    } else if bytes.len() <= (255 * 2) {
562
8
        let password = bytes.split_off(255);
563
8
        Protocol::Socks(SocksVersion::V5, SocksAuth::Username(bytes, password))
564
    } else {
565
4
        return Err(ProxyError::InvalidSocksRequest(
566
4
            tor_socksproto::Error::NotImplemented("PT settings list too long for SOCKS 5".into()),
567
4
        ));
568
    })
569
24
}
570

            
571
#[cfg(test)]
572
mod test {
573
    // @@ begin test lint list maintained by maint/add_warning @@
574
    #![allow(clippy::bool_assert_comparison)]
575
    #![allow(clippy::clone_on_copy)]
576
    #![allow(clippy::dbg_macro)]
577
    #![allow(clippy::mixed_attributes_style)]
578
    #![allow(clippy::print_stderr)]
579
    #![allow(clippy::print_stdout)]
580
    #![allow(clippy::single_char_pattern)]
581
    #![allow(clippy::unwrap_used)]
582
    #![allow(clippy::unchecked_time_subtraction)]
583
    #![allow(clippy::useless_vec)]
584
    #![allow(clippy::needless_pass_by_value)]
585
    #![allow(clippy::string_slice)] // See arti#2571
586
    //! <!-- @@ end test lint list maintained by maint/add_warning @@ -->
587
    #[allow(unused_imports)]
588
    use super::*;
589

            
590
    #[test]
591
    fn protocol_debug_redacts_http_connect_auth() {
592
        let proto = Protocol::HttpConnect {
593
            auth: Some((
594
                Sensitive::new("user_name".to_owned()),
595
                Sensitive::new("pass_word".to_owned()),
596
            )),
597
        };
598

            
599
        let formatted = format!("{proto:?}");
600
        assert!(formatted.contains("HttpConnect"));
601
        assert!(!formatted.contains("user_name"));
602
        assert!(!formatted.contains("pass_word"));
603
    }
604

            
605
    #[cfg(feature = "pt-client")]
606
    #[test]
607
    fn setting_encoding() {
608
        fn check(settings: Vec<(&str, &str)>, expected: &str) {
609
            assert_eq!(encode_settings(settings.into_iter()), expected);
610
        }
611

            
612
        // Easy cases, no escapes.
613
        check(vec![], "");
614
        check(vec![("hello", "world")], "hello=world");
615
        check(
616
            vec![("hey", "verden"), ("hello", "world")],
617
            "hey=verden;hello=world",
618
        );
619
        check(
620
            vec![("hey", "verden"), ("hello", "world"), ("selv", "tak")],
621
            "hey=verden;hello=world;selv=tak",
622
        );
623

            
624
        check(
625
            vec![("semi;colon", "equals=sign")],
626
            r"semi\;colon=equals=sign",
627
        );
628
        check(
629
            vec![("equals=sign", "semi;colon")],
630
            r"equals\=sign=semi\;colon",
631
        );
632
        check(
633
            vec![("semi;colon", "equals=sign"), ("also", "back\\slash")],
634
            r"semi\;colon=equals=sign;also=back\\slash",
635
        );
636
    }
637

            
638
    #[cfg(feature = "pt-client")]
639
    #[test]
640
    fn split_settings() {
641
        use SocksVersion::*;
642
        let long_string = "examplestrg".to_owned().repeat(50);
643
        assert_eq!(long_string.len(), 550);
644
        let sv = |v, a, b| settings_to_protocol(v, long_string[a..b].to_owned()).unwrap();
645
        let s = |a, b| sv(V5, a, b);
646
        let v = |a, b| long_string.as_bytes()[a..b].to_vec();
647

            
648
        assert_eq!(s(0, 0), Protocol::Socks(V5, SocksAuth::NoAuth));
649
        assert_eq!(
650
            s(0, 50),
651
            Protocol::Socks(V5, SocksAuth::Username(v(0, 50), vec![0]))
652
        );
653
        assert_eq!(
654
            s(0, 255),
655
            Protocol::Socks(V5, SocksAuth::Username(v(0, 255), vec![0]))
656
        );
657
        assert_eq!(
658
            s(0, 256),
659
            Protocol::Socks(V5, SocksAuth::Username(v(0, 255), v(255, 256)))
660
        );
661
        assert_eq!(
662
            s(0, 300),
663
            Protocol::Socks(V5, SocksAuth::Username(v(0, 255), v(255, 300)))
664
        );
665
        assert_eq!(
666
            s(0, 510),
667
            Protocol::Socks(V5, SocksAuth::Username(v(0, 255), v(255, 510)))
668
        );
669

            
670
        // This one needs to use socks4, or it won't fit. :P
671
        assert_eq!(
672
            sv(V4, 0, 511),
673
            Protocol::Socks(V4, SocksAuth::Socks4(v(0, 511)))
674
        );
675

            
676
        // Small requests with "0" bytes work fine...
677
        assert_eq!(
678
            settings_to_protocol(V5, "\0".to_owned()).unwrap(),
679
            Protocol::Socks(V5, SocksAuth::Username(vec![0], vec![0]))
680
        );
681
        assert_eq!(
682
            settings_to_protocol(V5, "\0".to_owned().repeat(510)).unwrap(),
683
            Protocol::Socks(V5, SocksAuth::Username(vec![0; 255], vec![0; 255]))
684
        );
685

            
686
        // Huge requests with "0" simply can't be encoded.
687
        assert!(settings_to_protocol(V5, "\0".to_owned().repeat(511)).is_err());
688

            
689
        // Huge requests without "0" can't be encoded as V5
690
        assert!(settings_to_protocol(V5, long_string[0..512].to_owned()).is_err());
691

            
692
        // Requests with "0" can't be encoded as V4.
693
        assert!(settings_to_protocol(V4, "\0".to_owned()).is_err());
694
    }
695

            
696
    #[test]
697
    fn parse_http_connect_200_ok() {
698
        let response = b"HTTP/1.1 200 OK\r\nContent-Length: 0\r\n\r\n";
699
        assert_eq!(parse_http_connect_response(response).unwrap(), 200);
700
    }
701

            
702
    #[test]
703
    fn parse_http_connect_407_auth_required() {
704
        let response = b"HTTP/1.1 407 Proxy Authentication Required\r\n\r\n";
705
        match parse_http_connect_response(response) {
706
            Err(ProxyError::HttpConnectError(407)) => (), // Expected
707
            other => panic!("Expected 407 error, got {:?}", other),
708
        }
709
    }
710

            
711
    #[test]
712
    fn parse_http_connect_malformed_no_status() {
713
        let response = b"INVALID HTTP";
714
        assert!(matches!(
715
            parse_http_connect_response(response),
716
            Err(ProxyError::HttpConnectMalformed)
717
        ));
718
    }
719

            
720
    #[test]
721
    fn parse_http_connect_with_headers() {
722
        let response = b"HTTP/1.1 200 Connection Established\r\nConnection: close\r\nProxy-Agent: Proxy/1.0\r\n\r\n";
723
        assert_eq!(parse_http_connect_response(response).unwrap(), 200);
724
    }
725

            
726
    #[test]
727
    fn parse_http_connect_rejects_pipelined_data() {
728
        let response = b"HTTP/1.1 200 OK\r\n\r\nEXTRA_DATA";
729
        assert!(matches!(
730
            parse_http_connect_response(response),
731
            Err(ProxyError::UnexpectedData)
732
        ));
733
    }
734

            
735
    #[test]
736
    fn parse_http_connect_oversized_headers() {
737
        let huge_header = vec![b'X'; MAX_HTTP_HEADER_BYTES + 1];
738
        assert!(matches!(
739
            parse_http_connect_response(&huge_header),
740
            Err(ProxyError::HttpConnectMalformed)
741
        ));
742
    }
743
}