Skip to content

Commit 2ed79cb

Browse files
committedNov 26, 2023
add mime support
1 parent de57280 commit 2ed79cb

File tree

5 files changed

+158
-0
lines changed

5 files changed

+158
-0
lines changed
 

‎curl-sys/lib.rs

+18
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,9 @@ pub enum curl_httppost {
6565
// pub userp: *mut c_void,
6666
}
6767

68+
pub enum curl_mime {}
69+
pub enum curl_mimepart {}
70+
6871
// pub const HTTPPOST_FILENAME: c_long = 1 << 0;
6972
// pub const HTTPPOST_READFILE: c_long = 1 << 1;
7073
// pub const HTTPPOST_PTRNAME: c_long = 1 << 2;
@@ -607,6 +610,8 @@ pub const CURLOPT_PROXY_SSL_OPTIONS: CURLoption = CURLOPTTYPE_LONG + 261;
607610

608611
pub const CURLOPT_ABSTRACT_UNIX_SOCKET: CURLoption = CURLOPTTYPE_OBJECTPOINT + 264;
609612

613+
pub const CURLOPT_MIMEPOST: CURLoption = CURLOPTTYPE_OBJECTPOINT + 269;
614+
610615
pub const CURLOPT_DOH_URL: CURLoption = CURLOPTTYPE_OBJECTPOINT + 279;
611616
pub const CURLOPT_UPLOAD_BUFFERSIZE: CURLoption = CURLOPTTYPE_LONG + 280;
612617

@@ -1161,6 +1166,19 @@ extern "C" {
11611166
sockfd: curl_socket_t,
11621167
sockp: *mut c_void,
11631168
) -> CURLMcode;
1169+
1170+
pub fn curl_mime_init(easy_handle: *mut CURL) -> *mut curl_mime;
1171+
pub fn curl_mime_free(mime_handle: *mut curl_mime);
1172+
pub fn curl_mime_addpart(mime_handle: *mut curl_mime) -> *mut curl_mimepart;
1173+
pub fn curl_mime_data(
1174+
mimepart: *mut curl_mimepart,
1175+
data: *const c_char,
1176+
datasize: size_t,
1177+
) -> CURLcode;
1178+
pub fn curl_mime_name(part: *mut curl_mimepart, name: *const c_char) -> CURLcode;
1179+
pub fn curl_mime_filename(part: *mut curl_mimepart, filename: *const c_char) -> CURLcode;
1180+
pub fn curl_mime_type(part: *mut curl_mimepart, mimetype: *const c_char) -> CURLcode;
1181+
pub fn curl_mime_subparts(part: *mut curl_mimepart, subparts: *mut curl_mime) -> CURLcode;
11641182
}
11651183

11661184
pub fn rust_crate_version() -> &'static str {

‎src/easy/handle.rs

+6
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ use libc::c_void;
1212
use crate::easy::handler::{self, InfoType, ReadError, SeekResult, WriteError};
1313
use crate::easy::handler::{Auth, NetRc, PostRedirections, ProxyType, SslOpt};
1414
use crate::easy::handler::{HttpVersion, IpResolve, SslVersion, TimeCondition};
15+
use crate::easy::mime::Mime;
1516
use crate::easy::{Easy2, Handler};
1617
use crate::easy::{Form, List};
1718
use crate::Error;
@@ -1470,6 +1471,11 @@ impl Easy {
14701471
pub fn take_error_buf(&self) -> Option<String> {
14711472
self.inner.take_error_buf()
14721473
}
1474+
1475+
/// Same as [`Easy2::add_mime`](struct.Easy2.html#method.add_mime)
1476+
pub fn add_mime(&mut self) -> Mime<EasyData> {
1477+
self.inner.add_mime()
1478+
}
14731479
}
14741480

14751481
impl EasyData {

‎src/easy/handler.rs

+21
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ use socket2::Socket;
1515

1616
use crate::easy::form;
1717
use crate::easy::list;
18+
use crate::easy::mime::Mime;
1819
use crate::easy::windows;
1920
use crate::easy::{Form, List};
2021
use crate::panic;
@@ -378,6 +379,8 @@ pub fn ssl_ctx(cx: *mut c_void) -> Result<(), Error> {
378379
/// ```
379380
pub struct Easy2<H> {
380381
inner: Box<Inner<H>>,
382+
/// Mime handles to free upon drop
383+
mimes: Vec<*mut curl_sys::curl_mime>,
381384
}
382385

383386
struct Inner<H> {
@@ -599,6 +602,7 @@ impl<H: Handler> Easy2<H> {
599602
error_buf: RefCell::new(vec![0; curl_sys::CURL_ERROR_SIZE]),
600603
handler,
601604
}),
605+
mimes: vec![],
602606
};
603607
ret.default_configure();
604608
ret
@@ -3509,6 +3513,20 @@ impl<H> Easy2<H> {
35093513
}
35103514
Err(err)
35113515
}
3516+
3517+
/// Create a mime handle attached to this [Easy2] instance.
3518+
pub fn add_mime(&mut self) -> Mime<H> {
3519+
Mime::new(self)
3520+
}
3521+
3522+
pub(crate) fn mimepost(&mut self, mime_handle: *mut curl_sys::curl_mime) -> Result<(), Error> {
3523+
self.mimes.push(mime_handle);
3524+
3525+
let rc = unsafe {
3526+
curl_sys::curl_easy_setopt(self.raw(), curl_sys::CURLOPT_MIMEPOST, mime_handle)
3527+
};
3528+
self.cvt(rc)
3529+
}
35123530
}
35133531

35143532
impl<H: fmt::Debug> fmt::Debug for Easy2<H> {
@@ -3524,6 +3542,9 @@ impl<H> Drop for Easy2<H> {
35243542
fn drop(&mut self) {
35253543
unsafe {
35263544
curl_sys::curl_easy_cleanup(self.inner.handle);
3545+
for &mime_handle in self.mimes.iter() {
3546+
curl_sys::curl_mime_free(mime_handle);
3547+
}
35273548
}
35283549
}
35293550
}

‎src/easy/mime.rs

+112
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
use crate::easy::Easy2;
2+
use crate::error::Error;
3+
use curl_sys::{
4+
curl_mime_addpart, curl_mime_data, curl_mime_filename, curl_mime_free, curl_mime_init,
5+
curl_mime_name, curl_mime_type, curl_mimepart, CURLcode, CURLE_OK,
6+
};
7+
use std::ffi::CString;
8+
use std::marker::PhantomData;
9+
use std::ptr::null_mut;
10+
11+
#[derive(Debug)]
12+
pub struct Mime<'e, E> {
13+
handle: *mut curl_sys::curl_mime,
14+
easy: &'e mut Easy2<E>,
15+
}
16+
17+
impl<'a, T> Mime<'a, T> {
18+
/// Create a mime handle
19+
pub(crate) fn new(easy: &'a mut Easy2<T>) -> Self {
20+
let handle = unsafe { curl_mime_init(easy.raw()) };
21+
assert!(!handle.is_null());
22+
23+
Self { handle, easy }
24+
}
25+
26+
/// Finalize creation of a mime post.
27+
pub fn post(mut self) -> Result<(), Error> {
28+
// once giving the mime handle to `Easy2` it is now their responsibility to free the handle.
29+
// so we need to make sure `Drop` below won't try to free it.
30+
let mime_handle = self.handle;
31+
self.handle = null_mut();
32+
self.easy.mimepost(mime_handle)
33+
}
34+
35+
/// Append a new empty part to a mime structure
36+
pub fn add_part(&mut self) -> MimePart<'a> {
37+
MimePart::new(self)
38+
}
39+
}
40+
41+
impl<E> Drop for Mime<'_, E> {
42+
fn drop(&mut self) {
43+
// we only need to free mime handles which hadn't been given to the ownership of `Easy2`.
44+
if !self.handle.is_null() {
45+
unsafe { curl_mime_free(self.handle) }
46+
}
47+
}
48+
}
49+
50+
#[derive(Debug)]
51+
pub struct MimePart<'a> {
52+
handle: *mut curl_mimepart,
53+
// attach to the lifetime of our [Mime] handle, but without taking ownership
54+
_lifetime: PhantomData<&'a ()>,
55+
}
56+
57+
impl<'a> MimePart<'a> {
58+
fn new<E>(mime: &mut Mime<E>) -> Self {
59+
let handle = unsafe { curl_mime_addpart(mime.handle) };
60+
assert!(!handle.is_null());
61+
62+
Self {
63+
handle,
64+
_lifetime: Default::default(),
65+
}
66+
}
67+
68+
/// Set a mime part's body data
69+
pub fn set_data(self, data: impl AsRef<[u8]>) -> Result<Self, Error> {
70+
let data = data.as_ref();
71+
let code = unsafe { curl_mime_data(self.handle, data.as_ptr() as *const _, data.len()) };
72+
code_ok(code).map(|_| self)
73+
}
74+
75+
/// Set a mime part's name
76+
///
77+
/// # Panics
78+
/// If `name` contains nul bytes, panic will occur.
79+
pub fn set_name(self, name: &str) -> Result<Self, Error> {
80+
let data = CString::new(name).unwrap();
81+
let code = unsafe { curl_mime_name(self.handle, data.as_ptr()) };
82+
code_ok(code).map(|_| self)
83+
}
84+
85+
/// Set a mime part's remote file name
86+
///
87+
/// # Panics
88+
/// If `filename` contains nul bytes, panic will occur.
89+
pub fn set_filename(self, filename: &str) -> Result<Self, Error> {
90+
let data = CString::new(filename).unwrap();
91+
let code = unsafe { curl_mime_filename(self.handle, data.as_ptr()) };
92+
code_ok(code).map(|_| self)
93+
}
94+
95+
/// Set a mime part's content type
96+
///
97+
/// # Panics
98+
/// If `content_type` contains nul bytes, panic will occur.
99+
pub fn set_content_type(self, content_type: &str) -> Result<Self, Error> {
100+
let data = CString::new(content_type).unwrap();
101+
let code = unsafe { curl_mime_type(self.handle, data.as_ptr()) };
102+
code_ok(code).map(|_| self)
103+
}
104+
}
105+
106+
fn code_ok(code: CURLcode) -> Result<(), Error> {
107+
if code == CURLE_OK {
108+
Ok(())
109+
} else {
110+
Err(Error::new(code))
111+
}
112+
}

‎src/easy/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ mod form;
1111
mod handle;
1212
mod handler;
1313
mod list;
14+
mod mime;
1415
mod windows;
1516

1617
pub use self::form::{Form, Part};

0 commit comments

Comments
 (0)
Please sign in to comment.