Added support for consolidated zip files

Signed-off-by: Janos SUTO <sj@acts.hu>
This commit is contained in:
Janos SUTO 2020-08-31 22:06:09 +02:00
parent f2e9dc8343
commit ae97f52adf

View File

@ -18,6 +18,7 @@
#include <openssl/evp.h> #include <openssl/evp.h>
#include <zlib.h> #include <zlib.h>
#include <assert.h> #include <assert.h>
#include <zip.h>
#include <piler.h> #include <piler.h>
@ -130,33 +131,88 @@ int inf(unsigned char *in, int len, int mode, char **buffer, FILE *dest){
} }
unsigned char *extract_file_from_zip(char *zipfile, char *filename, zip_uint64_t *size){
int i=0, errorp;
unsigned char *p=NULL;
struct zip *z;
struct zip_stat sb;
struct zip_file *zf;
z = zip_open(zipfile, ZIP_RDONLY, &errorp);
if(!z){
syslog(LOG_INFO, "%s: error: corrupt zip file=%s, error code=%d", zipfile, filename, errorp);
return NULL;
}
while(zip_stat_index(z, i, 0, &sb) == 0){
if(sb.size > 0 && sb.comp_size > 0 && strncmp(sb.name, filename, strlen(filename)) == 0){
*size = sb.comp_size;
zf = zip_fopen_index(z, i, 0);
if(zf){
p = malloc(sb.comp_size);
if(p){
if(zip_fread(zf, p, sb.comp_size) == -1){
syslog(LOG_PRIORITY, "zip_fread(): Error reading %s from %s", filename, zipfile);
free(p);
p = NULL;
}
}
zip_fclose(zf);
}
else syslog(LOG_PRIORITY, "cannot extract '%s' from '%s'", filename, zipfile);
}
i++;
}
zip_close(z);
return p;
}
int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *dest, struct config *cfg){ int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *dest, struct config *cfg){
int rc=0, n, olen, tlen, len, fd=-1; int rc=0, n, olen, tlen, len, fd=-1;
unsigned char *s=NULL, *addr=NULL, inbuf[REALLYBIGBUFSIZE]; char *relfilename;
unsigned char *s=NULL, *addr=NULL, *zipbuf=NULL, inbuf[REALLYBIGBUFSIZE];
struct stat st; struct stat st;
zip_uint64_t zipped_size=0;
#if OPENSSL_VERSION_NUMBER < 0x10100000L #if OPENSSL_VERSION_NUMBER < 0x10100000L
EVP_CIPHER_CTX ctx; EVP_CIPHER_CTX ctx;
#else #else
EVP_CIPHER_CTX *ctx=NULL; EVP_CIPHER_CTX *ctx=NULL;
#endif #endif
syslog(LOG_PRIORITY, "retrieveing %s", filename);
if(filename == NULL) return 1; if(filename == NULL) return 1;
relfilename = strchr(filename, ' ');
if(relfilename){
// We'll read the given file from a zip file
// filename is decomposed to <zipfile> and <relative filename>
*relfilename = '\0';
relfilename++;
zipbuf = extract_file_from_zip(filename, relfilename, &zipped_size);
len = zipped_size+EVP_MAX_BLOCK_LENGTH;
}
else {
// We have a distinct .m file or an attachment to read from
fd = open(filename, O_RDONLY); fd = open(filename, O_RDONLY);
if(fd == -1){ if(fd == -1){
syslog(LOG_PRIORITY, "%s: cannot open()", filename); syslog(LOG_PRIORITY, "%s: cannot open()", filename);
return 1; return 1;
} }
if(fstat(fd, &st)){ if(fstat(fd, &st)){
syslog(LOG_PRIORITY, "%s: cannot fstat()", filename); syslog(LOG_PRIORITY, "%s: cannot fstat()", filename);
close(fd); close(fd);
return 1; return 1;
} }
len = st.st_size+EVP_MAX_BLOCK_LENGTH;
}
if(cfg->encrypt_messages == 1){ if(cfg->encrypt_messages == 1){
#if OPENSSL_VERSION_NUMBER < 0x10100000L #if OPENSSL_VERSION_NUMBER < 0x10100000L
@ -170,8 +226,6 @@ int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *de
EVP_DecryptInit_ex(ctx, EVP_bf_cbc(), NULL, cfg->key, cfg->iv); EVP_DecryptInit_ex(ctx, EVP_bf_cbc(), NULL, cfg->key, cfg->iv);
#endif #endif
len = st.st_size+EVP_MAX_BLOCK_LENGTH;
s = malloc(len); s = malloc(len);
if(!s){ if(!s){
@ -181,6 +235,7 @@ int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *de
tlen = 0; tlen = 0;
if(!relfilename){
while((n = read(fd, inbuf, sizeof(inbuf)))){ while((n = read(fd, inbuf, sizeof(inbuf)))){
#if OPENSSL_VERSION_NUMBER < 0x10100000L #if OPENSSL_VERSION_NUMBER < 0x10100000L
@ -194,7 +249,13 @@ int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *de
tlen += olen; tlen += olen;
} }
} else {
if(!EVP_DecryptUpdate(ctx, s+tlen, &olen, zipbuf, zipped_size)){
printf("error in EVP_DecryptUpdate\n");
goto CLEANUP;
}
tlen += olen;
}
#if OPENSSL_VERSION_NUMBER < 0x10100000L #if OPENSSL_VERSION_NUMBER < 0x10100000L
if(EVP_DecryptFinal(&ctx, s + tlen, &olen) != 1){ if(EVP_DecryptFinal(&ctx, s + tlen, &olen) != 1){
@ -202,6 +263,7 @@ int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *de
if(EVP_DecryptFinal(ctx, s + tlen, &olen) != 1){ if(EVP_DecryptFinal(ctx, s + tlen, &olen) != 1){
#endif #endif
syslog(LOG_PRIORITY, "%s: EVP_DecryptFinal()", filename); syslog(LOG_PRIORITY, "%s: EVP_DecryptFinal()", filename);
printf("error in EVP_DecryptFinal\n");
goto CLEANUP; goto CLEANUP;
} }
@ -222,6 +284,7 @@ int retrieve_file_from_archive(char *filename, int mode, char **buffer, FILE *de
CLEANUP: CLEANUP:
if(fd != -1) close(fd); if(fd != -1) close(fd);
if(s) free(s); if(s) free(s);
if(zipbuf) free(zipbuf);
if(cfg->encrypt_messages == 1) if(cfg->encrypt_messages == 1)
#if OPENSSL_VERSION_NUMBER < 0x10100000L #if OPENSSL_VERSION_NUMBER < 0x10100000L
EVP_CIPHER_CTX_cleanup(&ctx); EVP_CIPHER_CTX_cleanup(&ctx);
@ -233,13 +296,57 @@ CLEANUP:
} }
void assemble_filename(char *filename, int len, char *s, struct config *cfg){
char zipfilename[SMALLBUFSIZE];
struct stat st;
/*
* If the zip file exists, then fix the filename to be '<zipfile> <relative filename>'
*/
snprintf(zipfilename, sizeof(zipfilename)-1, "%s/%02x/%c%c%c.zip", cfg->queuedir, cfg->server_id, s[8], s[9], s[10]);
if(stat(zipfilename, &st)){
snprintf(filename, len-1, "%s/%02x/%c%c%c/%c%c/%c%c/%s.m",
cfg->queuedir, cfg->server_id, s[8], s[9], s[10], s[RND_STR_LEN-4], s[RND_STR_LEN-3], s[RND_STR_LEN-2], s[RND_STR_LEN-1], s);
#ifdef HAVE_SUPPORT_FOR_COMPAT_STORAGE_LAYOUT
if(stat(filename, &st)){
snprintf(filename, len-1, "%s/%02x/%c%c/%c%c/%c%c/%s.m",
cfg->queuedir, cfg->server_id, s[RND_STR_LEN-6], s[RND_STR_LEN-5], s[RND_STR_LEN-4], s[RND_STR_LEN-3], s[RND_STR_LEN-2], s[RND_STR_LEN-1], s);
}
#endif
} else {
snprintf(filename, len-1, "%s %c%c%c/%c%c/%c%c/%s.m",
zipfilename, s[8], s[9], s[10], s[RND_STR_LEN-4], s[RND_STR_LEN-3], s[RND_STR_LEN-2], s[RND_STR_LEN-1], s);
}
}
void assemble_attachment_filename(char *filename, int len, char *s, int attachment_id, struct config *cfg){
char zipfilename[SMALLBUFSIZE];
struct stat st;
snprintf(zipfilename, sizeof(zipfilename)-1, "%s/%02x/%c%c%c.zip", cfg->queuedir, cfg->server_id, s[8], s[9], s[10]);
if(stat(zipfilename, &st)){
snprintf(filename, len-1, "%s/%02x/%c%c%c/%c%c/%c%c/%s.a%d", cfg->queuedir, cfg->server_id, s[8], s[9], s[10], s[RND_STR_LEN-4], s[RND_STR_LEN-3], s[RND_STR_LEN-2], s[RND_STR_LEN-1], s, attachment_id);
#ifdef HAVE_SUPPORT_FOR_COMPAT_STORAGE_LAYOUT
if(stat(filename, &st)){
snprintf(filename, len-1, "%s/%02x/%c%c/%c%c/%c%c/%s.a%d",
cfg->queuedir, cfg->server_id, s[RND_STR_LEN-6], s[RND_STR_LEN-5], s[RND_STR_LEN-4], s[RND_STR_LEN-3], s[RND_STR_LEN-2], s[RND_STR_LEN-1], s, attachment_id);
}
#endif
} else {
snprintf(filename, len-1, "%s %c%c%c/%c%c/%c%c/%s.a%d",
zipfilename, s[8], s[9], s[10], s[RND_STR_LEN-4], s[RND_STR_LEN-3], s[RND_STR_LEN-2], s[RND_STR_LEN-1], s, attachment_id);
}
}
int retrieve_email_from_archive(struct session_data *sdata, FILE *dest, struct config *cfg){ int retrieve_email_from_archive(struct session_data *sdata, FILE *dest, struct config *cfg){
int attachments; int attachments;
char *buffer=NULL, *saved_buffer, *p, filename[SMALLBUFSIZE]; char *buffer=NULL, *saved_buffer, *p, filename[SMALLBUFSIZE];
struct ptr_array ptr_arr[MAX_ATTACHMENTS]; struct ptr_array ptr_arr[MAX_ATTACHMENTS];
#ifdef HAVE_SUPPORT_FOR_COMPAT_STORAGE_LAYOUT
struct stat st;
#endif
if(strlen(sdata->ttmpfile) != RND_STR_LEN){ if(strlen(sdata->ttmpfile) != RND_STR_LEN){
printf("invalid piler-id: %s\n", sdata->ttmpfile); printf("invalid piler-id: %s\n", sdata->ttmpfile);
@ -253,12 +360,8 @@ int retrieve_email_from_archive(struct session_data *sdata, FILE *dest, struct c
return 1; return 1;
} }
snprintf(filename, sizeof(filename)-1, "%s/%02x/%c%c%c/%c%c/%c%c/%s.m", cfg->queuedir, cfg->server_id, *(sdata->ttmpfile+8), *(sdata->ttmpfile+9), *(sdata->ttmpfile+10), *(sdata->ttmpfile+RND_STR_LEN-4), *(sdata->ttmpfile+RND_STR_LEN-3), *(sdata->ttmpfile+RND_STR_LEN-2), *(sdata->ttmpfile+RND_STR_LEN-1), sdata->ttmpfile);
#ifdef HAVE_SUPPORT_FOR_COMPAT_STORAGE_LAYOUT assemble_filename(filename, sizeof(filename), sdata->ttmpfile, cfg);
if(stat(filename, &st)){
snprintf(filename, sizeof(filename)-1, "%s/%02x/%c%c/%c%c/%c%c/%s.m", cfg->queuedir, cfg->server_id, *(sdata->ttmpfile+RND_STR_LEN-6), *(sdata->ttmpfile+RND_STR_LEN-5), *(sdata->ttmpfile+RND_STR_LEN-4), *(sdata->ttmpfile+RND_STR_LEN-3), *(sdata->ttmpfile+RND_STR_LEN-2), *(sdata->ttmpfile+RND_STR_LEN-1), sdata->ttmpfile);
}
#endif
if(attachments == 0){ if(attachments == 0){
retrieve_file_from_archive(filename, WRITE_TO_STDOUT, &buffer, dest, cfg); retrieve_file_from_archive(filename, WRITE_TO_STDOUT, &buffer, dest, cfg);
@ -280,14 +383,7 @@ int retrieve_email_from_archive(struct session_data *sdata, FILE *dest, struct c
buffer = p + strlen(pointer); buffer = p + strlen(pointer);
if(strlen(ptr_arr[i].piler_id) == RND_STR_LEN){ if(strlen(ptr_arr[i].piler_id) == RND_STR_LEN){
snprintf(filename, sizeof(filename)-1, "%s/%02x/%c%c%c/%c%c/%c%c/%s.a%d", cfg->queuedir, cfg->server_id, ptr_arr[i].piler_id[8], ptr_arr[i].piler_id[9], ptr_arr[i].piler_id[10], ptr_arr[i].piler_id[RND_STR_LEN-4], ptr_arr[i].piler_id[RND_STR_LEN-3], ptr_arr[i].piler_id[RND_STR_LEN-2], ptr_arr[i].piler_id[RND_STR_LEN-1], ptr_arr[i].piler_id, ptr_arr[i].attachment_id); assemble_attachment_filename(filename, sizeof(filename), ptr_arr[i].piler_id, ptr_arr[i].attachment_id, cfg);
#ifdef HAVE_SUPPORT_FOR_COMPAT_STORAGE_LAYOUT
if(stat(filename, &st)){
snprintf(filename, sizeof(filename)-1, "%s/%02x/%c%c/%c%c/%c%c/%s.a%d", cfg->queuedir, cfg->server_id, ptr_arr[i].piler_id[RND_STR_LEN-6], ptr_arr[i].piler_id[RND_STR_LEN-5], ptr_arr[i].piler_id[RND_STR_LEN-4], ptr_arr[i].piler_id[RND_STR_LEN-3], ptr_arr[i].piler_id[RND_STR_LEN-2], ptr_arr[i].piler_id[RND_STR_LEN-1], ptr_arr[i].piler_id, ptr_arr[i].attachment_id);
}
#endif
retrieve_file_from_archive(filename, WRITE_TO_STDOUT, NULL, dest, cfg); retrieve_file_from_archive(filename, WRITE_TO_STDOUT, NULL, dest, cfg);
} }
} }