#![allow(unused)]
use std::{fmt, fmt::Write};
use base64::prelude::*;
use bytes::Bytes;
use crate::header::{Entry, HeaderMap, HeaderValue, OccupiedEntry};
pub(crate) fn basic_auth<U, P>(username: U, password: Option<P>) -> HeaderValue
where
U: fmt::Display,
P: fmt::Display,
{
let encoded = {
let mut buf = b"Basic ".to_vec();
let mut buf_str = String::with_capacity(32);
let _ = write!(buf_str, "{username}:");
if let Some(password) = password {
let _ = write!(buf_str, "{password}");
}
let encoded = BASE64_STANDARD.encode(buf_str.as_bytes());
buf.extend(encoded.into_bytes());
buf
};
let mut header = HeaderValue::from_maybe_shared(Bytes::from(encoded))
.expect("base64 is always valid HeaderValue");
header.set_sensitive(true);
header
}
pub(crate) fn fast_random() -> u64 {
use std::{
cell::Cell,
collections::hash_map::RandomState,
hash::{BuildHasher, Hasher},
};
thread_local! {
static KEY: RandomState = RandomState::new();
static COUNTER: Cell<u64> = const { Cell::new(0) };
}
KEY.with(|key| {
COUNTER.with(|ctr| {
let n = ctr.get().wrapping_add(1);
ctr.set(n);
let mut h = key.build_hasher();
h.write_u64(n);
h.finish()
})
})
}
pub(crate) fn replace_headers(dst: &mut HeaderMap, src: HeaderMap) {
let mut prev_entry: Option<OccupiedEntry<_>> = 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"),
},
}
}
}
pub(crate) struct Escape<'a>(&'a [u8]);
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 {
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")?;
} else if (0x20..0x7f).contains(&c) {
write!(f, "{}", c as char)?;
} else {
write!(f, "\\x{c:02x}")?;
}
}
Ok(())
}
}