twilight_http/request/
multipart.rs#[derive(Clone, Debug)]
#[must_use = "has no effect if not built into a Form"]
pub struct Form {
boundary: [u8; 15],
buffer: Vec<u8>,
}
impl Form {
const APPLICATION_JSON: &'static [u8; 16] = b"application/json";
const BOUNDARY_TERMINATOR: &'static [u8; 2] = b"--";
const CONTENT_DISPOSITION_1: &'static [u8; 38] = b"Content-Disposition: form-data; name=\"";
const CONTENT_DISPOSITION_2: &'static [u8; 13] = b"\"; filename=\"";
const CONTENT_DISPOSITION_3: &'static [u8; 1] = b"\"";
const CONTENT_TYPE: &'static [u8; 14] = b"Content-Type: ";
const NEWLINE: &'static [u8; 2] = b"\r\n";
pub fn new() -> Self {
Self::default()
}
pub fn build(mut self) -> Vec<u8> {
self.buffer.extend(Self::BOUNDARY_TERMINATOR);
self.buffer
}
pub fn content_type(&self) -> Vec<u8> {
const NAME: &str = "multipart/form-data; boundary=";
let mut content_type = Vec::with_capacity(NAME.len() + self.boundary.len());
content_type.extend(NAME.as_bytes());
content_type.extend(self.boundary);
content_type
}
pub fn part(mut self, name: &[u8], value: &[u8]) -> Self {
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::CONTENT_DISPOSITION_1);
self.buffer.extend(name);
self.buffer.extend(Self::CONTENT_DISPOSITION_3);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(value);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::BOUNDARY_TERMINATOR);
self.buffer.extend(self.boundary);
self
}
pub fn file_part(mut self, name: &[u8], filename: &[u8], value: &[u8]) -> Self {
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::CONTENT_DISPOSITION_1);
self.buffer.extend(name);
self.buffer.extend(Self::CONTENT_DISPOSITION_2);
self.buffer.extend(filename);
self.buffer.extend(Self::CONTENT_DISPOSITION_3);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(value);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::BOUNDARY_TERMINATOR);
self.buffer.extend(self.boundary);
self
}
#[allow(clippy::len_without_is_empty)]
pub fn len(&self) -> usize {
self.buffer.len() + Self::BOUNDARY_TERMINATOR.len()
}
pub fn json_part(mut self, name: &[u8], value: &[u8]) -> Self {
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::CONTENT_DISPOSITION_1);
self.buffer.extend(name);
self.buffer.extend(Self::CONTENT_DISPOSITION_3);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::CONTENT_TYPE);
self.buffer.extend(Self::APPLICATION_JSON);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(value);
self.buffer.extend(Self::NEWLINE);
self.buffer.extend(Self::BOUNDARY_TERMINATOR);
self.buffer.extend(self.boundary);
self
}
}
impl Default for Form {
fn default() -> Self {
let mut form = Self {
boundary: random_boundary(),
buffer: Vec::new(),
};
form.buffer.extend(Self::BOUNDARY_TERMINATOR);
form.buffer.extend(form.boundary);
form
}
}
pub fn random_boundary() -> [u8; 15] {
let mut boundary = [0; 15];
for value in &mut boundary {
*value = fastrand::alphanumeric() as u8;
}
boundary
}
#[cfg(test)]
mod tests {
use super::*;
use std::str;
#[test]
fn form_builder() {
let form = Form::new()
.json_part(b"payload_json", b"json_value")
.file_part(b"files[0]", b"filename.jpg", b"file_value");
let boundary = str::from_utf8(&form.boundary).unwrap();
let expected = format!(
"--{boundary}\r\n\
Content-Disposition: form-data; name=\"payload_json\"\r\n\
Content-Type: application/json\r\n\
\r\n\
json_value\r\n\
--{boundary}\r\n\
Content-Disposition: form-data; name=\"files[0]\"; filename=\"filename.jpg\"\r\n\
\r\n\
file_value\r\n\
--{boundary}--",
);
let buffer_len = form.len();
let buffer = form.build();
assert_eq!(expected.as_bytes(), buffer);
assert_eq!(buffer_len, buffer.len());
}
}