use super::document::CPDF_DOCS;
use super::{clear_error, set_error};
use crate::cpdf::attachments;
use crate::ffi::Handle;
use std::ffi::{CStr, c_char, c_int};
#[unsafe(no_mangle)]
pub extern "C" fn cpdf_listAttachmentsJSON(doc: Handle, len_out: *mut usize) -> *mut u8 {
clear_error();
let d = match CPDF_DOCS.get(doc) {
Some(arc) => arc,
None => {
set_error(1, &format!("invalid handle: {doc}"));
return std::ptr::null_mut();
}
};
let guard = d.lock().unwrap();
match attachments::list_attachments_json(&guard) {
Ok(json) => {
let bytes = json.into_bytes();
let len = bytes.len();
let ptr = unsafe {
let layout = std::alloc::Layout::from_size_align_unchecked(len, 1);
let p = std::alloc::alloc(layout);
if p.is_null() {
set_error(1, "alloc failed");
return std::ptr::null_mut();
}
std::ptr::copy_nonoverlapping(bytes.as_ptr(), p, len);
p
};
if !len_out.is_null() {
unsafe {
*len_out = len;
}
}
ptr
}
Err(e) => {
set_error(1, &e.to_string());
std::ptr::null_mut()
}
}
}
#[unsafe(no_mangle)]
pub extern "C" fn cpdf_attachFile(doc: Handle, path: *const c_char) -> Handle {
clear_error();
let d = match CPDF_DOCS.get(doc) {
Some(arc) => arc.lock().unwrap().clone(),
None => {
set_error(1, &format!("invalid handle: {doc}"));
return 0;
}
};
let p = if path.is_null() {
""
} else {
unsafe { CStr::from_ptr(path) }.to_str().unwrap_or("")
};
match attachments::attach_file(d, p, None) {
Ok(doc) => CPDF_DOCS.insert(doc),
Err(e) => {
set_error(1, &e.to_string());
0
}
}
}
#[unsafe(no_mangle)]
pub extern "C" fn cpdf_removeAttachments(doc: Handle) -> Handle {
clear_error();
let d = match CPDF_DOCS.get(doc) {
Some(arc) => arc.lock().unwrap().clone(),
None => {
set_error(1, &format!("invalid handle: {doc}"));
return 0;
}
};
match attachments::remove_attachments(d) {
Ok(doc) => CPDF_DOCS.insert(doc),
Err(e) => {
set_error(1, &e.to_string());
0
}
}
}
#[unsafe(no_mangle)]
pub extern "C" fn cpdf_dumpAttachments(doc: Handle, output_dir: *const c_char) -> c_int {
clear_error();
let d = match CPDF_DOCS.get(doc) {
Some(arc) => arc,
None => {
set_error(1, &format!("invalid handle: {doc}"));
return 1;
}
};
let guard = d.lock().unwrap();
let dir = if output_dir.is_null() {
"."
} else {
unsafe { CStr::from_ptr(output_dir) }
.to_str()
.unwrap_or(".")
};
match attachments::extract_attachments(&guard, dir) {
Ok(()) => 0,
Err(e) => {
set_error(1, &e.to_string());
1
}
}
}