1
0
mirror of https://github.com/profanity-im/profanity.git synced 2025-01-03 14:57:42 -05:00

Add I/O error handling and use filenames instead of file descriptors

This commit is contained in:
William Wennerström 2020-07-21 09:31:47 +02:00
parent 73f313b921
commit 62cbad1c6e
No known key found for this signature in database
GPG Key ID: E1382990BEDD319B
10 changed files with 102 additions and 71 deletions

1
.gitignore vendored
View File

@ -86,3 +86,4 @@ breaks
*.tar.* *.tar.*
*.zip *.zip
*.log*

View File

@ -9192,19 +9192,13 @@ _url_open_fallback_method(ProfWin* window, const char* url)
void void
_url_save_fallback_method(ProfWin* window, const char* url, const char* filename) _url_save_fallback_method(ProfWin* window, const char* url, const char* filename)
{ {
FILE* fh = fopen(filename, "wb");
if (!fh) {
cons_show_error("Cannot open file '%s' for writing.", filename);
return;
}
gchar* scheme = g_uri_parse_scheme(url); gchar* scheme = g_uri_parse_scheme(url);
if (g_strcmp0(scheme, "aesgcm") == 0) { if (g_strcmp0(scheme, "aesgcm") == 0) {
AESGCMDownload* download = malloc(sizeof(AESGCMDownload)); AESGCMDownload* download = malloc(sizeof(AESGCMDownload));
download->window = window; download->window = window;
download->url = strdup(url); download->url = strdup(url);
download->filehandle = fh; download->filename = strdup(filename);
pthread_create(&(download->worker), NULL, &aesgcm_file_get, download); pthread_create(&(download->worker), NULL, &aesgcm_file_get, download);
aesgcm_download_add_download(download); aesgcm_download_add_download(download);
@ -9212,7 +9206,7 @@ _url_save_fallback_method(ProfWin* window, const char* url, const char* filename
HTTPDownload* download = malloc(sizeof(HTTPDownload)); HTTPDownload* download = malloc(sizeof(HTTPDownload));
download->window = window; download->window = window;
download->url = strdup(url); download->url = strdup(url);
download->filehandle = fh; download->filename = strdup(filename);
pthread_create(&(download->worker), NULL, &http_file_get, download); pthread_create(&(download->worker), NULL, &http_file_get, download);
http_download_add_download(download); http_download_add_download(download);

View File

@ -35,7 +35,6 @@
#include <assert.h> #include <assert.h>
#include <signal/signal_protocol.h> #include <signal/signal_protocol.h>
#include <signal/signal_protocol_types.h> #include <signal/signal_protocol_types.h>
#include <gcrypt.h>
#include "log.h" #include "log.h"
#include "omemo/omemo.h" #include "omemo/omemo.h"
@ -377,7 +376,7 @@ out:
return res; return res;
} }
int gcry_error_t
aes256gcm_crypt_file(FILE* in, FILE* out, off_t file_size, aes256gcm_crypt_file(FILE* in, FILE* out, off_t file_size,
unsigned char key[], unsigned char nonce[], bool encrypt) unsigned char key[], unsigned char nonce[], bool encrypt)
{ {
@ -416,7 +415,7 @@ aes256gcm_crypt_file(FILE* in, FILE* out, off_t file_size,
off_t bytes_read = 0, bytes_available = 0, read_size = 0; off_t bytes_read = 0, bytes_available = 0, read_size = 0;
while (bytes_read < file_size) { while (bytes_read < file_size) {
bytes_available = file_size - bytes_read; bytes_available = file_size - bytes_read;
if (!bytes_available) { if (!bytes_available || ferror(in) != 0) {
break; break;
} }

View File

@ -35,6 +35,7 @@
#include <stdio.h> #include <stdio.h>
#include <stdbool.h> #include <stdbool.h>
#include <signal/signal_protocol_types.h> #include <signal/signal_protocol_types.h>
#include <gcrypt.h>
#define AES128_GCM_KEY_LENGTH 16 #define AES128_GCM_KEY_LENGTH 16
#define AES128_GCM_IV_LENGTH 12 #define AES128_GCM_IV_LENGTH 12
@ -183,7 +184,7 @@ int aes128gcm_decrypt(unsigned char* plaintext,
size_t ciphertext_len, const unsigned char* const iv, size_t iv_len, size_t ciphertext_len, const unsigned char* const iv, size_t iv_len,
const unsigned char* const key, const unsigned char* const tag); const unsigned char* const key, const unsigned char* const tag);
int aes256gcm_crypt_file(FILE* in, FILE* out, off_t file_size, gcry_error_t aes256gcm_crypt_file(FILE* in, FILE* out, off_t file_size,
unsigned char key[], unsigned char nonce[], bool encrypt); unsigned char key[], unsigned char nonce[], bool encrypt);
char* aes256gcm_create_secure_fragment(unsigned char* key, char* aes256gcm_create_secure_fragment(unsigned char* key,

View File

@ -1713,7 +1713,7 @@ _bytes_from_hex(const char* hex, size_t hex_size,
} }
} }
int gcry_error_t
omemo_decrypt_file(FILE* in, FILE* out, off_t file_size, const char* fragment) omemo_decrypt_file(FILE* in, FILE* out, off_t file_size, const char* fragment)
{ {
char nonce_hex[AESGCM_URL_NONCE_LEN]; char nonce_hex[AESGCM_URL_NONCE_LEN];
@ -1733,7 +1733,8 @@ omemo_decrypt_file(FILE* in, FILE* out, off_t file_size, const char* fragment)
_bytes_from_hex(key_hex, AESGCM_URL_KEY_LEN, _bytes_from_hex(key_hex, AESGCM_URL_KEY_LEN,
key, OMEMO_AESGCM_KEY_LENGTH); key, OMEMO_AESGCM_KEY_LENGTH);
int crypt_res = aes256gcm_crypt_file(in, out, file_size, key, nonce, false); gcry_error_t crypt_res;
crypt_res = aes256gcm_crypt_file(in, out, file_size, key, nonce, false);
gcry_free(key); gcry_free(key);

View File

@ -101,6 +101,5 @@ char* omemo_on_message_send(ProfWin* win, const char* const message, gboolean re
char* omemo_on_message_recv(const char* const from, uint32_t sid, const unsigned char* const iv, size_t iv_len, GList* keys, const unsigned char* const payload, size_t payload_len, gboolean muc, gboolean* trusted); char* omemo_on_message_recv(const char* const from, uint32_t sid, const unsigned char* const iv, size_t iv_len, GList* keys, const unsigned char* const payload, size_t payload_len, gboolean muc, gboolean* trusted);
char* omemo_encrypt_file(FILE* in, FILE* out, off_t file_size, int* gcry_res); char* omemo_encrypt_file(FILE* in, FILE* out, off_t file_size, int* gcry_res);
int omemo_decrypt_file(FILE* in, FILE* out, off_t file_size, const char* fragment); gcry_error_t omemo_decrypt_file(FILE* in, FILE* out, off_t file_size, const char* fragment); void omemo_free(void* a);
void omemo_free(void* a);
int omemo_parse_aesgcm_url(const char* aesgcm_url, char** https_url, char** fragment); int omemo_parse_aesgcm_url(const char* aesgcm_url, char** https_url, char** fragment);

View File

@ -67,56 +67,87 @@ aesgcm_file_get(void* userdata)
char* https_url = NULL; char* https_url = NULL;
char* fragment = NULL; char* fragment = NULL;
const size_t err_len = 100;
char err_buf[err_len];
if (omemo_parse_aesgcm_url(aesgcm_dl->url, &https_url, &fragment) != 0) { if (omemo_parse_aesgcm_url(aesgcm_dl->url, &https_url, &fragment) != 0) {
http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url, http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url,
"Download failed: Cannot parse URL."); "Download failed: Cannot parse URL '%s'.",
aesgcm_dl->url);
return NULL; return NULL;
} }
int tmpfd;
char* tmpname = NULL; char* tmpname = NULL;
if ((tmpfd = g_file_open_tmp("profanity.XXXXXX", &tmpname, NULL)) == -1) { if (g_file_open_tmp("profanity.XXXXXX", &tmpname, NULL) == -1) {
strerror_r(errno, err_buf, err_len);
http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url, http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url,
"Downloading '%s' failed: Unable to create " "Downloading '%s' failed: Unable to create "
"temporary ciphertext file for writing.", "temporary ciphertext file for writing "
https_url); "(%s).",
https_url, err_buf);
return NULL; return NULL;
} }
FILE* tmpfh = fdopen(tmpfd, "wb");
// Remove the file once it is closed. FILE* outfh = fopen(aesgcm_dl->filename, "wb");
remove(tmpname); if (outfh == NULL) {
free(tmpname); strerror_r(errno, err_buf, err_len);
http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url,
"Downloading '%s' failed: Unable to open "
"output file at '%s' for writing (%s).",
https_url, aesgcm_dl->filename, err_buf);
return NULL;
}
HTTPDownload* http_dl = malloc(sizeof(HTTPDownload)); HTTPDownload* http_dl = malloc(sizeof(HTTPDownload));
http_dl->window = aesgcm_dl->window; http_dl->window = aesgcm_dl->window;
http_dl->worker = aesgcm_dl->worker; http_dl->worker = aesgcm_dl->worker;
http_dl->url = https_url; http_dl->url = strdup(https_url);
http_dl->filehandle = tmpfh; http_dl->filename = strdup(tmpname);
http_dl->close = 0;
aesgcm_dl->http_dl = http_dl; aesgcm_dl->http_dl = http_dl;
// TODO: Verify result. http_file_get(http_dl); // TODO(wstrm): Verify result.
http_file_get(http_dl);
// Force flush as the decrypt function will read from the same stream. FILE* tmpfh = fopen(tmpname, "rb");
fflush(tmpfh); if (tmpfh == NULL) {
rewind(tmpfh); strerror_r(errno, err_buf, err_len);
int crypt_res = omemo_decrypt_file(tmpfh, aesgcm_dl->filehandle,
http_dl->bytes_received, fragment);
fclose(tmpfh);
if (crypt_res != 0) {
http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url, http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url,
"Downloading '%s' failed: Failed to decrypt" "Downloading '%s' failed: Unable to open "
"file.", "temporary file at '%s' for reading (%s).",
https_url); aesgcm_dl->url, aesgcm_dl->filename, err_buf);
return NULL;
} }
fclose(aesgcm_dl->filehandle); gcry_error_t crypt_res;
crypt_res = omemo_decrypt_file(tmpfh, outfh,
http_dl->bytes_received, fragment);
if (fclose(tmpfh) == EOF) {
strerror_r(errno, err_buf, err_len);
cons_show_error(err_buf);
}
remove(tmpname);
free(tmpname);
if (crypt_res != GPG_ERR_NO_ERROR) {
http_print_transfer_update(aesgcm_dl->window, aesgcm_dl->url,
"Downloading '%s' failed: Failed to decrypt "
"file (%s).",
https_url, gcry_strerror(crypt_res));
}
if (fclose(outfh) == EOF) {
strerror_r(errno, err_buf, err_len);
cons_show_error(err_buf);
}
free(https_url);
free(fragment);
free(aesgcm_dl->filename);
free(aesgcm_dl->url);
free(aesgcm_dl);
return NULL; return NULL;
} }

View File

@ -50,7 +50,7 @@
typedef struct aesgcm_download_t typedef struct aesgcm_download_t
{ {
char* url; char* url;
FILE* filehandle; char* filename;
ProfWin* window; ProfWin* window;
pthread_t worker; pthread_t worker;
HTTPDownload* http_dl; HTTPDownload* http_dl;

View File

@ -109,6 +109,9 @@ http_file_get(void* userdata)
{ {
HTTPDownload* download = (HTTPDownload*)userdata; HTTPDownload* download = (HTTPDownload*)userdata;
const size_t err_len = 100;
char err_buf[err_len];
char* err = NULL; char* err = NULL;
CURL* curl; CURL* curl;
@ -118,12 +121,18 @@ http_file_get(void* userdata)
download->bytes_received = 0; download->bytes_received = 0;
pthread_mutex_lock(&lock); pthread_mutex_lock(&lock);
char* msg; http_print_transfer_update(download->window, download->url,
if (asprintf(&msg, "Downloading '%s': 0%%", download->url) == -1) { "Downloading '%s': 0%%", download->url);
msg = strdup(FALLBACK_MSG);
FILE* outfh = fopen(download->filename, "wb");
if (outfh == NULL) {
strerror_r(errno, err_buf, err_len);
http_print_transfer_update(download->window, download->url,
"Downloading '%s' failed: Unable to open "
"output file at '%s' for writing (%s).",
download->url, download->filename, err_buf);
return NULL;
} }
win_print_http_transfer(download->window, msg, download->url);
free(msg);
char* cert_path = prefs_get_string(PREF_TLS_CERTPATH); char* cert_path = prefs_get_string(PREF_TLS_CERTPATH);
pthread_mutex_unlock(&lock); pthread_mutex_unlock(&lock);
@ -142,7 +151,7 @@ http_file_get(void* userdata)
#endif #endif
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L); curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L);
curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void*)download->filehandle); curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void*)outfh);
curl_easy_setopt(curl, CURLOPT_USERAGENT, "profanity"); curl_easy_setopt(curl, CURLOPT_USERAGENT, "profanity");
@ -157,35 +166,31 @@ http_file_get(void* userdata)
curl_easy_cleanup(curl); curl_easy_cleanup(curl);
curl_global_cleanup(); curl_global_cleanup();
if (download->filehandle && download->close) { if (fclose(outfh) == EOF) {
fclose(download->filehandle); strerror_r(errno, err_buf, err_len);
err = strdup(err_buf);
} }
pthread_mutex_lock(&lock); pthread_mutex_lock(&lock);
g_free(cert_path); g_free(cert_path);
if (err) { if (err) {
char* msg;
if (download->cancel) { if (download->cancel) {
if (asprintf(&msg, "Downloading '%s' failed: Download was canceled", download->url) == -1) { http_print_transfer_update(download->window, download->url,
msg = strdup(FALLBACK_MSG); "Downloading '%s' failed: "
} "Download was canceled",
download->url);
} else { } else {
if (asprintf(&msg, "Downloading '%s' failed: %s", download->url, err) == -1) { http_print_transfer_update(download->window, download->url,
msg = strdup(FALLBACK_MSG); "Downloading '%s' failed: %s",
download->url, err);
} }
win_update_entry_message(download->window, download->url, msg);
}
cons_show_error(msg);
free(msg);
free(err); free(err);
} else { } else {
if (!download->cancel) { if (!download->cancel) {
if (asprintf(&msg, "Downloading '%s': 100%%", download->url) == -1) { http_print_transfer_update(download->window, download->url,
msg = strdup(FALLBACK_MSG); "Downloading '%s': 100%%",
} download->url);
win_update_entry_message(download->window, download->url, msg);
win_mark_received(download->window, download->url); win_mark_received(download->window, download->url);
free(msg);
} }
} }
@ -193,6 +198,7 @@ http_file_get(void* userdata)
pthread_mutex_unlock(&lock); pthread_mutex_unlock(&lock);
free(download->url); free(download->url);
free(download->filename);
free(download); free(download);
return NULL; return NULL;

View File

@ -49,12 +49,11 @@
typedef struct http_download_t typedef struct http_download_t
{ {
char* url; char* url;
FILE* filehandle; char* filename;
curl_off_t bytes_received; curl_off_t bytes_received;
ProfWin* window; ProfWin* window;
pthread_t worker; pthread_t worker;
int cancel; int cancel;
int close;
} HTTPDownload; } HTTPDownload;
void* http_file_get(void* userdata); void* http_file_get(void* userdata);