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:
parent
73f313b921
commit
62cbad1c6e
1
.gitignore
vendored
1
.gitignore
vendored
@ -86,3 +86,4 @@ breaks
|
|||||||
|
|
||||||
*.tar.*
|
*.tar.*
|
||||||
*.zip
|
*.zip
|
||||||
|
*.log*
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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,
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
|
@ -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);
|
||||||
|
Loading…
Reference in New Issue
Block a user