diff options
author | Vsevolod Stakhov <vsevolod@rspamd.com> | 2024-02-28 10:00:12 +0000 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@rspamd.com> | 2024-02-28 10:00:12 +0000 |
commit | 8e8a09ffbfa9ac60fb59d045f0fa4c4234119d6f (patch) | |
tree | 2b2ae0025e53d0497ae4d3f884ff2bba57eefaa8 /src/libmime/archives.c | |
parent | 642ca8c3a410d0fcd53d756534c22ea4c3b9013e (diff) | |
download | rspamd-8e8a09ffbfa9ac60fb59d045f0fa4c4234119d6f.tar.gz rspamd-8e8a09ffbfa9ac60fb59d045f0fa4c4234119d6f.zip |
[Project] Use libarchive for troublesome archives
Diffstat (limited to 'src/libmime/archives.c')
-rw-r--r-- | src/libmime/archives.c | 49 |
1 files changed, 40 insertions, 9 deletions
diff --git a/src/libmime/archives.c b/src/libmime/archives.c index ea0ea551d..c4e9b9e90 100644 --- a/src/libmime/archives.c +++ b/src/libmime/archives.c @@ -1,11 +1,11 @@ -/*- - * Copyright 2016 Vsevolod Stakhov +/* + * Copyright 2024 Vsevolod Stakhov * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * - * http://www.apache.org/licenses/LICENSE-2.0 + * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, @@ -24,6 +24,9 @@ #include <unicode/utf16.h> #include <unicode/ucnv.h> +#include <archive.h> +#include <archive_entry.h> + #define msg_debug_archive(...) rspamd_conditional_debug_fast(NULL, NULL, \ rspamd_archive_log_id, "archive", task->task_pool->tag.uid, \ G_STRFUNC, \ @@ -1643,7 +1646,8 @@ end: static const guchar * rspamd_7zip_read_next_section(struct rspamd_task *task, const guchar *p, const guchar *end, - struct rspamd_archive *arch) + struct rspamd_archive *arch, + struct rspamd_mime_part *part) { guchar t = *p; @@ -1660,10 +1664,37 @@ rspamd_7zip_read_next_section(struct rspamd_task *task, * In fact, headers are just packed, but we assume it as * encrypted to distinguish from the normal archives */ - msg_debug_archive("7zip: encoded header, needs to be uncompressed"); - arch->flags |= RSPAMD_ARCHIVE_CANNOT_READ; - p = NULL; /* Cannot get anything useful */ - break; + { + msg_debug_archive("7zip: encoded header, needs to be uncompressed"); + struct archive *a = archive_read_new(); + archive_read_support_format_7zip(a); + int r = archive_read_open_memory(a, part->parsed_data.begin, part->parsed_data.len); + if (r != ARCHIVE_OK) { + msg_debug_archive("7zip: cannot open memory archive: %s", archive_error_string(a)); + archive_read_free(a); + return NULL; + } + + /* Clean the existing files if any */ + rspamd_archive_dtor(arch); + arch->files = g_ptr_array_new(); + + struct archive_entry *ae; + + while (archive_read_next_header(a, &ae) == ARCHIVE_OK) { + const char *name = archive_entry_pathname(ae); + if (name) { + msg_debug_archive("7zip: found file %s", name); + struct rspamd_archive_file *f = g_malloc0(sizeof(*f)); + f->fname = g_string_new(name); + g_ptr_array_add(arch->files, f); + } + archive_read_data_skip(a); + } + archive_read_free(a); + p = NULL; /* Stop internal processor, as we rely on libarchive here */ + break; + } case kArchiveProperties: p = rspamd_7zip_read_archive_props(task, p, end, arch); break; @@ -1739,7 +1770,7 @@ rspamd_archive_process_7zip(struct rspamd_task *task, return; } - while ((p = rspamd_7zip_read_next_section(task, p, end, arch)) != NULL) + while ((p = rspamd_7zip_read_next_section(task, p, end, arch, part)) != NULL) ; part->part_type = RSPAMD_MIME_PART_ARCHIVE; |