aboutsummaryrefslogtreecommitdiffstats
path: root/camel/camel-gpg-context.c
diff options
context:
space:
mode:
Diffstat (limited to 'camel/camel-gpg-context.c')
-rw-r--r--camel/camel-gpg-context.c179
1 files changed, 66 insertions, 113 deletions
diff --git a/camel/camel-gpg-context.c b/camel/camel-gpg-context.c
index ad60f88fd3..7919056f9e 100644
--- a/camel/camel-gpg-context.c
+++ b/camel/camel-gpg-context.c
@@ -44,6 +44,9 @@
#include "gal/util/e-iconv.h"
#include "camel-gpg-context.h"
+#include "camel-mime-filter-charset.h"
+#include "camel-stream-filter.h"
+#include "camel-stream-mem.h"
#include "camel-stream-fs.h"
#include "camel-operation.h"
@@ -260,7 +263,8 @@ struct _GpgCtx {
CamelStream *istream;
CamelStream *ostream;
- GByteArray *diagnostics;
+ GByteArray *diagbuf;
+ CamelStream *diagnostics;
int exit_status;
@@ -285,6 +289,8 @@ static struct _GpgCtx *
gpg_ctx_new (CamelSession *session)
{
struct _GpgCtx *gpg;
+ const char *charset;
+ CamelStream *stream;
gpg = g_new (struct _GpgCtx, 1);
gpg->mode = GPG_CTX_MODE_SIGN;
@@ -326,7 +332,24 @@ gpg_ctx_new (CamelSession *session)
gpg->istream = NULL;
gpg->ostream = NULL;
- gpg->diagnostics = g_byte_array_new ();
+ stream = camel_stream_mem_new ();
+ gpg->diagbuf = CAMEL_STREAM_MEM (stream)->buffer;
+
+ if ((charset = e_iconv_locale_charset ()) && !strcasecmp (charset, "UTF-8")) {
+ CamelMimeFilterCharset *filter;
+ CamelStreamFilter *fstream;
+
+ if ((filter = camel_mime_filter_charset_new_convert (charset, "UTF-8"))) {
+ fstream = camel_stream_filter_new_with_stream (stream);
+ camel_stream_filter_add (fstream, (CamelMimeFilter *) filter);
+ camel_object_unref (filter);
+ camel_object_unref (stream);
+
+ stream = (CamelStream *) fstream;
+ }
+ }
+
+ gpg->diagnostics = stream;
return gpg;
}
@@ -400,92 +423,18 @@ gpg_ctx_set_ostream (struct _GpgCtx *gpg, CamelStream *ostream)
gpg->ostream = ostream;
}
-static char *
+static const char *
gpg_ctx_get_diagnostics (struct _GpgCtx *gpg)
{
- return g_strndup (gpg->diagnostics->data, gpg->diagnostics->len);
-}
-
-static char *
-gpg_ctx_get_utf8_diagnostics (struct _GpgCtx *gpg)
-{
- size_t inleft, outleft, converted = 0;
- const char *charset;
- char *out, *outbuf;
- const char *inbuf;
- size_t outlen;
- iconv_t cd;
-
- if (gpg->diagnostics->len == 0)
- return NULL;
-
- /* if the locale is C/POSIX/ASCII/UTF-8 - then there's nothing to do here */
- if (!(charset = e_iconv_locale_charset ()) || !strcasecmp (charset, "UTF-8"))
- return gpg_ctx_get_diagnostics (gpg);
-
- cd = e_iconv_open ("UTF-8", charset);
-
- inbuf = gpg->diagnostics->data;
- inleft = gpg->diagnostics->len;
-
- outleft = outlen = (inleft * 2) + 16;
- outbuf = out = g_malloc (outlen + 1);
-
- do {
- converted = e_iconv (cd, &inbuf, &inleft, &outbuf, &outleft);
- if (converted == (size_t) -1) {
- if (errno == E2BIG) {
- /*
- * E2BIG There is not sufficient room at *outbuf.
- *
- * We just need to grow our outbuffer and try again.
- */
-
- converted = outbuf - out;
- outlen += inleft * 2 + 16;
- out = g_realloc (out, outlen + 1);
- outbuf = out + converted;
- outleft = outlen - converted;
- } else if (errno == EILSEQ) {
- /*
- * EILSEQ An invalid multibyte sequence has been encountered
- * in the input.
- *
- * What we do here is eat the invalid bytes in the sequence and continue
- */
-
- inbuf++;
- inleft--;
- } else if (errno == EINVAL) {
- /*
- * EINVAL An incomplete multibyte sequence has been encounĀ­
- * tered in the input.
- *
- * We assume that this can only happen if we've run out of
- * bytes for a multibyte sequence, if not we're in trouble.
- */
-
- break;
- } else
- goto noop;
- }
- } while (((int) inleft) > 0);
-
- /* flush the iconv conversion */
- e_iconv (cd, NULL, NULL, &outbuf, &outleft);
- e_iconv_close (cd);
-
- /* nul-terminate the string */
- outbuf[0] = '\0';
-
- return out;
-
- noop:
-
- g_free (out);
- e_iconv_close (cd);
+ if (!gpg->diagbuf->len || gpg->diagbuf->data[gpg->diagbuf->len - 1] != '\0') {
+ camel_stream_flush (gpg->diagnostics);
+ if (gpg->diagbuf->len == 0)
+ return NULL;
+
+ g_byte_array_append (gpg->diagbuf, "", 1);
+ }
- return gpg_ctx_get_diagnostics (gpg);
+ return gpg->diagbuf->data;
}
static void
@@ -539,7 +488,7 @@ gpg_ctx_free (struct _GpgCtx *gpg)
if (gpg->ostream)
camel_object_unref (CAMEL_OBJECT (gpg->ostream));
- g_byte_array_free (gpg->diagnostics, TRUE);
+ camel_object_unref (gpg->diagnostics);
g_free (gpg);
}
@@ -813,6 +762,7 @@ gpg_ctx_parse_status (struct _GpgCtx *gpg, CamelException *ex)
status += 9;
if (!strncmp (status, "USERID_HINT ", 12)) {
+ size_t nread, nwritten;
char *hint, *user;
status += 12;
@@ -829,7 +779,9 @@ gpg_ctx_parse_status (struct _GpgCtx *gpg, CamelException *ex)
goto recycle;
}
- user = g_strdup (status);
+ if (!(user = g_locale_to_utf8 (status, -1, &nread, &nwritten, NULL)))
+ user = g_strdup (status);
+
g_strstrip (user);
g_hash_table_insert (gpg->userid_hint, hint, user);
@@ -890,9 +842,11 @@ gpg_ctx_parse_status (struct _GpgCtx *gpg, CamelException *ex)
return -1;
} else if (!strncmp (status, "NODATA", 6)) {
/* this is an error */
- if (gpg->diagnostics->len)
- camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM, "%.*s",
- gpg->diagnostics->len, gpg->diagnostics->data);
+ const char *diagnostics;
+
+ diagnostics = gpg_ctx_get_diagnostics (gpg);
+ if (diagnostics && *diagnostics)
+ camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM, diagnostics);
else
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
_("No data provided"));
@@ -1003,6 +957,7 @@ static int
gpg_ctx_op_step (struct _GpgCtx *gpg, CamelException *ex)
{
fd_set rdset, wrset, *wrsetp = NULL;
+ const char *diagnostics;
struct timeval timeout;
const char *mode;
int maxfd = 0;
@@ -1108,7 +1063,9 @@ gpg_ctx_op_step (struct _GpgCtx *gpg, CamelException *ex)
goto exception;
if (nread > 0) {
- g_byte_array_append (gpg->diagnostics, buffer, nread);
+ printf ("pre-diag: %.*s\n", nread, buffer);
+ camel_stream_write (gpg->diagnostics, buffer, nread);
+ printf ("post-diag: %.*s\n", gpg->diagbuf->len, gpg->diagbuf->data);
} else {
gpg->seen_eof2 = TRUE;
}
@@ -1204,12 +1161,12 @@ gpg_ctx_op_step (struct _GpgCtx *gpg, CamelException *ex)
break;
}
- if (gpg->diagnostics->len) {
+ diagnostics = gpg_ctx_get_diagnostics (gpg);
+ if (diagnostics && *diagnostics) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("Failed to GPG %s: %s\n\n%.*s"),
+ _("Failed to GPG %s: %s\n\n%s"),
mode, g_strerror (errno),
- gpg->diagnostics->len,
- gpg->diagnostics->data);
+ diagnostics);
} else {
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
_("Failed to GPG %s: %s\n"),
@@ -1346,13 +1303,12 @@ gpg_sign (CamelCipherContext *context, const char *userid, CamelCipherHash hash,
}
if (gpg_ctx_op_wait (gpg) != 0) {
- char *diagnostics;
+ const char *diagnostics;
diagnostics = gpg_ctx_get_diagnostics (gpg);
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
diagnostics && *diagnostics ? diagnostics :
_("Failed to execute gpg."));
- g_free (diagnostics);
gpg_ctx_free (gpg);
@@ -1403,7 +1359,7 @@ gpg_verify (CamelCipherContext *context, CamelCipherHash hash,
CamelException *ex)
{
CamelCipherValidity *validity;
- char *diagnostics = NULL;
+ const char *diagnostics = NULL;
struct _GpgCtx *gpg;
char *sigfile = NULL;
gboolean valid;
@@ -1448,15 +1404,12 @@ gpg_verify (CamelCipherContext *context, CamelCipherHash hash,
}
}
- diagnostics = gpg_ctx_get_utf8_diagnostics (gpg);
-
valid = gpg_ctx_op_wait (gpg) == 0;
- gpg_ctx_free (gpg);
-
validity = camel_cipher_validity_new ();
+ diagnostics = gpg_ctx_get_diagnostics (gpg);
camel_cipher_validity_set_valid (validity, valid);
camel_cipher_validity_set_description (validity, diagnostics);
- g_free (diagnostics);
+ gpg_ctx_free (gpg);
if (sigfile) {
unlink (sigfile);
@@ -1526,13 +1479,12 @@ gpg_encrypt (CamelCipherContext *context, gboolean sign, const char *userid,
}
if (gpg_ctx_op_wait (gpg) != 0) {
- char *diagnostics;
+ const char *diagnostics;
diagnostics = gpg_ctx_get_diagnostics (gpg);
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
diagnostics && *diagnostics ? diagnostics :
_("Failed to execute gpg."));
- g_free (diagnostics);
gpg_ctx_free (gpg);
@@ -1583,13 +1535,12 @@ gpg_decrypt (CamelCipherContext *context, CamelStream *istream,
}
if (gpg_ctx_op_wait (gpg) != 0) {
- char *diagnostics;
+ const char *diagnostics;
diagnostics = gpg_ctx_get_diagnostics (gpg);
camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
diagnostics && *diagnostics ? diagnostics :
_("Failed to execute gpg."));
- g_free (diagnostics);
gpg_ctx_free (gpg);
@@ -1629,11 +1580,12 @@ gpg_import_keys (CamelCipherContext *context, CamelStream *istream, CamelExcepti
}
if (gpg_ctx_op_wait (gpg) != 0) {
- char *diagnostics;
+ const char *diagnostics;
diagnostics = gpg_ctx_get_diagnostics (gpg);
- camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM, diagnostics);
- g_free (diagnostics);
+ camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
+ diagnostics && *diagnostics ? diagnostics :
+ _("Failed to execute gpg."));
gpg_ctx_free (gpg);
@@ -1679,11 +1631,12 @@ gpg_export_keys (CamelCipherContext *context, GPtrArray *keys, CamelStream *ostr
}
if (gpg_ctx_op_wait (gpg) != 0) {
- char *diagnostics;
+ const char *diagnostics;
diagnostics = gpg_ctx_get_diagnostics (gpg);
- camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM, diagnostics);
- g_free (diagnostics);
+ camel_exception_set (ex, CAMEL_EXCEPTION_SYSTEM,
+ diagnostics && *diagnostics ? diagnostics :
+ _("Failed to execute gpg."));
gpg_ctx_free (gpg);