[pan2/testing] basic gpg support with gmime 2.6



commit 5cecf5150b47edde007ff02525dbe3f4fd1c7840
Author: Heinrich MÃller <henmull src gnome org>
Date:   Mon Dec 12 22:58:33 2011 +0100

    basic gpg support with gmime 2.6

 README                          |    3 +-
 configure.in                    |   13 +-
 gtkspell/examples/advanced.c    |   82 --
 gtkspell/examples/simple.c      |   58 -
 gtkspell/gtkspell/deprecated.c  |   35 -
 gtkspell/gtkspell/gtkspell.c    |  888 -------------
 pan/data-impl/article-filter.cc |   10 +-
 pan/data-impl/profiles.cc       |    1 -
 pan/data/article-cache.cc       |   14 +-
 pan/data/article-cache.h        |   11 +-
 pan/gui/Makefile.am             |   11 +-
 pan/gui/body-pane.cc            |  108 +--
 pan/gui/body-pane.h             |   11 +-
 pan/gui/gpg.cc                  |  133 --
 pan/gui/gpg.h                   |   79 --
 pan/gui/gui.cc                  |   18 +-
 pan/gui/pan.cc                  |    6 +-
 pan/gui/post-ui.cc              |   51 +-
 pan/gui/post-ui.h               |    4 -
 pan/gui/profiles-dialog.cc      |   80 +-
 pan/gui/profiles-dialog.h       |    3 +-
 pan/tasks/task-upload.cc        |    2 -
 pan/usenet-utils/Makefile.am    |   10 +-
 pan/usenet-utils/gpg.cc         |  134 ++
 pan/usenet-utils/gpg.h          |  132 ++
 pan/usenet-utils/mime-utils.cc  |  591 +++------
 pan/usenet-utils/mime-utils.h   |   40 +-
 uulib/uuscan.c                  |   70 +-
 yenclib/crc32.c                 |  614 ---------
 yenclib/crc32.h                 |   38 -
 yenclib/uuencode.c              | 2692 ---------------------------------------
 yenclib/uunconc.c               | 1718 -------------------------
 32 files changed, 612 insertions(+), 7048 deletions(-)
---
diff --git a/README b/README
index 0896543..05322cd 100644
--- a/README
+++ b/README
@@ -7,7 +7,7 @@ REQUIREMENTS
 
     Pan requires these libraries:
     * GTK+ 2.16.0 or higher (optionally GTK+ 3.0.0 or higher) <http://www.gtk.org/>
-    * gmime 2.4.0 or higher <http://spruce.sourceforge.net/gmime/>
+    * gmime 2.6.0 or higher <http://spruce.sourceforge.net/gmime/>
 
     Spellchecking requires the following library:
     * gtkspell 2.0.7 or higher <http://gtkspell.sourceforge.net>
@@ -16,7 +16,6 @@ REQUIREMENTS
     * Libnotify (0.4.1)
     * Gnome-Keyring (3.2.2)
     * DBUS
-    * GPGME(made-easy) (1.3.0)    
 
     Most Linux distributions already have these; they can also be 
     downloaded from their home pages or installed via packet managers.
diff --git a/configure.in b/configure.in
index a89b89a..1c85995 100644
--- a/configure.in
+++ b/configure.in
@@ -46,7 +46,7 @@ dnl GtkSpell is optional: GTKSPELL_REQUIRED refers to the minimum version
 dnl needed if you want to build Pan with spellchecking in the Post window.
 
 GLIB_REQUIRED=2.14.0
-GMIME_REQUIRED=2.4.0
+GMIME_REQUIRED=2.6.0
 GTK_REQUIRED=2.16.0
 GTK3_REQUIRED=3.0.0
 GTKSPELL_REQUIRED=2.0.7
@@ -54,7 +54,6 @@ OPENSSL_REQUIRED=1.0.0
 LIBNOTIFY_REQUIRED=0.4.1
 LIBGSASL_REQUIRED=1.6.1
 LIBGKR_REQUIRED=3.2.2
-LIBGPGME_REQUIRED=1.3.0
 AC_SUBST(GLIB_REQUIRED)
 AC_SUBST(GMIME_REQUIRED)
 AC_SUBST(GTK_REQUIRED)
@@ -156,16 +155,6 @@ if test "x$enable_gkr" = "xyes" ; then
   fi
 fi
 
-dnl Check for gpgme for message encryption / signing
-AC_ARG_ENABLE([gpgme],
-AC_HELP_STRING([--enable-gpgme],[enable gpgme support]),[enable_gpgme=$enableval],[enable_gpgme=yes])
-if test "x$enable_gpgme" = "xyes" ; then
-  AM_PATH_GPGME([$LIBGPGME_REQUIRED],[HAVE_GPGME="yes"],[HAVE_GPGME="no"])
-  if test "x$HAVE_GPGME" = "xyes"; then
-    AC_DEFINE([HAVE_GPGME],[1],[gpgme for message encryption / signing])
-  fi
-fi
-
 dnl Check to see if strftime supports the use of %l and %k
 AC_MSG_CHECKING(for %l and %k support in strftime)
 AC_TRY_RUN([
diff --git a/pan/data-impl/article-filter.cc b/pan/data-impl/article-filter.cc
index 8bee5a7..82f2dce 100644
--- a/pan/data-impl/article-filter.cc
+++ b/pan/data-impl/article-filter.cc
@@ -189,13 +189,9 @@ ArticleFilter :: test_article (const Data        & data,
         {
           if (cache.contains(article.message_id)) {
             ArticleCache::mid_sequence_t mid(1, article.message_id);
-#ifdef HAVE_GPGME
-            GPGDecErr unused_for_now;
-            GPGSignersInfo unused;
-            GMimeMessage *msg = cache.get_message(mid, unused, unused_for_now);
-#else
-            GMimeMessage *msg = cache.get_message(mid);
-#endif
+            GPGDecErr err;
+            GMimeMessage *msg = cache.get_message(mid,err);
+
             const char *hdr = g_mime_object_get_header(GMIME_OBJECT(msg), criteria._header);
             pass = criteria._text.test (hdr);
             g_object_unref(msg);
diff --git a/pan/data-impl/profiles.cc b/pan/data-impl/profiles.cc
index b844689..646977a 100644
--- a/pan/data-impl/profiles.cc
+++ b/pan/data-impl/profiles.cc
@@ -105,7 +105,6 @@ namespace
           p.use_sigfile = !strcmp (*v, "true");
         p.sig_type = p.GPGSIG;
         p.use_gpgsig = true;
-        std::cerr<<"got profile "<<p.use_sigfile<<" "<<p.use_gpgsig<<"\n";
       }
     }
   }
diff --git a/pan/data/article-cache.cc b/pan/data/article-cache.cc
index f76333c..2d82a31 100644
--- a/pan/data/article-cache.cc
+++ b/pan/data/article-cache.cc
@@ -398,11 +398,7 @@ ArticleCache :: get_message_mem_stream (const Quark& mid) const
 }
 
 GMimeMessage*
-#ifdef HAVE_GPGME
-ArticleCache :: get_message (const mid_sequence_t& mids, GPGSignersInfo& signer_info, GPGDecErr& err) const
-#else
-ArticleCache :: get_message (const mid_sequence_t& mids) const
-#endif
+ArticleCache :: get_message (const mid_sequence_t& mids, GPGDecErr& err) const
 {
    debug ("trying to get a message with " << mids.size() << " parts");
    GMimeMessage * retval = NULL;
@@ -423,13 +419,11 @@ ArticleCache :: get_message (const mid_sequence_t& mids) const
         streams.push_back (stream);
    }
 
+
    // build the message
    if (!streams.empty())
-#ifdef HAVE_GPGME
-     retval = mime :: construct_message (&streams.front(), streams.size(), signer_info, err);
-#else
-     retval = mime :: construct_message (&streams.front(), streams.size());
-#endif
+     retval = mime :: construct_message (&streams.front(), streams.size(), err);
+
    // cleanup
    foreach (streams_t, streams, it)
      g_object_unref (*it);
diff --git a/pan/data/article-cache.h b/pan/data/article-cache.h
index 31937e8..bffe5a7 100644
--- a/pan/data/article-cache.h
+++ b/pan/data/article-cache.h
@@ -28,9 +28,7 @@ extern "C" {
 #include <pan/general/string-view.h>
 #include <pan/general/quark.h>
 
-#ifdef HAVE_GPGME
-  #include <pan/gui/gpg.h>
-#endif
+#include <pan/usenet-utils/gpg.h>
 
 extern "C"
 {
@@ -74,11 +72,8 @@ namespace pan
       void resize ();
       void clear ();
 
-#ifdef HAVE_GPGME
-      GMimeMessage* get_message (const mid_sequence_t&, GPGSignersInfo&, GPGDecErr&) const;
-#else
-      GMimeMessage* get_message (const mid_sequence_t&) const;
-#endif
+      GMimeMessage* get_message (const mid_sequence_t&, GPGDecErr&) const;
+
       typedef std::vector<std::string> strings_t;
       strings_t get_filenames (const mid_sequence_t&);
 
diff --git a/pan/gui/Makefile.am b/pan/gui/Makefile.am
index 2e930cf..7730a7c 100644
--- a/pan/gui/Makefile.am
+++ b/pan/gui/Makefile.am
@@ -1,5 +1,5 @@
 AM_CPPFLAGS = -I top_srcdir@ @GTKSPELL_CFLAGS@ @GTK_CFLAGS@ @GMIME_CFLAGS@ @GLIB_CFLAGS@ \
-							@OPENSSL_CFLAGS@ @LIBNOTIFY_CFLAGS@ @LIBGNOME_KEYRING_1_CFLAGS@ @GPGME_CFLAGS@ -DPANLOCALEDIR=\""$(panlocaledir)"\"
+							@OPENSSL_CFLAGS@ @LIBNOTIFY_CFLAGS@ @LIBGNOME_KEYRING_1_CFLAGS@ -DPANLOCALEDIR=\""$(panlocaledir)"\"
 
 noinst_LIBRARIES = libpangui.a
 
@@ -33,8 +33,7 @@ libpangui_a_SOURCES = \
  server-ui.cc \
  task-pane.cc \
  xface.c \
- url.cc \
- gpg.cc
+ url.cc
 
 noinst_HEADERS = \
  action-manager.h \
@@ -76,8 +75,7 @@ noinst_HEADERS = \
  task-pane.h \
  url.h \
  wait.h \
- xface.h \
- gpg.h
+ xface.h
 
 EXTRA_DIST = \
  panrc.rc
@@ -97,8 +95,7 @@ endif
 
 pan_SOURCES = gui.cc pan.cc $(WINRC)
 pan_LDADD = ./libpangui.a $(WINRCOBJ) ../data-impl/libpandata.a ../tasks/libtasks.a ../data/libdata.a ../usenet-utils/libusenetutils.a ../general/libgeneralutils.a \
-            ../../uulib/libuu.a @GTKSPELL_LIBS@ @GTK_LIBS@ @GMIME_LIBS@ @GLIB_LIBS@ @OPENSSL_LIBS@ @LIBNOTIFY_LIBS@ @LIBGNOME_KEYRING_1_LIBS@ @GPGME_LIBS@
-
+            ../../uulib/libuu.a @GTKSPELL_LIBS@ @GTK_LIBS@ @GMIME_LIBS@ @GLIB_LIBS@ @OPENSSL_LIBS@ @LIBNOTIFY_LIBS@ @LIBGNOME_KEYRING_1_LIBS@
 if HAVE_WIN32
 pan_LDFLAGS = -mwindows
 endif
diff --git a/pan/gui/body-pane.cc b/pan/gui/body-pane.cc
index bd0a79e..55f80c0 100644
--- a/pan/gui/body-pane.cc
+++ b/pan/gui/body-pane.cc
@@ -916,25 +916,6 @@ namespace
 
 }
 
-#ifdef HAVE_GPGME
-bool
-BodyPane ::get_gpgsig_from_gmime_part (GMimeObject* parent, GMimeObject* base, GMimePart * part)
-{
-  GMimeDataWrapper * wrapper (g_mime_part_get_content_object (part));
-  GMimeStream * mem_stream (g_mime_stream_mem_new ());
-  if (wrapper)
-  {
-    g_mime_data_wrapper_write_to_stream (wrapper, mem_stream);
-    g_mime_stream_reset(mem_stream);
-    gpg_decrypt_and_verify(_signer_info, _gpgerr, mem_stream,
-                           g_mime_multipart_index_of(GMIME_MULTIPART(parent),base),
-                           parent);
-    return true;
-  }
-  return false;
-}
-#endif
-
 void
 BodyPane :: append_part (GMimeObject * parent, GMimeObject * obj, GtkAllocation * widget_size)
 {
@@ -1000,8 +981,7 @@ BodyPane :: append_part (GMimeObject * parent, GMimeObject * obj, GtkAllocation
   }
 
   // or, if it's text, display it
-  else if (g_mime_content_type_is_type (type, "text", "*") ||
-           (g_mime_content_type_is_type (type, "*", "pgp-signature")))
+  else if (g_mime_content_type_is_type (type, "text", "*"))
   {
     const char * fallback_charset (_charset.c_str());
     const char * p_flowed (g_mime_object_get_content_type_parameter(obj,"format"));
@@ -1017,16 +997,6 @@ BodyPane :: append_part (GMimeObject * parent, GMimeObject * obj, GtkAllocation
     const bool do_urls (_prefs.get_flag ("highlight-urls", true));
     append_text_buffer_nolock (&_tm, _buffer, str, do_mute, do_smilies, do_markup, do_urls);
     is_done = true;
-#ifdef HAVE_GPGME
-    /* verify signature */
-
-    if (g_mime_content_type_is_type (type, "*", "pgp-signature"))
-    {
-      bool res = get_gpgsig_from_gmime_part(parent, obj, part);
-      std::cerr<<"1023\n";
-      if (res) update_sig_valid(_gpgerr.verify_ok);
-    }
-#endif
   }
 
   // otherwise, bitch and moan.
@@ -1218,6 +1188,7 @@ BodyPane :: set_text_from_message (GMimeMessage * message)
   // the all-headers don't get included in the followup
 
   // set the text buffer...
+
   if (message)
     g_mime_message_foreach (message, foreach_part_cb, this);
 
@@ -1231,7 +1202,6 @@ BodyPane :: set_text_from_message (GMimeMessage * message)
     gtk_text_buffer_get_start_iter  (_buffer, &iter);
     gtk_text_view_scroll_to_iter (GTK_TEXT_VIEW(_text), &iter, 0.0, true, 0.0, 0.0);
   }
-  std::cerr<<"1234\n";
 
 }
 
@@ -1244,17 +1214,23 @@ BodyPane :: refresh ()
 namespace
 {
 
-  std::string get_email_address(std::string& s)
+  /** Separates user id into name and email address */
+  std::pair<std::string,std::string> get_email_address(std::string& s)
   {
+    std::pair<std::string,std::string> ret;
     size_t in = s.find("<");
     size_t out = s.find(">");
-    if (in == std::string::npos || out == std::string::npos) return "...";
+    if (in == std::string::npos ||
+        out == std::string::npos)
+      return ret;
+
+    ret.first = s.substr(0,in-1);
+    ret.second = s.substr (in+1,out-in-1);
 
-    return s.substr (in+1,out-in-1);
+    return ret;
   }
 }
 
-#ifdef HAVE_GPGME
 gboolean
 BodyPane:: on_tooltip_query(GtkWidget  *widget,
                             gint        x,
@@ -1265,27 +1241,23 @@ BodyPane:: on_tooltip_query(GtkWidget  *widget,
 {
   BodyPane* pane = static_cast<BodyPane*>(data);
   GPGDecErr& err = pane->_gpgerr;
-  GPGSignersInfo& info = pane->_signer_info;
-
-//  g_return_val_if_fail(err.dec_ok, false);
-  g_return_val_if_fail(err.err == GPG_ERR_NO_ERROR || err.err == GPG_ERR_NO_DATA, false);
+  GPGSignersInfo& info = err.signers;
 
   if (err.no_sigs) return false;
-  if (!err.v_res) return false;
-  if (!err.v_res->signatures) return false;
-  if (!err.v_res->signatures->fpr) return false;
+  if (info.signers.empty()) return false;
 
   EvolutionDateMaker ed;
 
   char buf[2048];
+  std::pair<std::string,std::string> name_and_email = get_email_address(info.signers[0].name);
   g_snprintf(buf, sizeof(buf),
-             "<u>This is a <b>GPG-Signed</b> message.</u>\n\n"
-             "<b>Signer</b> : %s (\"%s\")\n"
+             "<u>This is a <b>PGP-Signed</b> message.</u>\n\n"
+             "<b>Signer</b> : %s ('%s')\n"
              "<b>Valid until</b> : %s\n"
              "<b>Created on</b> : %s",
-             info.real_name.c_str(), get_email_address(info.uid).c_str(),
-             ed.get_date_string(info.expires),
-             ed.get_date_string(info.creation_timestamp)
+             name_and_email.first.c_str(), name_and_email.second.c_str(),
+             (info.signers[0].never_expires ? "always" : ed.get_date_string(info.signers[0].expires)),
+             ed.get_date_string(info.signers[0].created)
              );
 
   gtk_tooltip_set_icon_from_stock (tooltip, GTK_STOCK_DIALOG_INFO, GTK_ICON_SIZE_DIALOG);
@@ -1299,12 +1271,8 @@ void
 BodyPane :: update_sig_valid(int i)
 {
 
-  std::cerr<<"update sig "<<i<<"\n";
-
   gtk_image_clear(GTK_IMAGE(_sig_icon));
 
-  i = 1;
-
   switch (i)
   {
     case 0:
@@ -1316,36 +1284,30 @@ BodyPane :: update_sig_valid(int i)
       break;
   }
 }
-#endif
 
 void
 BodyPane :: set_article (const Article& a)
 {
   _article = a;
 
-  const char* gpg_sign(0);
-
   if (_message)
     g_object_unref (_message);
-#ifdef HAVE_GPGME
-  _message = _cache.get_message (_article.get_part_mids(), _signer_info, _gpgerr);
-  gpg_sign = g_mime_object_get_header(GMIME_OBJECT(_message), "X-GPG-Signed");
-#else
-  _message = _cache.get_message (_article.get_part_mids());
-#endif
+
+  _gpgerr.clear();
+
+  _message = _cache.get_message (_article.get_part_mids(), _gpgerr);
 
   int val(-1);
-  if (gpg_sign)
   {
-    if (!strcmp(gpg_sign, "valid"))
+    if (_gpgerr.verify_ok && !_gpgerr.no_sigs)
       val = 1;
-    else if (!strcmp(gpg_sign, "invalid"))
+    else if (!_gpgerr.verify_ok && !_gpgerr.no_sigs)
       val = 0;
+    else
+      val = -1;
   }
-#ifdef HAVE_GPGME
-  std::cerr<<"1344\n";
+
   update_sig_valid(val);
-#endif
   refresh ();
 
   _data.mark_read (_article);
@@ -1358,9 +1320,8 @@ BodyPane :: clear ()
     g_object_unref (_message);
   _message = 0;
   refresh ();
-#ifdef HAVE_GPGME
   update_sig_valid(-1);
-#endif
+
 }
 
 void
@@ -1545,7 +1506,8 @@ BodyPane :: BodyPane (Data& data, ArticleCache& cache, Prefs& prefs):
   _cache (cache),
   _hscroll_visible (false),
   _vscroll_visible (false),
-  _message (0)
+  _message (0),
+  _gpgerr(GPG_DECODE)
 {
 
   for (guint i=0; i<NUM_ICONS; ++i)
@@ -1585,12 +1547,10 @@ BodyPane :: BodyPane (Data& data, ArticleCache& cache, Prefs& prefs):
   gtk_label_set_ellipsize (GTK_LABEL(w), PANGO_ELLIPSIZE_MIDDLE);
   gtk_label_set_use_markup (GTK_LABEL(w), true);
   gtk_box_pack_start (GTK_BOX(hbox), w, true, true, PAD_SMALL);
-#ifdef HAVE_GPGME
   gtk_widget_set_size_request (_sig_icon, 32, 32);
   gtk_box_pack_start (GTK_BOX(hbox), _sig_icon, true, true, PAD_SMALL);
   gtk_widget_set_has_tooltip (_sig_icon, true);
   g_signal_connect(_sig_icon,"query-tooltip",G_CALLBACK(on_tooltip_query), this);
-#endif
   w = _xface = gtk_image_new();
   gtk_widget_set_size_request (w, 48, 48);
   gtk_box_pack_start (GTK_BOX(hbox), w, false, false, PAD_SMALL);
@@ -1887,7 +1847,9 @@ BodyPane :: create_followup_or_reply (bool is_reply)
     g_object_unref (wrapper);
     g_object_unref (part);
     g_object_unref (stream);
-//std::cerr << LINE_ID << " here is the modified clone\n [" << g_mime_object_to_string((GMimeObject *)msg) << ']' << std::endl;
+
+  //std::cerr << LINE_ID << " here is the modified clone\n [" << g_mime_object_to_string((GMimeObject *)msg) << ']' << std::endl;
+
   }
 
   return msg;
diff --git a/pan/gui/body-pane.h b/pan/gui/body-pane.h
index 2120f50..8ced5d0 100644
--- a/pan/gui/body-pane.h
+++ b/pan/gui/body-pane.h
@@ -42,15 +42,16 @@ namespace pan
       Data& _data;
       ArticleCache& _cache;
       GtkWidget* _sig_icon;
-#ifdef HAVE_GPGME
+
       void update_sig_valid(int i);
+
       static gboolean on_tooltip_query(GtkWidget  *widget,
                                        gint        x,
                                        gint        y,
                                        gboolean    keyboard_tip,
                                        GtkTooltip *tooltip,
                                        gpointer    data);
-#endif
+
     public:
       BodyPane (Data&, ArticleCache&, Prefs&);
       ~BodyPane ();
@@ -111,11 +112,6 @@ namespace pan
 
       /* updated with values from gmimemessage */
    public:
-#ifdef HAVE_GPGME
-      GPGDecErr _gpgerr;
-      GPGSignersInfo _signer_info;
-      bool get_gpgsig_from_gmime_part (GMimeObject* parent, GMimeObject* base, GMimePart * part);
-#endif
 
     private:
       std::string _hover_url;
@@ -135,6 +131,7 @@ namespace pan
       GMimeMessage * _message;
       TextMassager _tm;
       std::string _charset;
+      GPGDecErr _gpgerr;
   };
 }
 
diff --git a/pan/gui/gui.cc b/pan/gui/gui.cc
index aa8e1ce..334f7b5 100644
--- a/pan/gui/gui.cc
+++ b/pan/gui/gui.cc
@@ -69,9 +69,7 @@ extern "C" {
 
 #include "profiles-dialog.h"
 
-#ifdef HAVE_GPGME
-  #include "gpg.h"
-#endif
+#include <pan/usenet-utils/gpg.h>
 
 namespace pan
 {
@@ -344,9 +342,8 @@ GUI :: GUI (Data& data, Queue& queue, Prefs& prefs, GroupPrefs& group_prefs):
         on_queue_task_active_changed (queue, *(*it), true);
     }
   }
-#ifdef HAVE_GPGME
+
   init_gpg();
-#endif
 }
 
 namespace
@@ -396,9 +393,7 @@ GUI :: ~GUI ()
     g_object_unref (*it);
   g_object_unref (G_OBJECT(_ui_manager));
 
-#ifdef HAVE_GPGME
   deinit_gpg();
-#endif
 }
 
 /***
@@ -695,13 +690,8 @@ namespace
     virtual void on_progress_finished (Progress&, int status)
     {
       if (status == OK) {
-#ifdef HAVE_GPGME
-        GPGDecErr unused;
-        GPGSignersInfo unuse;
-        GMimeMessage * message = _cache.get_message (_article.get_part_mids(), unuse, unused);
-#else
-        GMimeMessage * message = _cache.get_message (_article.get_part_mids());
-#endif
+        GPGDecErr err;
+        GMimeMessage * message = _cache.get_message (_article.get_part_mids(),err);
         g_mime_message_foreach (message, foreach_part_cb, this);
         g_object_unref (message);
       }
diff --git a/pan/gui/pan.cc b/pan/gui/pan.cc
index 9dd8968..bb16456 100644
--- a/pan/gui/pan.cc
+++ b/pan/gui/pan.cc
@@ -51,11 +51,7 @@ extern "C" {
   #include <pan/tasks/socket-impl-openssl.h>
 #endif
 
-#ifdef HAVE_GPGME
-  #include <gpgme.h>
-  #include <pan/gui/gpg.h>
-#endif
-
+#include <pan/usenet-utils/gpg.h>
 #include <pan/data/cert-store.h>
 #include <pan/tasks/socket-impl-gio.h>
 #include <pan/tasks/socket-impl-main.h>
diff --git a/pan/gui/post-ui.cc b/pan/gui/post-ui.cc
index 9d7fd94..ca0247d 100644
--- a/pan/gui/post-ui.cc
+++ b/pan/gui/post-ui.cc
@@ -38,6 +38,7 @@ extern "C" {
 #include <pan/general/macros.h>
 #include <pan/general/utf8-utils.h>
 #include <pan/usenet-utils/gnksa.h>
+#include <pan/usenet-utils/gpg.h>
 #include <pan/usenet-utils/message-check.h>
 #include <pan/usenet-utils/mime-utils.h>
 #include <pan/data/data.h>
@@ -59,11 +60,6 @@ extern "C" {
 #define DEFAULT_SPELLCHECK_FLAG false
 #endif
 
-#ifdef HAVE_GPGME
-  #include <gpgme.h>
-  #include <pan/gui/gpg.h>
-#endif
-
 using namespace pan;
 
 #define USER_AGENT_PREFS_KEY "add-user-agent-header-when-posting"
@@ -132,7 +128,6 @@ namespace pan
     if (!self->_realized) return;
 
     gpg_sign = gtk_toggle_action_get_active (toggle);
-    std::cerr<<"gpg sign toggle "<<gpg_sign<<"\n";
   }
 
   void on_spellcheck_toggled (GtkToggleAction * toggle, gpointer post_g)
@@ -1025,44 +1020,19 @@ PostUI :: maybe_post_message (GMimeMessage * message)
 
   if(_file_queue_empty)
   {
-#ifdef HAVE_GPGME
-    GPGEncErr fail;
-    if (user_has_gpg && gpg_enc && !gpg_sign)
-    {
-      std::cerr<<"encrypt\n";
-      Profile p(get_current_profile());
-      std::string res = gpg_encrypt(p.gpg_sig_uid, get_body(), fail);
-      if (gpgme_err_code(fail.err) == GPG_ERR_NO_ERROR)
-        gtk_text_buffer_set_text (_body_buf, res.c_str(), res.size());
-      else
-      {
-        Log::add_err_va("Failed to encode the Message with your key : \"%s\"",gpgme_strerror(fail.err));
-        return false;
-      }
-    }
-    if (user_has_gpg && gpg_enc && gpg_sign)
-    {
-      std::cerr<<"encrypt\n";
-      Profile p(get_current_profile());
-      std::string res = gpg_encrypt_and_sign(p.gpg_sig_uid,get_body(), fail);
-      if (gpgme_err_code(fail.err) == GPG_ERR_NO_ERROR)
-        gtk_text_buffer_set_text (_body_buf, res.c_str(), res.size());
-      else
-      {
-        Log::add_err_va("Failed to sign and encode the Message with your key : \"%s\"",gpgme_strerror(fail.err));
-        return false;
-      }
-    }
 
-#endif
     GMimeMessage* msg = new_message_from_ui(POSTING);
     bool go_on(true);
 
-#ifdef HAVE_GPGME
+    GPtrArray * rcp = g_ptr_array_new();
     Profile p(get_current_profile());
+    g_ptr_array_add(rcp, (gpointer)p.gpg_sig_uid.c_str());
     if (user_has_gpg && gpg_sign && !gpg_enc)
-      go_on = go_on && message_add_signed_part(p.gpg_sig_uid, get_body(), msg, fail);
-#endif
+      go_on = go_on && message_add_signed_part(p.gpg_sig_uid, get_body(), msg);
+    else if (user_has_gpg && gpg_enc && !gpg_sign)
+      go_on = go_on && gpg_encrypt(p.gpg_sig_uid, get_body(), msg, rcp, false);
+    else if (user_has_gpg && gpg_enc && gpg_sign)
+      go_on = go_on && gpg_encrypt(p.gpg_sig_uid, get_body(), msg, rcp, true);
 
     if (go_on)
     {
@@ -1070,6 +1040,7 @@ PostUI :: maybe_post_message (GMimeMessage * message)
       _post_task->add_listener (this);
       _queue.add_task (_post_task, Queue::TOP);
     }
+
   } else {
 
     // prepend header for xml file (if one was chosen)
@@ -1545,7 +1516,7 @@ PostUI :: new_message_from_ui (Mode mode, bool copy_body)
 
     GMimeContentType * type = g_mime_content_type_new_from_string (pch);
     g_free (pch);
-//    g_mime_object_set_content_type ((GMimeObject *) part, type); // part owns type now. type isn't refcounted.
+    g_mime_object_set_content_type ((GMimeObject *) part, type); // part owns type now. type isn't refcounted.
     g_mime_part_set_content_object (part, content_object);
     g_mime_part_set_content_encoding (part, GMIME_CONTENT_ENCODING_8BIT);
     g_object_unref (content_object);
@@ -1782,7 +1753,7 @@ namespace
     }
     else if (type == Profile::GPGSIG)
     {
-      /// TODO : Perhaps show but omit in gmimemessage, or make a multipart ??
+      /// TODO : Perhaps show but omit in gmimemessage ??
     }
 
     /* Convert signature to UTF-8. Since the signature is a local file,
diff --git a/pan/gui/post-ui.h b/pan/gui/post-ui.h
index c51319a..4ae458c 100644
--- a/pan/gui/post-ui.h
+++ b/pan/gui/post-ui.h
@@ -31,10 +31,6 @@
 
 #include <pan/usenet-utils/MersenneTwister.h>
 
-#ifdef HAVE_GPGME
-  #include <gpgme.h>
-#endif
-
 namespace pan
 {
   class Profiles;
diff --git a/pan/gui/profiles-dialog.cc b/pan/gui/profiles-dialog.cc
index e6f317d..14716f6 100644
--- a/pan/gui/profiles-dialog.cc
+++ b/pan/gui/profiles-dialog.cc
@@ -35,10 +35,7 @@ extern "C" {
 #include "pan-file-entry.h"
 #include "profiles-dialog.h"
 
-#ifdef HAVE_GPGME
-  #include <gpgme.h>
-  #include <pan/gui/gpg.h>
-#endif
+#include <pan/usenet-utils/gpg.h>
 
 using namespace pan;
 
@@ -74,13 +71,15 @@ namespace
 
     if (row == 2) // GPG
     {
+      gtk_widget_set_tooltip_text(d->_signature_file_combo_box,
+                                  _("Please choose your Email Address according to your PGP key's user id."));
       gtk_widget_hide (d->_signature_file);
-      gtk_widget_show (d->_gpg_sig_entry);
+
     }
     else
     {
+      gtk_widget_set_has_tooltip(d->_signature_file_combo_box,false);
       gtk_widget_show (d->_signature_file);
-      gtk_widget_hide (d->_gpg_sig_entry);
     }
   }
 
@@ -115,7 +114,6 @@ namespace
   }
 }
 
-/// TODO (perhaps) beautify this!
 
 ProfileDialog :: ProfileDialog (const Data         & data,
                                 const StringView   & profile_name,
@@ -148,6 +146,7 @@ ProfileDialog :: ProfileDialog (const Data         & data,
     HIG :: workarea_add_row (t, &row, _("_Full Name:"), w);
     w = _address_entry = gtk_entry_new ();
     set_entry (w, profile.address);
+    gtk_widget_set_tooltip_text(w, _("Your Email Address. Note that this has to match your GPG Signature's Address to verify messages correctly."));
     HIG :: workarea_add_row (t, &row, _("_Email Address:"), w);
     w = _server_combo = make_servers_combo (data, profile.posting_server);
     HIG :: workarea_add_row (t, &row, _("_Post Articles via:"), w);
@@ -167,49 +166,18 @@ ProfileDialog :: ProfileDialog (const Data         & data,
     GtkTreeIter iter;
     GtkListStore * store;
     GtkCellRenderer * renderer;
-#ifdef HAVE_GPGME
-    std::map<std::string, int> author_numbers;
-    store = gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_STRING);
-    int cnt(0);
-    foreach (signers_m, gpg_signers, it)
-    {
-      gtk_list_store_append (store, &iter);
-      gtk_list_store_set (store, &iter, 0, it->second.real_name.c_str(),  1, it->first.c_str(), -1);
-      author_numbers.insert(std::pair<std::string, int>(it->first,cnt++));
-    }
-    w = gtk_combo_box_new_with_model (GTK_TREE_MODEL(store));
-    hbox = gtk_hbox_new(FALSE, 3);
-    l = gtk_label_new(_("Signer : "));
-    gtk_misc_set_alignment (GTK_MISC(l), 0.0f, 0.5f);
-    gtk_box_pack_start(GTK_BOX(hbox), l, false, false, 0);
-    gtk_box_pack_start(GTK_BOX(hbox), w, true, true, 0);
-    _gpg_sig_entry = hbox;
-    _gpg_sig_entry_box = w;
-
-    int signer_no(0);
-    if (!profile.gpg_sig_uid.empty())
-    {
-      if (author_numbers.count(profile.gpg_sig_uid) != 0)
-        signer_no = author_numbers.find(profile.gpg_sig_uid)->second;
-    }
-    gtk_combo_box_set_active (GTK_COMBO_BOX(w), signer_no);
 
-    renderer = gtk_cell_renderer_text_new ();
-    gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (w), renderer, true);
-    gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (w), renderer, "text", 0, NULL);
-#endif
     store = gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_INT);
     gtk_list_store_append (store, &iter);
     gtk_list_store_set (store, &iter, 0, _("Text File"),    1, Profile::FILE, -1);
     gtk_list_store_append (store, &iter);
     gtk_list_store_set (store, &iter, 0, _("Text"),         1, Profile::TEXT, -1);
     gtk_list_store_append (store, &iter);
-#ifdef HAVE_GPGME
     gtk_list_store_set (store, &iter, 0, _("PGP Signature"),1, Profile::GPGSIG, -1);
     gtk_list_store_append (store, &iter);
-#endif
     gtk_list_store_set (store, &iter, 0, _("Command"),      1, Profile::COMMAND, -1);
     w = gtk_combo_box_new_with_model (GTK_TREE_MODEL(store));
+
     hbox = gtk_hbox_new(FALSE, 3);
     l = gtk_label_new(_("Signature Type : "));
     gtk_misc_set_alignment (GTK_MISC(l), 0.0f, 0.5f);
@@ -217,32 +185,22 @@ ProfileDialog :: ProfileDialog (const Data         & data,
     gtk_box_pack_start(GTK_BOX(hbox), w, true, true, 0);
     _signature_file_combo = hbox;
     _signature_file_combo_box = w;
-#ifdef HAVE_GPGME
+
     g_signal_connect (w, "changed", G_CALLBACK(on_signature_type_changed), this);
-#endif
+
     renderer = gtk_cell_renderer_text_new ();
     gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (w), renderer, true);
     gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (w), renderer, "text", 0, NULL);
 
     int active = ROW_FILE;
     if (profile.sig_type == profile.TEXT) active = ROW_TEXT;
-#ifdef HAVE_GPGME
     if (profile.sig_type == profile.GPGSIG) active = ROW_GPGSIG;
     if (profile.sig_type == profile.COMMAND) active = ROW_COMMAND;
-#else
-    if (profile.sig_type == profile.COMMAND) active = ROW_GPGSIG;
-#endif
-
 
     gtk_combo_box_set_active (GTK_COMBO_BOX(w), active);
-
     GtkWidget* vbox = gtk_vbox_new(TRUE, 3);
-
     gtk_box_pack_start(GTK_BOX(vbox), _signature_file_combo, false, false, 0);
     gtk_box_pack_start(GTK_BOX(vbox), _signature_file, false, false, 0);
-#ifdef HAVE_GPGME
-    gtk_box_pack_start(GTK_BOX(vbox), _gpg_sig_entry, false, false, 0);
-#endif
     HIG :: workarea_add_row (t, &row, "",vbox);
 
   HIG :: workarea_add_section_divider (t, &row);
@@ -302,10 +260,8 @@ ProfileDialog :: ProfileDialog (const Data         & data,
     gtk_container_add (GTK_CONTAINER(eventbox), scrolled_window);
     HIG :: workarea_add_row (t, &row, _("E_xtra Headers:"), eventbox, w);
 
-  /// REMOVED for now!
-//  on_sig_file_toggled (GTK_TOGGLE_BUTTON(_signature_file_check), _gpg_sig_entry_box);
-//  on_sig_file_toggled (GTK_TOGGLE_BUTTON(_signature_file_check), _signature_file);
-//  on_sig_file_toggled (GTK_TOGGLE_BUTTON(_signature_file_check), _signature_file_combo_box);
+  on_sig_file_toggled (GTK_TOGGLE_BUTTON(_signature_file_check), _signature_file);
+  on_sig_file_toggled (GTK_TOGGLE_BUTTON(_signature_file_check), _signature_file_combo_box);
 
   gtk_box_pack_start (GTK_BOX( gtk_dialog_get_content_area( GTK_DIALOG(_root))), t, true, true, 0);
   gtk_widget_show_all (t);
@@ -314,9 +270,8 @@ ProfileDialog :: ProfileDialog (const Data         & data,
     gtk_window_set_transient_for (GTK_WINDOW(_root), parent);
     gtk_window_set_position (GTK_WINDOW(_root), GTK_WIN_POS_CENTER_ON_PARENT);
   }
-#ifdef HAVE_GPGME
+
   on_signature_type_changed(GTK_COMBO_BOX(_signature_file_combo_box), this);
-#endif
 }
 
 ProfileDialog :: ~ProfileDialog ()
@@ -380,14 +335,9 @@ ProfileDialog :: get_profile (std::string& profile_name, Profile& profile)
   profile.use_gpgsig = (type == profile.GPGSIG);
   if (!profile.use_gpgsig)
     from_entry (file_entry_gtk_entry(_signature_file), profile.signature_file);
-#ifdef HAVE_GPGME
-  char* uid;
-  combo = GTK_COMBO_BOX (_gpg_sig_entry_box);
-  gtk_combo_box_get_active_iter (combo, &iter);
-  model = gtk_combo_box_get_model (combo);
-  gtk_tree_model_get (model, &iter, 1, &uid, -1);
-  profile.gpg_sig_uid = uid;
-#endif
+
+  profile.gpg_sig_uid = profile.address;
+
   char * pch;
   combo = GTK_COMBO_BOX (_server_combo);
   gtk_combo_box_get_active_iter (combo, &iter);
diff --git a/pan/gui/profiles-dialog.h b/pan/gui/profiles-dialog.h
index b595a4d..28a4296 100644
--- a/pan/gui/profiles-dialog.h
+++ b/pan/gui/profiles-dialog.h
@@ -55,11 +55,10 @@ namespace pan
       GtkWidget * _signature_file_check;
       GtkWidget * _server_combo;
       GtkWidget * _extra_headers_tv;
-      GtkWidget * _signature_file_combo_box;
       GtkWidget * _signature_file_combo;
-      GtkWidget * _gpg_sig_entry_box;
     public:
 
+      GtkWidget * _signature_file_combo_box;
       GtkWidget * _signature_file;
       GtkWidget * _gpg_sig_entry;
 
diff --git a/pan/tasks/task-upload.cc b/pan/tasks/task-upload.cc
index e487ca8..d4b1f99 100644
--- a/pan/tasks/task-upload.cc
+++ b/pan/tasks/task-upload.cc
@@ -212,8 +212,6 @@ TaskUpload :: prepend_headers(GMimeMessage* msg, TaskUpload::Needed * n, std::st
       all = g_mime_object_to_string ((GMimeObject *) msg);
     else if (_first && _queue_pos == 0)
       all = g_mime_object_get_headers ((GMimeObject *) msg);
-    else
-      all = g_mime_object_get_headers ((GMimeObject *) msg);
 
     out << all << "\n";
     if (_first && _queue_pos == -1) g_free(all);
diff --git a/pan/usenet-utils/Makefile.am b/pan/usenet-utils/Makefile.am
index 1ebac86..33889ad 100644
--- a/pan/usenet-utils/Makefile.am
+++ b/pan/usenet-utils/Makefile.am
@@ -1,6 +1,4 @@
-AM_CPPFLAGS = -I top_srcdir@ @GMIME_CFLAGS@ @GLIB_CFLAGS@ @GPGME_CFLAGS@
-
-AM_LDFLAGS = @GPGME_LIBS@
+AM_CPPFLAGS = -I top_srcdir@ @GMIME_CFLAGS@ @GLIB_CFLAGS@
 
 noinst_LIBRARIES = libusenetutils.a
 
@@ -14,7 +12,8 @@ libusenetutils_a_SOURCES = \
  scorefile.cc \
  text-massager.cc \
  url-find.cc \
- blowfish.cc
+ blowfish.cc \
+ gpg.cc
 
 noinst_HEADERS = \
  defgroup.h \
@@ -29,7 +28,8 @@ noinst_HEADERS = \
  url-find.h \
  ssl-utils.h \
  blowfish.h \
- blowfish_cyphers.h
+ blowfish_cyphers.h \
+ gpg.h
 
 #noinst_PROGRAMS = \
 # gnksa-test \
diff --git a/pan/usenet-utils/gpg.cc b/pan/usenet-utils/gpg.cc
new file mode 100644
index 0000000..869e2c7
--- /dev/null
+++ b/pan/usenet-utils/gpg.cc
@@ -0,0 +1,134 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/*
+ * Pan - A Newsreader for Gtk+
+ * Copyright (C) 2002-2006  Charles Kerr <charles rebelbase com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; version 2 of the License.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#include <config.h>
+#include <vector>
+#include <iostream>
+#include <pan/general/log.h>
+#include <pan/general/macros.h>
+#include "gpg.h"
+
+extern "C" {
+  #include <stdlib.h>
+  #include <unistd.h>
+  #include <string.h>
+}
+
+namespace pan
+{
+
+  static gboolean
+  request_passwd (GMimeCryptoContext *ctx, const char *user_id, const char *prompt_ctx,
+                  gboolean reprompt, GMimeStream *response, GError **err)
+  {
+    return TRUE;
+  }
+
+  GMimeCryptoContext *gpg_ctx;
+  bool gpg_inited;
+
+
+  void deinit_gpg()
+  {
+    gpg_inited=false;
+    g_object_unref (gpg_ctx);
+  }
+
+  void fill_signer_info(GPGSignersInfo& info, GMimeSignatureList * sig_list)
+  {
+    g_return_if_fail(sig_list);
+
+    int l = g_mime_signature_list_length(sig_list);
+    info.signers.reserve(l);
+    GMimeSignature * sig(0);
+    Signer signer;
+    for (int i=0;i<l;++i)
+    {
+      sig = g_mime_signature_list_get_signature (sig_list, i);
+
+      signer.name = sig->cert->name ? sig->cert->name : "(null)";
+      signer.key_id = sig->cert->keyid ? sig->cert->keyid : "(null)";
+      signer.fpr = sig->cert->fingerprint ? sig->cert->fingerprint : "(null)";
+
+      switch (sig->cert->trust) {
+      case GMIME_CERTIFICATE_TRUST_NONE:
+        signer.trust = "None";
+        break;
+      case GMIME_CERTIFICATE_TRUST_NEVER:
+        signer.trust = "Never";
+      case GMIME_CERTIFICATE_TRUST_UNDEFINED:
+        signer.trust = "Undefined";
+        break;
+      case GMIME_CERTIFICATE_TRUST_MARGINAL:
+        signer.trust = "Marginal";
+        break;
+      case GMIME_CERTIFICATE_TRUST_FULLY:
+        signer.trust = "Fully";
+        break;
+      case GMIME_CERTIFICATE_TRUST_ULTIMATE:
+        signer.trust = "Ultimate";
+        break;
+      }
+
+      switch (sig->status) {
+      case GMIME_SIGNATURE_STATUS_GOOD:
+        signer.status = "GOOD";
+        break;
+      case GMIME_SIGNATURE_STATUS_BAD:
+        signer.status = "BAD";
+        break;
+      case GMIME_SIGNATURE_STATUS_ERROR:
+        signer.status = "ERROR";
+        break;
+      }
+
+      signer.created = sig->created;
+      signer.expires = sig->expires;
+      if (sig->expires == (time_t) 0)
+        signer.never_expires = true;
+
+      if (sig->errors) {
+
+        if (sig->errors & GMIME_SIGNATURE_ERROR_EXPSIG)
+          signer.error = "Expired";
+        if (sig->errors & GMIME_SIGNATURE_ERROR_NO_PUBKEY)
+          signer.error = "No Pub Key";
+        if (sig->errors & GMIME_SIGNATURE_ERROR_EXPKEYSIG)
+          signer.error = "Key Expired";
+        if (sig->errors & GMIME_SIGNATURE_ERROR_REVKEYSIG)
+          signer.error = "Key Revoked";
+      } else {
+          signer.error = "No errors for this signer";
+      }
+
+      info.signers.push_back(signer);
+    }
+  }
+
+  void init_gpg()
+  {
+    gpg_ctx = g_mime_gpg_context_new (request_passwd, "gpg");
+    if (!gpg_ctx) gpg_inited = false; else gpg_inited = true;
+    g_mime_gpg_context_set_auto_key_retrieve(GMIME_GPG_CONTEXT(gpg_ctx),true);
+    g_mime_gpg_context_set_always_trust(GMIME_GPG_CONTEXT(gpg_ctx),false);
+    g_mime_gpg_context_set_use_agent(GMIME_GPG_CONTEXT(gpg_ctx), true);
+  }
+
+
+}
diff --git a/pan/usenet-utils/gpg.h b/pan/usenet-utils/gpg.h
new file mode 100644
index 0000000..e959597
--- /dev/null
+++ b/pan/usenet-utils/gpg.h
@@ -0,0 +1,132 @@
+/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/*
+ * Pan - A Newsreader for Gtk+
+ * Copyright (C) 2002-2006  Charles Kerr <charles rebelbase com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; version 2 of the License.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ */
+
+#ifndef _HAVE_GPGDEFS_H
+#define _HAVE_GPGDEFS_H
+
+#include <gmime/gmime.h>
+#include <map>
+#include <vector>
+
+namespace pan
+{
+
+  struct Signer
+  {
+    std::string name;
+    std::string key_id;
+    std::string fpr;
+    std::string trust;
+    std::string status;
+    std::string error;
+
+    bool ok;
+    bool never_expires;
+    time_t created, expires;
+
+    Signer() : ok(false), never_expires(false), created((time_t)0), expires((time_t)0)  {}
+  };
+
+  struct GPGSignersInfo
+  {
+    std::vector<Signer> signers;
+  };
+
+  typedef std::map<std::string,GPGSignersInfo> signers_m;
+
+  extern GMimeCryptoContext* gpg_ctx;
+  extern bool gpg_inited;
+
+  enum GPGDecType
+  {
+    GPG_VERIFY,
+    GPG_DECODE,
+    GPG_VERIFY_AND_DECODE
+  };
+
+  enum GPGEncType
+  {
+    GPG_ENCODE,
+    GPG_SIGN,
+    GPG_ENCODE_AND_SIGN
+  };
+
+  /* Error struct for gpg_decrypt */
+  struct GPGDecErr
+  {
+    GError* err;
+    std::string output;
+    bool dec_ok;
+    bool verify_ok;
+    bool no_sigs;
+    GPGDecType type;
+    GPGSignersInfo signers;
+    GMimeObject * decrypted;
+    GMimeDecryptResult * result;
+
+    GPGDecErr(GPGDecType t) : err(NULL), type(t), decrypted(0), result(g_mime_decrypt_result_new())  {}
+    ~GPGDecErr()
+      {
+        if (err) g_error_free(err);
+        err = NULL;
+        if (decrypted) g_object_unref(decrypted) ;
+        if (result) g_object_unref(result);
+      }
+      GPGDecErr() : no_sigs(true) {}
+
+    void clear()
+    {
+      if (err) g_error_free(err);
+      err = NULL;
+      signers.signers.clear();
+      verify_ok = false;
+      dec_ok = false;
+      no_sigs = true;
+    }
+
+  };
+
+  /* Error struct for gpg_sign_and_encrypt */
+  struct GPGEncErr
+  {
+    GError* err;
+    std::string output;
+    bool enc_ok;
+    bool sign_ok;
+    bool no_sigs;
+    GPGEncType type;
+    GPGSignersInfo signers;
+
+    GPGEncErr(GPGEncType t) : err(NULL), type(t), no_sigs(true)  {}
+    ~GPGEncErr()
+    {
+      if (err) g_error_free(err);
+      err = NULL;
+    }
+
+  };
+
+  void init_gpg();
+  void deinit_gpg();
+  void fill_signer_info(GPGSignersInfo& info, GMimeSignatureList * sig_list);
+
+}
+
+#endif
+
diff --git a/pan/usenet-utils/mime-utils.cc b/pan/usenet-utils/mime-utils.cc
index d888773..ee32bd6 100644
--- a/pan/usenet-utils/mime-utils.cc
+++ b/pan/usenet-utils/mime-utils.cc
@@ -35,10 +35,6 @@ extern "C"
 #include <pan/general/log.h>
 #include "mime-utils.h"
 
-#ifdef HAVE_GPGME
-  #include <pan/gui/gpg.h>
-#endif
-
 #define is_nonempty_string(a) ((a) && (*a))
 
 using namespace pan;
@@ -334,53 +330,6 @@ namespace
    }
 }
 
-/***
-**** GPG
-***/
-
-namespace
-{
-
-  bool
-  gpg_is_beginning_line (const StringView& line)
-  {
-    if (line.len <= 1) return false;
-    StringView l(line);
-    return  !strncmp (l.str, GPG_MARKER_BEGIN, GPG_MARKER_BEGIN_LEN) ||
-            !strncmp (l.str, GPG_MARKER_SIGNED_BEGIN, GPG_MARKER_SIGNED_BEGIN_LEN) ;
-  }
-
-  bool
-  gpg_is_ending_line (const StringView& line)
-  {
-    if (line.len <= 1) return false;
-    StringView l(line);
-    return  !strncmp (l.str, GPG_MARKER_END, GPG_MARKER_END_LEN) ||
-            !strncmp (l.str, GPG_MARKER_SIGNED_END, GPG_MARKER_SIGNED_END_LEN);
-  }
-
-  bool
-  gpg_is_signed_begin_line (const StringView& line)
-  {
-    if (line.len <= 1) return false;
-    StringView l(line);
-    return !strncmp (l.str, GPG_MARKER_SIGNED_BEGIN, GPG_MARKER_SIGNED_BEGIN_LEN);
-  }
-
-  bool
-  gpg_is_signed_end_line (const StringView& line)
-  {
-    if (line.len <= 1) return false;
-    StringView l(line);
-    return !strncmp (l.str, GPG_MARKER_SIGNED_END, GPG_MARKER_SIGNED_END_LEN);
-  }
-
-}
-
-/***
-****
-***/
-
 namespace
 {
   guint
@@ -411,11 +360,9 @@ namespace
 
 enum EncType
 {
-	ENC_PLAIN = 1 ,
-	ENC_YENC = 1 << 1,
-	ENC_UU = 1 << 2,
-	ENC_GPG = 1 << 3,
-	ENC_BASE64 = 1<< 4
+	ENC_PLAIN ,
+	ENC_YENC,
+	ENC_UU
 };
 
 namespace pan
@@ -483,7 +430,7 @@ namespace pan
 
     if (!part->stream) {
       part->stream = g_mime_stream_mem_new ();
-      if (part->type != ENC_PLAIN && part->type != ENC_GPG) {
+      if (part->type != ENC_PLAIN) {
         part->filter_stream = g_mime_stream_filter_new (part->stream);
         switch (part->type)
         {
@@ -501,8 +448,8 @@ namespace pan
       }
     }
 
-    g_mime_stream_write_to_stream (s, (part->type == ENC_PLAIN || part->type == ENC_GPG) ?
-				   part->stream : part->filter_stream);
+    g_mime_stream_write_to_stream (s, (part->type == ENC_PLAIN ?
+				   part->stream : part->filter_stream));
     g_object_unref (s);
   }
 
@@ -511,12 +458,7 @@ namespace pan
     temp_parts_t master_list;
     temp_parts_t current_list;
     TempPart *uu_temp;
-    std::string gpg_verified;
-#ifdef HAVE_GPGME
-    GPGDecErr gpgerr;
-    GPGSignersInfo signer_info;
-#endif
-    sep_state():uu_temp(NULL), gpg_verified("") {};
+    sep_state():uu_temp(NULL) {};
   };
 
   bool
@@ -528,9 +470,6 @@ namespace pan
     EncType type = ENC_PLAIN;
     GByteArray * line;
     gboolean yenc_looking_for_part_line = FALSE;
-    gboolean gpg_looking_for_line = FALSE;
-    gboolean gpg_is_signed = FALSE;
-    gboolean gpg_signed_end_found = FALSE;
     gint64 linestart_pos = 0;
     gint64 sub_begin = 0;
     gint64 signed_msg_start = 0;
@@ -553,17 +492,6 @@ namespace pan
         line_len = pch - line_str;
       }
 
-      if (gpg_is_signed_begin_line(line_str))
-      {
-        gpg_is_signed = true;
-        signed_msg_start = linestart_pos;
-      }
-      if (gpg_is_signed_end_line(line_str))
-      {
-        gboolean gpg_signed_end_found = true;
-        signed_msg_end = linestart_pos+line_len;
-      }
-
       switch (type)
       {
         case ENC_PLAIN:
@@ -629,14 +557,6 @@ namespace pan
               yenc_looking_for_part_line = cur->y_part!=0;
             }
           }
-          else if (gpg_is_beginning_line (line_str))
-          {
-
-            found = true;
-            cur = new TempPart (type = ENC_GPG);
-            gpg_looking_for_line = true;
-            sub_begin = linestart_pos;
-          }
           else if (state.uu_temp != NULL && is_uu_line(line_str, line_len) )
           {
             // continue an incomplete uu decode
@@ -727,36 +647,6 @@ namespace pan
           }
           break;
         }
-#ifdef HAVE_GPGME
-        case ENC_GPG:
-        {
-          if (gpg_is_ending_line(line_str))
-          {
-            GMimeStream * stream = g_mime_stream_substream (istream, sub_begin, linestart_pos+line_len);
-            GMimeStream * dec = gpg_decrypt_and_verify(state.signer_info, state.gpgerr, stream);
-            if (state.gpgerr.verify_ok)
-              state.gpg_verified = "valid";
-            else if (!state.gpgerr.verify_ok && !state.gpgerr.no_sigs)
-              state.gpg_verified = "invalid";
-
-            gpg_looking_for_line = false;
-            if (state.gpgerr.dec_ok)
-            {
-              apply_source_and_maybe_filter (cur, dec);
-              if( append_if_not_present (master, cur) )
-                append_if_not_present (appendme, cur);
-            }
-            cur = NULL;
-            type = ENC_PLAIN;
-            sub_begin = -1;
-          }
-          else if (gpg_looking_for_line)
-          {
-            ++cur->valid_lines;
-          }
-          break;
-        }
-#endif
       }
     }
 
@@ -990,12 +880,27 @@ namespace
   }
 }
 
-namespace{
+namespace pan
+{
   struct temp_p {
     GMimeObject *parent,*part;
 
     temp_p(GMimeObject *p, GMimeObject *par):parent(p),part(par) {};
   };
+
+  struct QueryMPType
+  {
+    GMimeObject* obj;
+    GPGDecType type;
+    std::string algo;
+
+    QueryMPType() : obj(0), type(GPG_DECODE) {}
+  };
+}
+
+namespace
+{
+
   typedef std::vector<temp_p> temp_p_t;
 
   void find_text_cb(GMimeObject *parent, GMimeObject *part, gpointer data)
@@ -1011,18 +916,53 @@ namespace{
     v->push_back(temp_p(parent,part) );
   }
 
-  void find_gpg_parts_cb(GMimeObject *parent, GMimeObject *part, gpointer data)
+
+  void mixed_mp_cb(GMimeObject *parent, GMimeObject *part, gpointer data)
   {
     if(!GMIME_IS_PART(part))
       return;
 
-    temp_p_t *v( (temp_p_t *) data);
+    QueryMPType * type = static_cast<QueryMPType*>(data);
 
     GMimeContentType * content_type = g_mime_object_get_content_type (part);
-    if (!(g_mime_content_type_is_type (content_type, "multipart", "encrypted") ||
-          g_mime_content_type_is_type (content_type, "text", "plain")))
+//    if (!g_mime_content_type_is_type (content_type, "text", "plain") &&
+//        !g_mime_content_type_is_type (content_type, "application", "pgp-signature"))
+//      return;
+
+//    if (g_mime_content_type_is_type (content_type, "application", "pgp-signature"))
+//    {
+//      g_mime_object_set_content_type_parameter(part, "micalg", "pgp-sha1");//type->algo.c_str());
+//      std::cerr<<g_mime_object_to_string(part)<<"\n";
+//    }
+
+    g_mime_multipart_add (GMIME_MULTIPART(type->obj), part);
+  }
+
+
+  void mixed_mp_find_gpg_params_cb (GMimeObject *parent, GMimeObject *part, gpointer data)
+  {
+
+    if(!GMIME_IS_PART(part))
       return;
-    v->push_back(temp_p(parent,part) );
+
+    QueryMPType * type = static_cast<QueryMPType*>(data);
+
+//    const char* micalg = g_mime_object_get_content_type_parameter (GMIME_OBJECT (part), "micalg");
+//    if (micalg)
+//    {
+//      std::cerr<<"micalg found : "<<micalg<<"\n";
+//      type->algo = micalg;
+//    }
+
+    GMimeContentType * content_type = g_mime_object_get_content_type (part);
+
+    if (!content_type) return;
+
+    if (g_mime_content_type_is_type (content_type, "application", "pgp-signature"))
+      type->type = GPG_VERIFY;
+    else if (g_mime_content_type_is_type (content_type, "application", "pgp-encrypted"))
+      type->type = GPG_DECODE;
+
   }
 
 }
@@ -1031,17 +971,10 @@ namespace{
 ****
 ***/
 
-#ifdef HAVE_GPGME
 GMimeMessage*
 mime :: construct_message (GMimeStream    ** istreams,
-                           int             qty,
-                           GPGSignersInfo & signer_info,
-                           GPGDecErr      & gpgerr)
-#else
-GMimeMessage*
-mime :: construct_message (GMimeStream    ** istreams,
-                           int             qty)
-#endif
+                           int               qty,
+                           GPGDecErr      &  err)
 {
   const char * message_id = "Foo <bar mum>";
   GMimeMessage * retval = 0;
@@ -1049,16 +982,75 @@ mime :: construct_message (GMimeStream    ** istreams,
   // sanity clause
   pan_return_val_if_fail (is_nonempty_string(message_id), NULL);
   pan_return_val_if_fail (istreams!=NULL, NULL);
-  pan_return_val_if_fail (qty>=1, NULL);
+  pan_return_val_if_fail (qty != 0, NULL);
   for (int i=0; i<qty; ++i)
     pan_return_val_if_fail (GMIME_IS_STREAM(istreams[i]), NULL);
 
   // build the GMimeMessages
-  GMimeParser * parser = g_mime_parser_new ();
+  GMimeParser * parser;// = g_mime_parser_new ();
   GMimeMessage ** messages = g_new (GMimeMessage*, qty);
+
   for (int i=0; i<qty; ++i) {
-    g_mime_parser_init_with_stream (parser, istreams[i]);
-    messages[i] = g_mime_parser_construct_message (parser);
+    parser = g_mime_parser_new_with_stream (istreams[i]);
+    messages[i] = g_mime_parser_construct_message(parser);
+    g_object_unref(parser);
+    g_mime_stream_reset(istreams[i]);
+    parser = g_mime_parser_new_with_stream (istreams[i]);
+    GMimeObject* part = g_mime_parser_construct_part(parser);
+    GMimeContentType * type = g_mime_object_get_content_type (part);
+    const bool multipart (GMIME_IS_MULTIPART_SIGNED(part) || GMIME_IS_MULTIPART_ENCRYPTED(part));
+
+    if (GMIME_IS_MULTIPART_SIGNED(part))
+    {
+      err.type = GPG_VERIFY;
+      err.verify_ok = gpg_verify_mps(part, err);
+    }
+    else if (GMIME_IS_MULTIPART_ENCRYPTED(part))
+    {
+      err.type = GPG_DECODE;
+      err.verify_ok =  gpg_verify_mps(part, err);
+    }
+
+    if (type)
+    {
+      if (g_mime_content_type_is_type (type, "multipart", "mixed"))
+      {
+        QueryMPType qtype;
+        g_mime_multipart_foreach (GMIME_MULTIPART(part), mixed_mp_find_gpg_params_cb, &qtype);
+        GMimeObject* o(0);
+
+        if (qtype.type == GPG_VERIFY)
+        {
+          GMimeMultipartSigned* new_mp = g_mime_multipart_signed_new();
+          err.type = GPG_VERIFY;
+          qtype.obj = o = GMIME_OBJECT(new_mp);
+          g_mime_multipart_foreach (GMIME_MULTIPART(part), mixed_mp_cb, &qtype);
+          err.verify_ok = gpg_verify_mps(GMIME_OBJECT(new_mp), err);
+          g_mime_message_set_mime_part(messages[i], GMIME_OBJECT(new_mp));
+          std::cerr<<"algo : "<<g_mime_object_to_string(GMIME_OBJECT(new_mp))<<"\n";
+        }
+
+        if (qtype.type == GPG_DECODE)
+        {
+          GMimeMultipartEncrypted* new_mp = g_mime_multipart_encrypted_new();
+          err.type = GPG_DECODE;
+          qtype.obj = o = GMIME_OBJECT(new_mp);
+          g_mime_multipart_foreach (GMIME_MULTIPART(part), mixed_mp_cb, &qtype);
+          err.verify_ok =  gpg_verify_mps(GMIME_OBJECT(new_mp), err);
+          g_mime_message_set_mime_part(messages[i], GMIME_OBJECT(new_mp));
+        }
+
+        g_object_unref(o);
+
+      }
+      else if (multipart)
+      {
+        if (err.decrypted)
+          g_mime_message_set_mime_part(messages[i], err.decrypted);
+      } else
+          g_mime_message_set_mime_part(messages[i], part);
+    }
+
   }
   g_object_unref (parser);
 
@@ -1084,16 +1076,11 @@ mime :: construct_message (GMimeStream    ** istreams,
   sep_state state;
   if (retval != NULL)
     g_mime_message_foreach(retval, find_text_cb, &partslist);
+
   foreach(temp_p_t, partslist, it)
   {
     temp_p &data(*it);
     handle_uu_and_yenc_in_text_plain_cb(data.parent, data.part, &state);
-    /* set gpg signature verify success/fail flag */
-#ifdef HAVE_GPGME
-    g_mime_object_set_header(GMIME_OBJECT(data.parent), "X-GPG-Signed", state.gpg_verified.c_str());
-    gpgerr = state.gpgerr;
-    signer_info = state.signer_info;
-#endif
   }
 
   // cleanup
@@ -1288,6 +1275,7 @@ namespace
   GMimeObject *
   handle_multipart_mixed (GMimeMultipart *multipart, gboolean *is_html)
   {
+
     GMimeObject *mime_part, *text_part = NULL;
     GMimeContentType *type, *first_type = NULL;
     int count = g_mime_multipart_get_count (multipart);
@@ -1421,121 +1409,7 @@ void pan::pan_g_mime_message_set_message_id (GMimeMessage *msg, const char *mid)
 }
 
 namespace pan
-
 {
-  #ifdef HAVE_GPGME
-  GMimeStream* gpg_decrypt_and_verify (GPGSignersInfo& signer_info, GPGDecErr& info,
-                                       GMimeStream* s, int index, GMimeObject* parent)
-  {
-
-    GMimeStream* decrypted = g_mime_stream_mem_new ();
-
-    ssize_t stream_len = g_mime_stream_length(s);
-
-    char* streambuf = new char[stream_len];
-
-    g_mime_stream_read(s, streambuf, stream_len);
-    g_object_unref(s);
-
-    info.err = GPG_ERR_NO_ERROR;
-
-    gpgme_data_t gpg_buf, gpg_out_buf;
-    gpgme_key_t key;
-
-    StringView v(streambuf);
-    gpgme_data_new_from_mem (&gpg_buf, v.str, v.len, 0);
-
-    gpgme_strerror(gpgme_data_new (&gpg_out_buf));
-
-    gpgme_data_set_encoding (gpg_out_buf, GPGME_DATA_ENCODING_NONE);
-
-    info.err = gpgme_op_decrypt_verify (gpg_ctx, gpg_buf, gpg_out_buf);
-
-    // no data to decrypt, check for signature validity anyway....
-    if (gpgme_err_code(info.err) == GPG_ERR_NO_DATA || gpgme_err_code(info.err) == GPG_ERR_NO_ERROR)
-      info.dec_ok = true;
-    else
-      return decrypted;
-
-    info.dec_res  = gpgme_op_decrypt_result (gpg_ctx);
-    info.v_res    = gpgme_op_verify_result  (gpg_ctx);
-
-    // verify attached sigs, too. that means, no data to decrypt and no signatures found, yet
-    if (gpgme_err_code(info.err) == GPG_ERR_NO_DATA &&
-        gpgme_err_code(info.v_res->signatures->status) == GPG_ERR_BAD_SIGNATURE && parent)
-    {
-      int count(0); int no(0);
-      size_t len;
-      char* body(0);
-      count = g_mime_multipart_get_count (GMIME_MULTIPART(parent));
-      no = index;
-      gpgme_data_t body_data;
-
-      for (int i=0;i<count;++i)
-      {
-
-        if (i == no) continue;
-        GMimeObject* part_of = g_mime_multipart_get_part (GMIME_MULTIPART(parent), i);
-        if (part_of)
-        {
-          body = pan_g_mime_part_get_content(GMIME_PART(part_of),&len);
-          if (body)
-          {
-            gpgme_data_new_from_mem (&body_data, body, len, 0);
-            info.err  = gpgme_op_verify (gpg_ctx, gpg_buf, body_data, 0);
-            info.v_res= gpgme_op_verify_result  (gpg_ctx);
-            gpgme_data_release(body_data);
-            g_free(body);
-            if (info.err == GPG_ERR_NO_ERROR &&
-                gpgme_err_code(info.v_res->signatures->status) == GPG_ERR_NO_ERROR) break;
-          }
-        }
-      }
-    }
-
-    if (info.v_res->signatures)
-    {
-      info.no_sigs = false;
-      if (gpgme_err_code(info.v_res->signatures->status) == GPG_ERR_NO_ERROR)
-        info.verify_ok = true;
-    }
-    else
-      info.no_sigs = true;
-
-    delete streambuf;
-
-    gpgme_data_seek (gpg_out_buf,SEEK_SET, 0);
-
-    ssize_t ret(0);
-    char buffer[4096]={0};
-    gint64 len(0);
-
-    while ((ret = gpgme_data_read (gpg_out_buf, buffer, sizeof(buffer))) > 0)
-    {
-      len += g_mime_stream_write(decrypted, buffer, ret);
-    }
-
-    g_mime_stream_flush (decrypted);
-    g_mime_stream_reset(decrypted);
-
-    gpgme_data_release(gpg_buf);
-    gpgme_data_release(gpg_out_buf);
-
-    GPGSignersInfo si;
-
-    if (!info.no_sigs)
-    {
-      if (info.v_res->signatures->fpr)
-        gpgme_get_key (gpg_ctx, info.v_res->signatures->fpr, &key, 0);
-      if (key)
-        fill_signer_info(si, key);
-    }
-
-    signer_info = si;
-    info.err = GPG_ERR_NO_ERROR;
-
-    return decrypted;
-  }
 
   void
   mime_part_set_content (GMimePart *part, const char *str)
@@ -1551,167 +1425,126 @@ namespace pan
     g_object_unref (content);
   }
 
-  GMimeMessage*
-  message_add_signed_part (const std::string& uid, const std::string& body_str, GMimeMessage* body, GPGEncErr& fail)
+  GMimeSignatureStatus
+  get_sig_status (GMimeSignatureList *signatures)
   {
-    if (uid.empty()) return 0;
-
-    GMimeMultipart *mp = g_mime_multipart_new_with_subtype ("mixed");
-    g_mime_multipart_set_boundary(mp, NULL);
-    g_mime_multipart_add(mp,g_mime_message_get_mime_part(body));
+    GMimeSignatureStatus status = GMIME_SIGNATURE_STATUS_GOOD;
+    GMimeSignature *sig;
+    int i;
 
-    gpgme_data_t gpg_buf, gpg_out_buf;
-    gpgme_key_t mykey(0), key;
+    if (!signatures || signatures->array->len == 0)
+      return GMIME_SIGNATURE_STATUS_ERROR;
 
-    StringView v(body_str);
-    gpgme_data_new_from_mem (&gpg_buf, v.str, v.len, 0);
-    gpgme_data_new (&gpg_out_buf);
-  //  gpgme_data_set_encoding (gpg_out_buf, GPGME_DATA_ENCODING_BASE64);
-  //  gpgme_data_set_file_name(gpg_out_buf, "signature.asc");
-
-    fail.err = gpgme_op_keylist_start (gpg_ctx, 0, 0);
-    while (!fail.err)
-    {
-      fail.err = gpgme_op_keylist_next (gpg_ctx, &key);
-      if (fail.err) break;
-      if (strcmp(key->subkeys->keyid, uid.c_str()) == 0) { mykey = key; break; }
-    }
-    if (!mykey) { fail.err = GPG_ERR_NO_PUBKEY; return 0; }
-
-    gpgme_signers_clear(gpg_ctx);
-    gpgme_signers_add(gpg_ctx, mykey);
-
-    fail.err      = gpgme_op_sign (gpg_ctx, gpg_buf, gpg_out_buf, GPGME_SIG_MODE_DETACH);
-    fail.sign_res = gpgme_op_sign_result (gpg_ctx);
-    if (fail.err) return 0;
-
-    gpgme_data_seek (gpg_out_buf,SEEK_SET, 0);
-    ssize_t ret;
-    std::stringstream ret_str;
-    char buffer[4096]={0};
-
-    while ((ret = gpgme_data_read (gpg_out_buf, buffer, sizeof(buffer))) > 0)
-    {
-      ret_str << buffer;
+    for (i = 0; i < g_mime_signature_list_length (signatures); i++) {
+      sig = g_mime_signature_list_get_signature (signatures, i);
+      status = MAX (status, sig->status);
     }
 
-    GMimePart *sig = g_mime_part_new_with_type("multipart", "signed");
-    g_mime_object_set_content_type_parameter(GMIME_OBJECT(sig),"protocol","pgp-signature");
-    mime_part_set_content (sig, ret_str.str().c_str());
-
-    gpgme_data_release(gpg_buf);
-    gpgme_data_release(gpg_out_buf);
-
-    g_mime_multipart_add (GMIME_MULTIPART (mp), GMIME_OBJECT (sig));
-
-    g_mime_message_set_mime_part(body,GMIME_OBJECT(mp));
-    g_object_unref(mp);
-
-    fail.err = GPG_ERR_NO_ERROR;
-
-    return body;
+    return status;
   }
 
-  std::string
-  gpg_encrypt(const std::string& uid, const std::string& body, GPGEncErr& fail)
+  bool
+  gpg_verify_mps (GMimeObject* obj, GPGDecErr& info)
   {
-    fail.err = GPG_ERR_GENERAL;
-
-    if (uid.empty()) return "";
 
-    gpgme_data_t gpg_buf, gpg_out_buf;
-    gpgme_key_t mykey(0), key;
+    if (!gpg_inited) return false;
 
-    StringView v(body);
-    gpgme_data_new_from_mem (&gpg_buf, v.str, v.len, 0);
-    gpgme_data_new (&gpg_out_buf);
-    gpgme_data_set_encoding (gpg_out_buf, GPGME_DATA_ENCODING_BASE64);
+    GMimeMultipartSigned * mps(0);
+    GMimeMultipartEncrypted * mpe(0);
 
-    /* find key to uid */
-    fail.err = gpgme_op_keylist_start (gpg_ctx, 0, 0);
-    while (!fail.err)
+    switch (info.type)
     {
-      fail.err = gpgme_op_keylist_next (gpg_ctx, &key);
-      if (fail.err) break;
-      if (strcmp(key->subkeys->keyid, uid.c_str()) == 0) { mykey = key; break; }
+      case GPG_VERIFY:
+        mps = (GMimeMultipartSigned *)obj;
+        break;
+
+      case GPG_DECODE:
+        mpe = (GMimeMultipartEncrypted *)obj;
+        break;
     }
-    if (!mykey) { fail.err = GPG_ERR_NO_PUBKEY; return std::string(""); }
 
-    gpgme_key_t enc_keys[] = { mykey, NULL};
+    info.clear();
 
-    fail.err     = gpgme_op_encrypt (gpg_ctx, enc_keys, GPGME_ENCRYPT_PREPARE, gpg_buf, gpg_out_buf);
-    fail.enc_res = gpgme_op_encrypt_result (gpg_ctx);
-    if (fail.err) return std::string("");
+    if (info.type == GPG_VERIFY)
+    {
+      GMimeSignatureList * sig_list = g_mime_multipart_signed_verify (mps, gpg_ctx, &info.err);
+      if (sig_list) info.no_sigs = false;
+      if (!sig_list) return false;
+      fill_signer_info(info.signers, sig_list);
 
-    gpgme_data_seek (gpg_out_buf,SEEK_SET, 0);
-    ssize_t ret;
-    std::stringstream ret_str;
-    char buffer[4096]={0};
+      return get_sig_status(sig_list) == GMIME_SIGNATURE_STATUS_GOOD;
+    }
 
-    while ((ret = gpgme_data_read (gpg_out_buf, buffer, sizeof(buffer))) > 0)
+    if (info.type == GPG_DECODE)
     {
-      ret_str << buffer;
-    }
+      info.decrypted = g_mime_multipart_encrypted_decrypt (mpe, gpg_ctx, &info.result, &info.err);
+      if (!info.decrypted)
+        if (info.err) return false;
 
-    gpgme_data_release(gpg_buf);
-    gpgme_data_release(gpg_out_buf);
+      GMimeSignatureList * sigs = g_mime_decrypt_result_get_signatures (info.result);
+      if (sigs)
+      {
+        info.no_sigs = false;
+        fill_signer_info(info.signers, sigs);
+      }
+      return get_sig_status(info.result->signatures) == GMIME_SIGNATURE_STATUS_GOOD || !info.err;
+    }
+    return false;
 
-    fail.err = GPG_ERR_NO_ERROR;
-    return ret_str.str();
   }
 
-  std::string
-  gpg_encrypt_and_sign(const std::string& uid, const std::string& body, GPGEncErr& fail)
+  GMimeMessage*
+  message_add_signed_part (const std::string& uid, const std::string& body_str, GMimeMessage* body)
   {
-    fail.err = GPG_ERR_GENERAL;
-
-    if (uid.empty()) return "";
+    if (uid.empty()) return 0;
 
-    gpgme_data_t gpg_buf, gpg_out_buf;
-    gpgme_key_t mykey(0), key;
+    GMimeMultipartSigned *mps;
+    GError* err(NULL);
+    GMimePart* part = g_mime_part_new_with_type("text", "plain");
+    mime_part_set_content (part, body_str.c_str());
 
-    StringView v(body);
-    gpgme_data_new_from_mem (&gpg_buf, v.str, v.len, 0);
-    gpgme_data_new (&gpg_out_buf);
-    gpgme_data_set_encoding (gpg_out_buf, GPGME_DATA_ENCODING_BASE64);
+    mps = g_mime_multipart_signed_new ();
 
-    /* find key to uid */
-    fail.err = gpgme_op_keylist_start (gpg_ctx, 0, 0);
-    while (!fail.err)
+    /* sign the part */
+    if (g_mime_multipart_signed_sign (mps, GMIME_OBJECT (part), gpg_ctx, uid.c_str(), GMIME_DIGEST_ALGO_SHA1, &err) <0)
     {
-      fail.err = gpgme_op_keylist_next (gpg_ctx, &key);
-      if (fail.err) break;
-      if (strcmp(key->subkeys->keyid, uid.c_str()) == 0) { mykey = key; break; }
+      g_object_unref(mps);
+      g_object_unref(G_OBJECT(part));
+      return 0;
     }
-    if (!mykey) { fail.err = GPG_ERR_NO_PUBKEY; return std::string(""); }
 
-    gpgme_signers_clear(gpg_ctx);
-    gpgme_signers_add(gpg_ctx, mykey);
+    g_mime_message_set_mime_part(body,GMIME_OBJECT(mps));
+    g_object_unref(G_OBJECT(part));
+    g_object_unref(mps);
+
+    return body;
+  }
 
-    gpgme_key_t enc_keys[] = { mykey, NULL};
+  GMimeMessage*
+  gpg_encrypt (const std::string& uid, const std::string& body_str,
+               GMimeMessage* body, GPtrArray* rcp, bool sign)
+  {
 
-    fail.err     = gpgme_op_encrypt_sign (gpg_ctx, enc_keys, GPGME_ENCRYPT_PREPARE, gpg_buf, gpg_out_buf);
-    fail.enc_res = gpgme_op_encrypt_result (gpg_ctx);
-    fail.sign_res= gpgme_op_sign_result (gpg_ctx);
-    if (fail.err) return std::string("");
+    GError* err(0);
+    GMimePart* part = g_mime_part_new_with_type("text", "plain");
+    mime_part_set_content (part, body_str.c_str());
 
-    gpgme_data_seek (gpg_out_buf,SEEK_SET, 0);
-    ssize_t ret;
-    std::stringstream ret_str;
-    char buffer[4096]={0};
+    GMimeMultipartEncrypted * mpe = g_mime_multipart_encrypted_new();
 
-    while ((ret = gpgme_data_read (gpg_out_buf, buffer, sizeof(buffer))) > 0)
+    if (g_mime_multipart_encrypted_encrypt(mpe, GMIME_OBJECT (part), gpg_ctx, sign,
+                                           uid.c_str(), GMIME_DIGEST_ALGO_SHA1, rcp, &err) < 0)
     {
-      ret_str << buffer;
+      g_object_unref(mpe);
+      g_object_unref(G_OBJECT(part));
+      return 0;
     }
 
-    gpgme_data_release(gpg_buf);
-    gpgme_data_release(gpg_out_buf);
+    g_mime_message_set_mime_part(body,GMIME_OBJECT(mpe));
+    g_object_unref(G_OBJECT(part));
+    g_object_unref(mpe);
 
-    fail.err = GPG_ERR_NO_ERROR;
-    return ret_str.str();
   }
-#endif
+
 }
 
 
diff --git a/pan/usenet-utils/mime-utils.h b/pan/usenet-utils/mime-utils.h
index b43650d..15e2188 100644
--- a/pan/usenet-utils/mime-utils.h
+++ b/pan/usenet-utils/mime-utils.h
@@ -26,11 +26,7 @@
 #include <gmime/gmime-stream.h>
 #include <gmime/gmime-message.h>
 #include <pan/general/string-view.h>
-
-#ifdef HAVE_GPGME
-  #include <gpgme.h>
-  #include <pan/gui/gpg.h>
-#endif
+#include <pan/usenet-utils/gpg.h>
 
 /***
 **** YENC
@@ -60,27 +56,12 @@
 #define YENC_SHIFT             42
 #define YENC_QUOTE_SHIFT       64
 
-/***
-**** GPG
-***/
-
-#define GPG_MARKER_BEGIN            "-----BEGIN PGP MESSAGE-----"
-#define GPG_MARKER_BEGIN_LEN        27
-#define GPG_MARKER_END              "-----END PGP MESSAGE-----"
-#define GPG_MARKER_END_LEN          25
-#define GPG_MARKER_SIGNED_BEGIN     "-----BEGIN PGP SIGNED MESSAGE-----"
-#define GPG_MARKER_SIGNED_BEGIN_LEN 34
-#define GPG_MARKER_SIGNED_END       "-----END PGP SIGNATURE-----"
-#define GPG_MARKER_SIGNED_END_LEN   27
-
 namespace pan
 {
-#ifdef HAVE_GPGME
-  GMimeStream* gpg_decrypt_and_verify (GPGSignersInfo& signer_info, GPGDecErr& info, GMimeStream* s, int index=0, GMimeObject* parent=0);
-  GMimeMessage* message_add_signed_part (const std::string& uid, const std::string& body_str, GMimeMessage* body, GPGEncErr& fail);
-  std::string gpg_encrypt(const std::string& uid, const std::string& body, GPGEncErr& fail);
-  std::string gpg_encrypt_and_sign(const std::string& uid, const std::string& body, GPGEncErr& fail);
-#endif
+
+  GMimeMessage* message_add_signed_part (const std::string& uid, const std::string& body_str, GMimeMessage* body);
+  GMimeMessage* gpg_encrypt (const std::string& uid, const std::string& body_str, GMimeMessage* body, GPtrArray* rcp, bool sign);
+  bool gpg_verify_mps (GMimeObject*, GPGDecErr&);
 
   /**
    * Utilities to build and parse GMimeMessages.
@@ -93,17 +74,10 @@ namespace pan
    */
   struct mime
   {
-#ifdef HAVE_GPGME
     static GMimeMessage *
     construct_message (GMimeStream      ** istreams,
-                         int              qty,
-                         GPGSignersInfo & signer_info,
-                         GPGDecErr      & gpgerr);
-#else
-    static GMimeMessage *
-    construct_message (GMimeStream      ** istreams,
-                         int              qty);
-#endif
+                       int                 qty,
+                       GPGDecErr         &);
 
     static const char *
     get_charset (GMimeMessage * message);
diff --git a/uulib/uuscan.c b/uulib/uuscan.c
index 55b0b09..f66868f 100644
--- a/uulib/uuscan.c
+++ b/uulib/uuscan.c
@@ -490,7 +490,7 @@ ParseHeader (headers *theheaders, char *line)
      * Some encoders mention the original filename as parameter to
      * Content-Type, others as parameter to Content-Disposition. We
      * do prefer the first solution, but accept this situation, too.
-     * TODO: Read RFC1806
+     * !!Read RFC1806
      */
     if ((ptr = _FP_stristr (line, "name")) != NULL) {
       if (theheaders->fname == NULL && (thenew=ParseValue(ptr)) != NULL) {
@@ -563,12 +563,12 @@ IsKnownHeader (char *line)
         *pch++ = tolower (*line++);
       *pch = '\0';
       pch = buf;
- 
-      // search for that in knownmsgheaders 
+
+      // search for that in knownmsgheaders
       if (bsearch (&pch, knownmsgheaders, n_msg, sizeof(char*), bsearch_strcmp) != NULL)
         retval = 1;
 
-      // search for that in knownmimeheaders 
+      // search for that in knownmimeheaders
       else if (bsearch (&pch, knownmimeheaders, n_mime, sizeof(char*), bsearch_strcmp) != NULL)
         retval = 2;
     }
@@ -717,7 +717,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
       dontcare=0;
     }
 
-    if (boundary != NULL && 
+    if (boundary != NULL &&
 	line[0] == '-' && line[1] == '-' &&
 	strncmp (line+2, boundary, blen) == 0) {
       fseek (datei, oldposition, SEEK_SET);
@@ -746,14 +746,14 @@ ScanData (FILE *datei, char *fname, int *errcode,
 	fseek (datei, oldposition, SEEK_SET);
 	break;
       }
-      
+
       if (*line == '<')
 	ptr = line + 10;
       else
 	ptr = line + 5;
 
       while (*ptr == ' ') ptr++;
-      while (isdigit (*ptr)) 
+      while (isdigit (*ptr))
 	result->mode = result->mode * 8 + *ptr++ - '0';
       while (*ptr == ' ') ptr++;
 
@@ -801,7 +801,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
 	ptr += 8;
 	while (isspace (*ptr)) ptr++;
 	p2 = ptr;
-	while (isalnum(*p2) || 
+	while (isalnum(*p2) ||
 	       *p2 == '.' || *p2=='_' || *p2 == '-' ||
 	       *p2 == '!' || *p2=='@' || *p2 == '$')
 	  p2++;
@@ -874,7 +874,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
 	}
 	else {
 	  p2 = ptr;
-	  while (isalnum(*p2) || 
+	  while (isalnum(*p2) ||
 		 *p2 == '.' || *p2=='_' || *p2 == '-' ||
 		 *p2 == '!' || *p2=='@' || *p2 == '$')
 	    p2++;
@@ -908,8 +908,8 @@ ScanData (FILE *datei, char *fname, int *errcode,
 	  *p2 = c;
 	}
       }
-      
-      /* 
+
+      /*
        * Handling for very short Base64 files.
        */
       if (uu_tinyb64 && !ismime && !uu_more_mime) {
@@ -965,7 +965,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
       /*
        * name continues to the end of the line
        */
-      
+
       _FP_free (result->filename);
       ptr = _FP_strstr (line, " name=") + 6;
       result->filename = _FP_strdup (ptr);
@@ -1007,7 +1007,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
 	if ((ptr = _FP_strstr (line, " end=")) == NULL) {
 	  break;
 	}
-       
+
 	yepartends = atoi (ptr + 5);
       }
       else {
@@ -1113,7 +1113,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
 	if (vflag == BH_ENCODED && bhnf == 0 && result->filename == NULL) {
 	  if (bhdsp == bhds2 ||
 	      ((bhdsp-bhds2) <= (int) bhds2[0] &&
-	       (bhdsp-bhds2) <  256)) { 
+	       (bhdsp-bhds2) <  256)) {
 	    dcc = UUDecodeLine (line, bhds1, BH_ENCODED);
 	    UUbhdecomp (bhds1, bhdsp, &bhl, &bhrpc,
 			dcc, 256-(bhdsp-bhds2), &bhopc);
@@ -1307,7 +1307,7 @@ ScanData (FILE *datei, char *fname, int *errcode,
 
 	  continue;
 	}
-	
+
 	/*
 	 * Select the encoding with the best "history"
 	 */
@@ -1750,7 +1750,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
 
     blen   = strlen (sstate.envelope.boundary);
     lcount = 0;
-    
+
     while (!feof (datei)) {
       if (_FP_fgets (line, line_len, datei) == NULL)
 	break;
@@ -1928,7 +1928,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
 	  preheaders = prevpos;
 	  lcount     = 0;
 	}
-	hcount++; 
+	hcount++;
 	lcount++;
 
 	if (hcount >= hlcount.restart) {
@@ -2083,7 +2083,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
    * stack and dive into the new environment, starting with another
    * preamble.
    */
-			   
+
   if (sstate.ismime && sstate.mimestate == MS_SUBPART) {
     memset (&localenv, 0, sizeof (headers));
     result->startpos = ftell (datei);
@@ -2114,18 +2114,18 @@ ScanPart (FILE *datei, char *fname, int *errcode)
       }
       sstate.isfolder = 0;
       sstate.ismime   = 0;
-      
+
       UUkillheaders (&localenv);
       *errcode = UURET_NOMEM;
       _FP_free (result);
       return NULL;
     }
-    
+
     /* Scan subheader. But what if there is no subheader? */
     hcount = 0;
     lcount = 0;
     preheaders = prevpos;
-    
+
     if (_FP_fgets (line, line_len, datei) == NULL) {
       sstate.isfolder = 0;
       sstate.ismime   = 0;
@@ -2247,7 +2247,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
       prevpos          = ftell (datei);
       blen = strlen (sstate.envelope.boundary);
       lcount = 0;
-      
+
       while (!feof (datei)) {
 	if (_FP_fgets (line, line_len, datei) == NULL) {
 	  line[0] = '\0';
@@ -2449,7 +2449,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
     else {
       UUMessage (uuscan_id, __LINE__, UUMSG_WARNING,
 		 uustring (S_MIME_B_NOT_FOUND));
-      
+
       while (mssdepth) {
 	mssdepth--;
 	UUkillheaders (&(multistack[mssdepth].envelope));
@@ -2469,7 +2469,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
        * Retry, listening to headers this time
        */
       fseek (datei, result->startpos, SEEK_SET);
-      
+
       UUkillfread (result);
       if ((result = (fileread *) malloc (sizeof (fileread))) == NULL) {
 	*errcode = UURET_NOMEM;
@@ -2575,7 +2575,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
      * kill local envelope
      */
     UUkillheaders (&localenv);
-    
+
     return result;
   }
 
@@ -2718,7 +2718,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
 	  }
 	}
 
-	while (!feof (datei) && !IsLineEmpty (line)) { 
+	while (!feof (datei) && !IsLineEmpty (line)) {
 	  if (IsKnownHeader (line))
 	    hcount++;
 	  lcount++;
@@ -2819,7 +2819,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
     }
     result->partno = sstate.envelope.partno;
     result->maxpno = sstate.envelope.numparts;
-    result->flags  = FL_PARTIAL | 
+    result->flags  = FL_PARTIAL |
       ((res==1 || uu_fast_scanning) ? FL_PROPER : 0) |
 	((uu_fast_scanning) ? FL_TOEND : 0);
     result->mimeid = _FP_strdup (sstate.envelope.mimeid);
@@ -2848,7 +2848,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
       sstate.isfolder  = 1;
       sstate.ismime    = 0;
       sstate.mimestate = MS_HEADERS;
-      
+
       UUkillheaders (&sstate.envelope);
       memset (&sstate.envelope, 0, sizeof (headers));
     }
@@ -2921,7 +2921,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
     }
 
     result->startpos = prevpos = ftell (datei);
-    
+
     /*
      * If this is Quoted-Printable or Plain Text, just try looking
      * for the next message header. If uu_fast_scanning, we know
@@ -2951,7 +2951,7 @@ ScanPart (FILE *datei, char *fname, int *errcode)
 	  }
 	}
 
-	while (!feof (datei) && !IsLineEmpty (line)) { 
+	while (!feof (datei) && !IsLineEmpty (line)) {
 	  if (IsKnownHeader (line))
 	    hcount++;
 	  lcount++;
@@ -3131,13 +3131,13 @@ ScanPart (FILE *datei, char *fname, int *errcode)
 
 	sstate.envelope.mimevers = _FP_strdup ("1.0");
 	sstate.envelope.boundary = _FP_strdup (line+2);
-	
+
 	/*
 	 * need restart
 	 */
-	
+
 	fseek (datei, prevpos, SEEK_SET);
-	
+
 	_FP_free (result);
 	return NULL;
       }
@@ -3158,8 +3158,8 @@ ScanPart (FILE *datei, char *fname, int *errcode)
 
   if (sstate.envelope.ctype)
     result->mimetype = _FP_strdup (sstate.envelope.ctype);
-  
-  if ((res=ScanData (datei, fname, errcode, NULL, 
+
+  if ((res=ScanData (datei, fname, errcode, NULL,
 		     sstate.ismime, 1, result))==-1) {
     /* oops, something went wrong */
     sstate.isfolder = 0;



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]