use crate::header::{Entry, HeaderMap, HeaderValue, OccupiedEntry}; use std::fmt; pub fn basic_auth(username: U, password: Option

) -> HeaderValue where U: std::fmt::Display, P: std::fmt::Display, { use base64::prelude::BASE64_STANDARD; use base64::write::EncoderWriter; use std::io::Write; let mut buf = b"Basic ".to_vec(); { let mut encoder = EncoderWriter::new(&mut buf, &BASE64_STANDARD); let _ = write!(encoder, "{username}:"); if let Some(password) = password { let _ = write!(encoder, "{password}"); } } let mut header = HeaderValue::from_bytes(&buf).expect("base64 is always valid HeaderValue"); header.set_sensitive(true); header } // xor-shift #[cfg(not(target_arch = "wasm32"))] pub(crate) fn fast_random() -> u64 { use std::cell::Cell; use std::collections::hash_map::RandomState; use std::hash::{BuildHasher, Hasher}; use std::num::Wrapping; thread_local! { static RNG: Cell> = Cell::new(Wrapping(seed())); } fn seed() -> u64 { let seed = RandomState::new(); let mut out = 0; let mut cnt = 0; while out == 0 { cnt += 1; let mut hasher = seed.build_hasher(); hasher.write_usize(cnt); out = hasher.finish(); } out } RNG.with(|rng| { let mut n = rng.get(); debug_assert_ne!(n.0, 0); n ^= n >> 12; n ^= n << 25; n ^= n >> 27; rng.set(n); n.0.wrapping_mul(0x2545_f491_4f6c_dd1d) }) } pub(crate) fn replace_headers(dst: &mut HeaderMap, src: HeaderMap) { // IntoIter of HeaderMap yields (Option, HeaderValue). // The first time a name is yielded, it will be Some(name), and if // there are more values with the same name, the next yield will be // None. let mut prev_entry: Option> = None; for (key, value) in src { match key { Some(key) => match dst.entry(key) { Entry::Occupied(mut e) => { e.insert(value); prev_entry = Some(e); } Entry::Vacant(e) => { let e = e.insert_entry(value); prev_entry = Some(e); } }, None => match prev_entry { Some(ref mut entry) => { entry.append(value); } None => unreachable!("HeaderMap::into_iter yielded None first"), }, } } } #[cfg(feature = "cookies")] #[cfg(not(target_arch = "wasm32"))] pub(crate) fn add_cookie_header( headers: &mut HeaderMap, cookie_store: &dyn crate::cookie::CookieStore, url: &url::Url, ) { if let Some(header) = cookie_store.cookies(url) { headers.insert(crate::header::COOKIE, header); } } pub(crate) struct Escape<'a>(&'a [u8]); #[cfg(not(target_arch = "wasm32"))] impl<'a> Escape<'a> { pub(crate) fn new(bytes: &'a [u8]) -> Self { Escape(bytes) } } impl fmt::Debug for Escape<'_> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { write!(f, "b\"{}\"", self)?; Ok(()) } } impl fmt::Display for Escape<'_> { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { for &c in self.0 { // https://doc.rust-lang.org/reference.html#byte-escapes if c == b'\n' { write!(f, "\\n")?; } else if c == b'\r' { write!(f, "\\r")?; } else if c == b'\t' { write!(f, "\\t")?; } else if c == b'\\' || c == b'"' { write!(f, "\\{}", c as char)?; } else if c == b'\0' { write!(f, "\\0")?; // ASCII printable } else if c >= 0x20 && c < 0x7f { write!(f, "{}", c as char)?; } else { write!(f, "\\x{c:02x}")?; } } Ok(()) } }