Skip to content

Commit 55e3e17

Browse files
committed
api: Add dc_msg_get_filedata_path(), deprecate dc_msg_get_file() (#4309)
`dc_msg_get_file()` should create a temporary directory inside the blobdir, save a copy with unmangled filename, and return its path. In the housekeeping we regularly clear these directories. This API is kept for compatibility, but deprecated. Instead, add `dc_msg_get_filedata_path()` that avoids unnecessary copying if an app needs just to open a file and its name doesn't matter.
1 parent 41e3b51 commit 55e3e17

File tree

11 files changed

+116
-31
lines changed

11 files changed

+116
-31
lines changed

deltachat-ffi/deltachat.h

+19-1
Original file line numberDiff line numberDiff line change
@@ -4081,14 +4081,32 @@ char* dc_msg_get_subject (const dc_msg_t* msg);
40814081
*
40824082
* Typically files are associated with images, videos, audios, documents.
40834083
* Plain text messages do not have a file.
4084-
* File name may be mangled. To obtain the original attachment filename use dc_msg_get_filename().
4084+
* The filename isn't meaningful, only the extension is preserved. To obtain the original attachment
4085+
* filename use dc_msg_get_filename().
40854086
*
40864087
* @memberof dc_msg_t
40874088
* @param msg The message object.
40884089
* @return The full path (with file name and extension) of the file associated with the message.
40894090
* If there is no file associated with the message, an empty string is returned.
40904091
* NULL is never returned and the returned value must be released using dc_str_unref().
40914092
*/
4093+
char* dc_msg_get_filedata_path (const dc_msg_t* msg);
4094+
4095+
4096+
/**
4097+
* Get full path to the copy of the file, associated with a message, with the original filename.
4098+
* Deprecated, use dc_msg_get_filedata_path() and dc_msg_get_filename() instead.
4099+
*
4100+
* Typically files are associated with images, videos, audios, documents.
4101+
* Plain text messages do not have a file.
4102+
*
4103+
* @memberof dc_msg_t
4104+
* @param msg The message object.
4105+
* @return The full path (with file name and extension) of the file associated with the message.
4106+
* If there is no file associated with the message, an empty string is returned.
4107+
* In case of an error an empty string is returned.
4108+
* NULL is never returned and the returned value must be released using dc_str_unref().
4109+
*/
40924110
char* dc_msg_get_file (const dc_msg_t* msg);
40934111

40944112

deltachat-ffi/src/lib.rs

+22-3
Original file line numberDiff line numberDiff line change
@@ -3359,20 +3359,39 @@ pub unsafe extern "C" fn dc_msg_get_subject(msg: *mut dc_msg_t) -> *mut libc::c_
33593359
}
33603360

33613361
#[no_mangle]
3362-
pub unsafe extern "C" fn dc_msg_get_file(msg: *mut dc_msg_t) -> *mut libc::c_char {
3362+
pub unsafe extern "C" fn dc_msg_get_filedata_path(msg: *mut dc_msg_t) -> *mut libc::c_char {
33633363
if msg.is_null() {
3364-
eprintln!("ignoring careless call to dc_msg_get_file()");
3364+
eprintln!("ignoring careless call to dc_msg_get_filedata_path()");
33653365
return "".strdup();
33663366
}
33673367
let ffi_msg = &*msg;
33683368
let ctx = &*ffi_msg.context;
33693369
ffi_msg
33703370
.message
3371-
.get_file(ctx)
3371+
.get_filedata_path(ctx)
33723372
.map(|p| p.to_string_lossy().strdup())
33733373
.unwrap_or_else(|| "".strdup())
33743374
}
33753375

3376+
#[no_mangle]
3377+
pub unsafe extern "C" fn dc_msg_get_file(msg: *mut dc_msg_t) -> *mut libc::c_char {
3378+
if msg.is_null() {
3379+
eprintln!("ignoring careless call to dc_msg_get_file()");
3380+
return "".strdup();
3381+
}
3382+
let ffi_msg = &*msg;
3383+
let ctx = &*ffi_msg.context;
3384+
let r = block_on(ffi_msg.message.get_file(ctx));
3385+
let Ok(r) = r else {
3386+
r.context("Failed to get file from message")
3387+
.log_err(ctx)
3388+
.unwrap_or_default();
3389+
return "".strdup();
3390+
};
3391+
r.map(|p| p.to_string_lossy().strdup())
3392+
.unwrap_or_else(|| "".strdup())
3393+
}
3394+
33763395
#[no_mangle]
33773396
pub unsafe extern "C" fn dc_msg_save_file(
33783397
msg: *mut dc_msg_t,

deltachat-jsonrpc/src/api/types/message.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -148,7 +148,7 @@ impl MessageObject {
148148
|| quote.get_viewtype() == Viewtype::Gif
149149
|| quote.get_viewtype() == Viewtype::Sticker
150150
{
151-
match quote.get_file(context) {
151+
match quote.get_filedata_path(context) {
152152
Some(path_buf) => path_buf.to_str().map(|s| s.to_owned()),
153153
None => None,
154154
}
@@ -221,7 +221,7 @@ impl MessageObject {
221221

222222
setup_code_begin: message.get_setupcodebegin(context).await,
223223

224-
file: match message.get_file(context) {
224+
file: match message.get_filedata_path(context) {
225225
Some(path_buf) => path_buf.to_str().map(|s| s.to_owned()),
226226
None => None,
227227
}, //BLOBS
@@ -420,7 +420,7 @@ impl MessageNotificationInfo {
420420
Viewtype::Image | Viewtype::Gif | Viewtype::Sticker
421421
) {
422422
message
423-
.get_file(context)
423+
.get_filedata_path(context)
424424
.map(|path_buf| path_buf.to_str().map(|s| s.to_owned()))
425425
.unwrap_or_default()
426426
} else {

python/src/deltachat/message.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -109,7 +109,7 @@ def set_html(self, html_text):
109109
@props.with_doc
110110
def file_path(self):
111111
"""file path if there was an attachment, otherwise empty string."""
112-
return from_dc_charpointer(lib.dc_msg_get_file(self._dc_msg))
112+
return from_dc_charpointer(lib.dc_msg_get_filedata_path(self._dc_msg))
113113

114114
def set_file(self, path, mime_type=None):
115115
"""set file for this message from path and mime_type."""

src/imex.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -280,7 +280,7 @@ pub async fn continue_key_transfer(
280280
"Message is no Autocrypt Setup Message."
281281
);
282282

283-
if let Some(filename) = msg.get_file(context) {
283+
if let Some(filename) = msg.get_filedata_path(context) {
284284
let file = open_file_std(context, filename)?;
285285
let sc = normalize_setup_code(setup_code);
286286
let armored_key = decrypt_setup_file(&sc, file).await?;

src/imex/transfer.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -651,8 +651,9 @@ mod tests {
651651
};
652652
let msg = Message::load_from_db(&ctx1, *msgid).await.unwrap();
653653

654-
let path = msg.get_file(&ctx1).unwrap();
655-
assert_eq!(path.with_file_name("hello.txt"), path);
654+
assert_eq!(&msg.get_filename().unwrap(), "hello.txt");
655+
let path = msg.get_filedata_path(&ctx1).unwrap();
656+
assert_eq!(path.with_extension("txt"), path);
656657
let text = fs::read_to_string(&path).await.unwrap();
657658
assert_eq!(text, "i am attachment");
658659

src/message.rs

+53-16
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
//! # Messages and their identifiers.
22
3-
use std::collections::BTreeSet;
3+
use std::collections::{hash_map::DefaultHasher, BTreeSet};
4+
use std::hash::{Hash, Hasher};
45
use std::path::{Path, PathBuf};
56

67
use anyhow::{ensure, format_err, Context as _, Result};
@@ -288,9 +289,14 @@ WHERE id=?;
288289
ret += &format!("Error: {error}");
289290
}
290291

291-
if let Some(path) = msg.get_file(context) {
292+
if let Some(path) = msg.get_filedata_path(context) {
292293
let bytes = get_filebytes(context, &path).await?;
293-
ret += &format!("\nFile: {}, {} bytes\n", path.display(), bytes);
294+
ret += &format!(
295+
"\nFile: {}, name: {}, {} bytes\n",
296+
path.display(),
297+
msg.get_filename().unwrap_or_default(),
298+
bytes
299+
);
294300
}
295301

296302
if msg.viewtype != Viewtype::Text {
@@ -575,14 +581,49 @@ impl Message {
575581
None
576582
}
577583

578-
/// Returns the full path to the file associated with a message.
579-
pub fn get_file(&self, context: &Context) -> Option<PathBuf> {
584+
/// Returns the full path to the file associated with a message. The filename isn't meaningful,
585+
/// only the extension is preserved.
586+
pub fn get_filedata_path(&self, context: &Context) -> Option<PathBuf> {
580587
self.param.get_path(Param::File, context).unwrap_or(None)
581588
}
582589

590+
/// Returns the full path to the copy of the file, associated with a message, with the original
591+
/// filename.
592+
///
593+
/// Deprecated, use `get_filedata_path()` and `get_filename()` instead.
594+
pub async fn get_file(&self, context: &Context) -> Result<Option<PathBuf>> {
595+
let Some(data_path) = self.get_filedata_path(context) else {
596+
return Ok(None);
597+
};
598+
let Some(filename) = self.get_filename() else {
599+
return Ok(Some(data_path));
600+
};
601+
if data_path.with_file_name(&filename) == data_path {
602+
return Ok(Some(data_path));
603+
}
604+
let mut hasher = DefaultHasher::new();
605+
data_path
606+
.file_name()
607+
.context("no filename??")?
608+
.hash(&mut hasher);
609+
let dirname = format!("tmp.{:016x}", hasher.finish());
610+
let dir_path = context.get_blobdir().join(&dirname);
611+
fs::create_dir_all(&dir_path).await?;
612+
let file_path = dir_path.join(&filename);
613+
let mut src = fs::OpenOptions::new().read(true).open(data_path).await?;
614+
let mut dst = fs::OpenOptions::new()
615+
.write(true)
616+
.create(true)
617+
.truncate(true)
618+
.open(&file_path)
619+
.await?;
620+
io::copy(&mut src, &mut dst).await?;
621+
Ok(Some(file_path))
622+
}
623+
583624
/// Save file copy at the user-provided path.
584625
pub async fn save_file(&self, context: &Context, path: &Path) -> Result<()> {
585-
let path_src = self.get_file(context).context("No file")?;
626+
let path_src = self.get_filedata_path(context).context("No file")?;
586627
let mut src = fs::OpenOptions::new().read(true).open(path_src).await?;
587628
let mut dst = fs::OpenOptions::new()
588629
.write(true)
@@ -728,8 +769,6 @@ impl Message {
728769
}
729770

730771
/// Returns original filename (as shown in chat).
731-
///
732-
/// To get the full path, use [`Self::get_file()`].
733772
pub fn get_filename(&self) -> Option<String> {
734773
if let Some(name) = self.param.get(Param::Filename) {
735774
return Some(name.to_string());
@@ -904,7 +943,7 @@ impl Message {
904943
return None;
905944
}
906945

907-
if let Some(filename) = self.get_file(context) {
946+
if let Some(filename) = self.get_filedata_path(context) {
908947
if let Ok(ref buf) = read_file(context, filename).await {
909948
if let Ok((typ, headers, _)) = split_armored_data(buf) {
910949
if typ == pgp::armor::BlockType::Message {
@@ -1925,7 +1964,7 @@ pub enum Viewtype {
19251964

19261965
/// Animated GIF message.
19271966
/// File, width and height are set via dc_msg_set_file(), dc_msg_set_dimension()
1928-
/// and retrieved via dc_msg_get_file(), dc_msg_get_width(), dc_msg_get_height().
1967+
/// and retrieved via dc_msg_get_filedata_path(), dc_msg_get_width(), dc_msg_get_height().
19291968
Gif = 21,
19301969

19311970
/// Message containing a sticker, similar to image.
@@ -1935,26 +1974,24 @@ pub enum Viewtype {
19351974

19361975
/// Message containing an Audio file.
19371976
/// File and duration are set via dc_msg_set_file(), dc_msg_set_duration()
1938-
/// and retrieved via dc_msg_get_file(), dc_msg_get_duration().
1977+
/// and retrieved via dc_msg_get_filedata_path(), dc_msg_get_duration().
19391978
Audio = 40,
19401979

19411980
/// A voice message that was directly recorded by the user.
19421981
/// For all other audio messages, the type #DC_MSG_AUDIO should be used.
19431982
/// File and duration are set via dc_msg_set_file(), dc_msg_set_duration()
1944-
/// and retrieved via dc_msg_get_file(), dc_msg_get_duration()
1983+
/// and retrieved via dc_msg_get_filedata_path(), dc_msg_get_duration().
19451984
Voice = 41,
19461985

19471986
/// Video messages.
19481987
/// File, width, height and durarion
19491988
/// are set via dc_msg_set_file(), dc_msg_set_dimension(), dc_msg_set_duration()
1950-
/// and retrieved via
1951-
/// dc_msg_get_file(), dc_msg_get_width(),
1989+
/// and retrieved via dc_msg_get_filedata_path(), dc_msg_get_width(),
19521990
/// dc_msg_get_height(), dc_msg_get_duration().
19531991
Video = 50,
19541992

19551993
/// Message containing any file, eg. a PDF.
1956-
/// The file is set via dc_msg_set_file()
1957-
/// and retrieved via dc_msg_get_file().
1994+
/// The file is set via dc_msg_set_file() and retrieved via dc_msg_get_filedata_path().
19581995
File = 60,
19591996

19601997
/// Message is an invitation to a videochat.

src/mimeparser.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -3458,7 +3458,10 @@ On 2020-10-25, Bob wrote:
34583458
assert_eq!(msg.chat_blocked, Blocked::Request);
34593459
assert_eq!(msg.state, MessageState::InFresh);
34603460
assert_eq!(msg.get_filebytes(&t).await.unwrap().unwrap(), 2115);
3461-
assert!(msg.get_file(&t).is_some());
3461+
assert_eq!(
3462+
msg.get_filedata_path(&t).unwrap().extension().unwrap(),
3463+
"png"
3464+
);
34623465
assert_eq!(msg.get_filename().unwrap(), "avatar64x64.png");
34633466
assert_eq!(msg.get_width(), 64);
34643467
assert_eq!(msg.get_height(), 64);

src/receive_imf/tests.rs

+8-1
Original file line numberDiff line numberDiff line change
@@ -2964,7 +2964,7 @@ async fn test_long_and_duplicated_filenames() -> Result<()> {
29642964
assert_eq!(msg.get_viewtype(), Viewtype::File);
29652965
let resulting_filename = msg.get_filename().unwrap();
29662966
assert_eq!(resulting_filename, filename);
2967-
let path = msg.get_file(t).unwrap();
2967+
let path = msg.get_filedata_path(t).unwrap();
29682968
let path2 = path.with_file_name("saved.txt");
29692969
msg.save_file(t, &path2).await.unwrap();
29702970
assert!(
@@ -2974,6 +2974,13 @@ async fn test_long_and_duplicated_filenames() -> Result<()> {
29742974
assert_eq!(fs::read_to_string(&path).await.unwrap(), content);
29752975
assert_eq!(fs::read_to_string(&path2).await.unwrap(), content);
29762976
fs::remove_file(path2).await.unwrap();
2977+
2978+
let path = msg.get_file(t).await.unwrap().unwrap();
2979+
assert_eq!(path.with_file_name(resulting_filename), path);
2980+
assert_eq!(fs::read_to_string(&path).await.unwrap(), content);
2981+
let path2 = msg.get_file(t).await.unwrap().unwrap();
2982+
assert_eq!(path2, path);
2983+
assert_eq!(fs::read_to_string(&path).await.unwrap(), content);
29772984
}
29782985
check_message(&msg_alice, &alice, filename_sent, &content).await;
29792986
check_message(&msg_bob, &bob, filename_sent, &content).await;

src/summary.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,7 @@ impl Summary {
9696
|| msg.viewtype == Viewtype::Gif
9797
|| msg.viewtype == Viewtype::Sticker
9898
{
99-
msg.get_file(context)
99+
msg.get_filedata_path(context)
100100
.and_then(|path| path.to_str().map(|p| p.to_owned()))
101101
} else {
102102
None

src/webxdc.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -745,7 +745,7 @@ impl Message {
745745
context: &Context,
746746
) -> Result<async_zip::read::fs::ZipFileReader> {
747747
let path = self
748-
.get_file(context)
748+
.get_filedata_path(context)
749749
.ok_or_else(|| format_err!("No webxdc instance file."))?;
750750
let path_abs = get_abs_path(context, &path);
751751
let archive = async_zip::read::fs::ZipFileReader::new(path_abs).await?;

0 commit comments

Comments
 (0)