Blame SOURCES/0079-ureport-publish-ureport.h-and-refactore-uReport-sour.patch

4b6aa8
From e67393a38e1739ff1b0453fdb3b68ff7e4fb36e9 Mon Sep 17 00:00:00 2001
4b6aa8
From: Jakub Filak <jfilak@redhat.com>
4b6aa8
Date: Fri, 12 Sep 2014 17:20:01 +0200
4b6aa8
Subject: [LIBREPORT PATCH 79/93] ureport: publish ureport.h and refactore
4b6aa8
 uReport source
4b6aa8
4b6aa8
Allow other plugins to submit uReport and allow external applications to
4b6aa8
use libreport to submit uReport.
4b6aa8
4b6aa8
Related to rhbz1139987
4b6aa8
4b6aa8
Signed-off-by: Jakub Filak <jfilak@redhat.com>
4b6aa8
---
4b6aa8
 src/include/Makefile.am        |   4 +
4b6aa8
 src/include/ureport.h          |  76 ++++
4b6aa8
 src/lib/Makefile.am            |   4 +-
4b6aa8
 src/lib/json.c                 | 180 ----------
4b6aa8
 src/lib/ureport.c              | 180 ++++++++++
4b6aa8
 src/lib/ureport.h              |  76 ----
4b6aa8
 src/plugins/Makefile.am        |   4 +-
4b6aa8
 src/plugins/reporter-ureport.c | 777 +++++++++++++++++++++++++++++++++++++++++
4b6aa8
 src/plugins/ureport.c          | 777 -----------------------------------------
4b6aa8
 9 files changed, 1043 insertions(+), 1035 deletions(-)
4b6aa8
 create mode 100644 src/include/ureport.h
4b6aa8
 delete mode 100644 src/lib/json.c
4b6aa8
 create mode 100644 src/lib/ureport.c
4b6aa8
 delete mode 100644 src/lib/ureport.h
4b6aa8
 create mode 100644 src/plugins/reporter-ureport.c
4b6aa8
 delete mode 100644 src/plugins/ureport.c
4b6aa8
4b6aa8
diff --git a/src/include/Makefile.am b/src/include/Makefile.am
4b6aa8
index 806b93e..de44cda 100644
4b6aa8
--- a/src/include/Makefile.am
4b6aa8
+++ b/src/include/Makefile.am
4b6aa8
@@ -15,3 +15,7 @@ libreport_include_HEADERS = \
4b6aa8
     internal_libreport.h \
4b6aa8
     internal_abrt_dbus.h \
4b6aa8
     xml_parser.h
4b6aa8
+
4b6aa8
+if BUILD_UREPORT
4b6aa8
+libreport_include_HEADERS += ureport.h
4b6aa8
+endif
4b6aa8
diff --git a/src/include/ureport.h b/src/include/ureport.h
4b6aa8
new file mode 100644
4b6aa8
index 0000000..d341f6e
4b6aa8
--- /dev/null
4b6aa8
+++ b/src/include/ureport.h
4b6aa8
@@ -0,0 +1,76 @@
4b6aa8
+/*
4b6aa8
+    Copyright (C) 2012  ABRT team
4b6aa8
+    Copyright (C) 2012  RedHat Inc
4b6aa8
+
4b6aa8
+    This program is free software; you can redistribute it and/or modify
4b6aa8
+    it under the terms of the GNU General Public License as published by
4b6aa8
+    the Free Software Foundation; either version 2 of the License, or
4b6aa8
+    (at your option) any later version.
4b6aa8
+
4b6aa8
+    This program is distributed in the hope that it will be useful,
4b6aa8
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
4b6aa8
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
4b6aa8
+    GNU General Public License for more details.
4b6aa8
+
4b6aa8
+    You should have received a copy of the GNU General Public License along
4b6aa8
+    with this program; if not, write to the Free Software Foundation, Inc.,
4b6aa8
+    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
4b6aa8
+*/
4b6aa8
+#ifndef UREPORT_H_
4b6aa8
+#define UREPORT_H_
4b6aa8
+
4b6aa8
+#include "problem_data.h"
4b6aa8
+
4b6aa8
+#ifdef __cplusplus
4b6aa8
+extern "C" {
4b6aa8
+#endif
4b6aa8
+
4b6aa8
+/*
4b6aa8
+ * uReport generation configuration
4b6aa8
+ */
4b6aa8
+struct ureport_preferences
4b6aa8
+{
4b6aa8
+    GList *urp_auth_items;  ///< list of file names included in 'auth' key
4b6aa8
+};
4b6aa8
+
4b6aa8
+/*
4b6aa8
+ * uReport server configuration
4b6aa8
+ */
4b6aa8
+struct ureport_server_config
4b6aa8
+{
4b6aa8
+    const char *ur_url;   ///< Web service URL
4b6aa8
+    bool ur_ssl_verify;   ///< Verify HOST and PEER certificates
4b6aa8
+    char *ur_client_cert; ///< Path to certificate used for client
4b6aa8
+                          ///< authentication (or NULL)
4b6aa8
+    char *ur_client_key;  ///< Private key for the certificate
4b6aa8
+    map_string_t *ur_http_headers; ///< Additional HTTP headers
4b6aa8
+
4b6aa8
+    struct ureport_preferences ur_prefs; ///< configuration for uReport generation
4b6aa8
+};
4b6aa8
+
4b6aa8
+struct abrt_post_state;
4b6aa8
+
4b6aa8
+#define ureport_post libreport_ureport_post
4b6aa8
+struct post_state *ureport_post(const char *json_ureport,
4b6aa8
+                                struct ureport_server_config *config);
4b6aa8
+
4b6aa8
+#define ureport_attach_rhbz libreport_ureport_attach_rhbz
4b6aa8
+struct post_state *ureport_attach_rhbz(const char *bthash, int rhbz_bug_id,
4b6aa8
+                                       struct ureport_server_config *config);
4b6aa8
+
4b6aa8
+#define ureport_attach_email libreport_ureport_attach_email
4b6aa8
+struct post_state *ureport_attach_email(const char *bthash, const char *email,
4b6aa8
+                                        struct ureport_server_config *config);
4b6aa8
+
4b6aa8
+#define ureport_from_dump_dir libreport_ureport_from_dump_dir
4b6aa8
+char *ureport_from_dump_dir(const char *dump_dir_path);
4b6aa8
+
4b6aa8
+#define ureport_from_dump_dir_ext libreport_ureport_from_dump_dir_ext
4b6aa8
+char *ureport_from_dump_dir_ext(const char *dump_dir_path,
4b6aa8
+                                const struct ureport_preferences *preferences);
4b6aa8
+
4b6aa8
+#ifdef __cplusplus
4b6aa8
+}
4b6aa8
+#endif
4b6aa8
+
4b6aa8
+#endif
4b6aa8
diff --git a/src/lib/Makefile.am b/src/lib/Makefile.am
4b6aa8
index d2ff675..7d9722a 100644
4b6aa8
--- a/src/lib/Makefile.am
4b6aa8
+++ b/src/lib/Makefile.am
4b6aa8
@@ -72,6 +72,7 @@ libreport_la_CPPFLAGS = \
4b6aa8
     -DDEFAULT_DUMP_DIR_MODE=$(DEFAULT_DUMP_DIR_MODE) \
4b6aa8
     -DDUMP_DIR_OWNED_BY_USER=$(DUMP_DIR_OWNED_BY_USER) \
4b6aa8
     -DLARGE_DATA_TMP_DIR=\"$(LARGE_DATA_TMP_DIR)\" \
4b6aa8
+    $(JSON_C_CFLAGS) \
4b6aa8
     $(GLIB_CFLAGS) \
4b6aa8
     $(GOBJECT_CFLAGS) \
4b6aa8
     $(AUGEAS_CFLAGS) \
4b6aa8
@@ -79,6 +80,7 @@ libreport_la_CPPFLAGS = \
4b6aa8
 libreport_la_LDFLAGS = \
4b6aa8
     -version-info 0:1:0
4b6aa8
 libreport_la_LIBADD = \
4b6aa8
+    $(JSON_C_LIBS) \
4b6aa8
     $(GLIB_LIBS) \
4b6aa8
     $(JOURNAL_LIBS) \
4b6aa8
     $(GOBJECT_LIBS) \
4b6aa8
@@ -113,7 +115,7 @@ libreport_web_o += abrt_xmlrpc.h abrt_xmlrpc.c
4b6aa8
 endif
4b6aa8
 
4b6aa8
 if BUILD_UREPORT
4b6aa8
-libreport_web_o += ureport.h json.c
4b6aa8
+libreport_web_o += ureport.c
4b6aa8
 endif
4b6aa8
 
4b6aa8
 libreport_web_la_SOURCES = $(libreport_web_o) \
4b6aa8
diff --git a/src/lib/json.c b/src/lib/json.c
4b6aa8
deleted file mode 100644
4b6aa8
index 6fbbf39..0000000
4b6aa8
--- a/src/lib/json.c
4b6aa8
+++ /dev/null
4b6aa8
@@ -1,180 +0,0 @@
4b6aa8
-/*
4b6aa8
-    Copyright (C) 2012  ABRT team
4b6aa8
-    Copyright (C) 2012  RedHat Inc
4b6aa8
-
4b6aa8
-    This program is free software; you can redistribute it and/or modify
4b6aa8
-    it under the terms of the GNU General Public License as published by
4b6aa8
-    the Free Software Foundation; either version 2 of the License, or
4b6aa8
-    (at your option) any later version.
4b6aa8
-
4b6aa8
-    This program is distributed in the hope that it will be useful,
4b6aa8
-    but WITHOUT ANY WARRANTY; without even the implied warranty of
4b6aa8
-    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
4b6aa8
-    GNU General Public License for more details.
4b6aa8
-
4b6aa8
-    You should have received a copy of the GNU General Public License along
4b6aa8
-    with this program; if not, write to the Free Software Foundation, Inc.,
4b6aa8
-    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
4b6aa8
-*/
4b6aa8
-
4b6aa8
-#include <json/json.h>
4b6aa8
-
4b6aa8
-#include <satyr/abrt.h>
4b6aa8
-#include <satyr/report.h>
4b6aa8
-
4b6aa8
-#include "internal_libreport.h"
4b6aa8
-#include "ureport.h"
4b6aa8
-#include "libreport_curl.h"
4b6aa8
-
4b6aa8
-
4b6aa8
-static void ureport_add_str(struct json_object *ur, const char *key,
4b6aa8
-                            const char *s)
4b6aa8
-{
4b6aa8
-    struct json_object *jstring = json_object_new_string(s);
4b6aa8
-    if (!jstring)
4b6aa8
-        die_out_of_memory();
4b6aa8
-
4b6aa8
-    json_object_object_add(ur, key, jstring);
4b6aa8
-}
4b6aa8
-
4b6aa8
-char *ureport_from_dump_dir_ext(const char *dump_dir_path, const struct ureport_preferences *preferences)
4b6aa8
-{
4b6aa8
-    char *error_message;
4b6aa8
-    struct sr_report *report = sr_abrt_report_from_dir(dump_dir_path,
4b6aa8
-                                                       &error_message);
4b6aa8
-
4b6aa8
-    if (!report)
4b6aa8
-        error_msg_and_die("%s", error_message);
4b6aa8
-
4b6aa8
-    if (preferences != NULL && preferences->urp_auth_items != NULL)
4b6aa8
-    {
4b6aa8
-        struct dump_dir *dd = dd_opendir(dump_dir_path, DD_OPEN_READONLY);
4b6aa8
-        if (!dd)
4b6aa8
-            xfunc_die(); /* dd_opendir() already printed an error message */
4b6aa8
-
4b6aa8
-        GList *iter = preferences->urp_auth_items;
4b6aa8
-        for ( ; iter != NULL; iter = g_list_next(iter))
4b6aa8
-        {
4b6aa8
-            const char *key = (const char *)iter->data;
4b6aa8
-            char *value = dd_load_text_ext(dd, key,
4b6aa8
-                    DD_LOAD_TEXT_RETURN_NULL_ON_FAILURE | DD_FAIL_QUIETLY_ENOENT);
4b6aa8
-
4b6aa8
-            if (value == NULL)
4b6aa8
-            {
4b6aa8
-                perror_msg("Cannot include '%s' in 'auth'", key);
4b6aa8
-                continue;
4b6aa8
-            }
4b6aa8
-
4b6aa8
-            sr_report_add_auth(report, key, value);
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        dd_close(dd);
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    char *json_ureport = sr_report_to_json(report);
4b6aa8
-    sr_report_free(report);
4b6aa8
-
4b6aa8
-    return json_ureport;
4b6aa8
-}
4b6aa8
-
4b6aa8
-char *ureport_from_dump_dir(const char *dump_dir_path)
4b6aa8
-{
4b6aa8
-    return ureport_from_dump_dir_ext(dump_dir_path, /*no preferences*/NULL);
4b6aa8
-}
4b6aa8
-
4b6aa8
-char *new_json_attachment(const char *bthash, const char *type, const char *data)
4b6aa8
-{
4b6aa8
-    struct json_object *attachment = json_object_new_object();
4b6aa8
-    if (!attachment)
4b6aa8
-        die_out_of_memory();
4b6aa8
-
4b6aa8
-    ureport_add_str(attachment, "bthash", bthash);
4b6aa8
-    ureport_add_str(attachment, "type", type);
4b6aa8
-    ureport_add_str(attachment, "data", data);
4b6aa8
-
4b6aa8
-    char *result = xstrdup(json_object_to_json_string(attachment));
4b6aa8
-    json_object_put(attachment);
4b6aa8
-
4b6aa8
-    return result;
4b6aa8
-}
4b6aa8
-
4b6aa8
-struct post_state *post_ureport(const char *json, struct ureport_server_config *config)
4b6aa8
-{
4b6aa8
-    int flags = POST_WANT_BODY | POST_WANT_ERROR_MSG;
4b6aa8
-
4b6aa8
-    if (config->ur_ssl_verify)
4b6aa8
-        flags |= POST_WANT_SSL_VERIFY;
4b6aa8
-
4b6aa8
-    struct post_state *post_state = new_post_state(flags);
4b6aa8
-
4b6aa8
-    if (config->ur_client_cert && config->ur_client_key)
4b6aa8
-    {
4b6aa8
-        post_state->client_cert_path = config->ur_client_cert;
4b6aa8
-        post_state->client_key_path = config->ur_client_key;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    char **headers = xmalloc(sizeof(char *) * (3 + size_map_string(config->ur_http_headers)));
4b6aa8
-    headers[0] = (char *)"Accept: application/json";
4b6aa8
-    headers[1] = (char *)"Connection: close";
4b6aa8
-    headers[2] = NULL;
4b6aa8
-
4b6aa8
-    if (config->ur_http_headers != NULL)
4b6aa8
-    {
4b6aa8
-        unsigned i = 2;
4b6aa8
-        const char *header;
4b6aa8
-        const char *value;
4b6aa8
-        map_string_iter_t iter;
4b6aa8
-        init_map_string_iter(&iter, config->ur_http_headers);
4b6aa8
-        while (next_map_string_iter(&iter, &header, &value))
4b6aa8
-            headers[i++] = xasprintf("%s: %s", header, value);
4b6aa8
-        headers[i] = NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    post_string_as_form_data(post_state, config->ur_url, "application/json",
4b6aa8
-                    (const char **)headers, json);
4b6aa8
-
4b6aa8
-    /* Client authentication failed. Try again without client auth.
4b6aa8
-     * CURLE_SSL_CONNECT_ERROR - cert not found/server doesnt trust the CA
4b6aa8
-     * CURLE_SSL_CERTPROBLEM - malformed certificate/no permission
4b6aa8
-     */
4b6aa8
-    if ((post_state->curl_result == CURLE_SSL_CONNECT_ERROR
4b6aa8
-         || post_state->curl_result == CURLE_SSL_CERTPROBLEM)
4b6aa8
-            && config->ur_client_cert && config->ur_client_key)
4b6aa8
-    {
4b6aa8
-        warn_msg("Authentication failed. Retrying unauthenticated.");
4b6aa8
-        free_post_state(post_state);
4b6aa8
-        post_state = new_post_state(flags);
4b6aa8
-
4b6aa8
-        post_string_as_form_data(post_state, config->ur_url, "application/json",
4b6aa8
-                         (const char **)headers, json);
4b6aa8
-
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    for (unsigned i = size_map_string(config->ur_http_headers); i != 0; --i)
4b6aa8
-        free(headers[i + 1]);
4b6aa8
-    free(headers);
4b6aa8
-
4b6aa8
-    return post_state;
4b6aa8
-}
4b6aa8
-
4b6aa8
-struct post_state *ureport_attach_rhbz(const char *bthash, int rhbz_bug_id,
4b6aa8
-                                       struct ureport_server_config *config)
4b6aa8
-{
4b6aa8
-    char *str_bug_id = xasprintf("%d", rhbz_bug_id);
4b6aa8
-    char *json_attachment = new_json_attachment(bthash, "RHBZ", str_bug_id);
4b6aa8
-    struct post_state *post_state = post_ureport(json_attachment, config);
4b6aa8
-    free(str_bug_id);
4b6aa8
-    free(json_attachment);
4b6aa8
-
4b6aa8
-    return post_state;
4b6aa8
-}
4b6aa8
-
4b6aa8
-struct post_state *ureport_attach_email(const char *bthash, const char *email,
4b6aa8
-                                       struct ureport_server_config *config)
4b6aa8
-{
4b6aa8
-    char *json_attachment = new_json_attachment(bthash, "email", email);
4b6aa8
-    struct post_state *post_state = post_ureport(json_attachment, config);
4b6aa8
-    free(json_attachment);
4b6aa8
-
4b6aa8
-    return post_state;
4b6aa8
-}
4b6aa8
diff --git a/src/lib/ureport.c b/src/lib/ureport.c
4b6aa8
new file mode 100644
4b6aa8
index 0000000..761fe62
4b6aa8
--- /dev/null
4b6aa8
+++ b/src/lib/ureport.c
4b6aa8
@@ -0,0 +1,180 @@
4b6aa8
+/*
4b6aa8
+    Copyright (C) 2012,2014  ABRT team
4b6aa8
+    Copyright (C) 2012,2014  RedHat Inc
4b6aa8
+
4b6aa8
+    This program is free software; you can redistribute it and/or modify
4b6aa8
+    it under the terms of the GNU General Public License as published by
4b6aa8
+    the Free Software Foundation; either version 2 of the License, or
4b6aa8
+    (at your option) any later version.
4b6aa8
+
4b6aa8
+    This program is distributed in the hope that it will be useful,
4b6aa8
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
4b6aa8
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
4b6aa8
+    GNU General Public License for more details.
4b6aa8
+
4b6aa8
+    You should have received a copy of the GNU General Public License along
4b6aa8
+    with this program; if not, write to the Free Software Foundation, Inc.,
4b6aa8
+    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
4b6aa8
+*/
4b6aa8
+
4b6aa8
+#include <json.h>
4b6aa8
+
4b6aa8
+#include <satyr/abrt.h>
4b6aa8
+#include <satyr/report.h>
4b6aa8
+
4b6aa8
+#include "internal_libreport.h"
4b6aa8
+#include "ureport.h"
4b6aa8
+#include "libreport_curl.h"
4b6aa8
+
4b6aa8
+
4b6aa8
+static void ureport_add_str(struct json_object *ur, const char *key,
4b6aa8
+                            const char *s)
4b6aa8
+{
4b6aa8
+    struct json_object *jstring = json_object_new_string(s);
4b6aa8
+    if (!jstring)
4b6aa8
+        die_out_of_memory();
4b6aa8
+
4b6aa8
+    json_object_object_add(ur, key, jstring);
4b6aa8
+}
4b6aa8
+
4b6aa8
+char *ureport_from_dump_dir_ext(const char *dump_dir_path, const struct ureport_preferences *preferences)
4b6aa8
+{
4b6aa8
+    char *error_message;
4b6aa8
+    struct sr_report *report = sr_abrt_report_from_dir(dump_dir_path,
4b6aa8
+                                                       &error_message);
4b6aa8
+
4b6aa8
+    if (!report)
4b6aa8
+        error_msg_and_die("%s", error_message);
4b6aa8
+
4b6aa8
+    if (preferences != NULL && preferences->urp_auth_items != NULL)
4b6aa8
+    {
4b6aa8
+        struct dump_dir *dd = dd_opendir(dump_dir_path, DD_OPEN_READONLY);
4b6aa8
+        if (!dd)
4b6aa8
+            xfunc_die(); /* dd_opendir() already printed an error message */
4b6aa8
+
4b6aa8
+        GList *iter = preferences->urp_auth_items;
4b6aa8
+        for ( ; iter != NULL; iter = g_list_next(iter))
4b6aa8
+        {
4b6aa8
+            const char *key = (const char *)iter->data;
4b6aa8
+            char *value = dd_load_text_ext(dd, key,
4b6aa8
+                    DD_LOAD_TEXT_RETURN_NULL_ON_FAILURE | DD_FAIL_QUIETLY_ENOENT);
4b6aa8
+
4b6aa8
+            if (value == NULL)
4b6aa8
+            {
4b6aa8
+                perror_msg("Cannot include '%s' in 'auth'", key);
4b6aa8
+                continue;
4b6aa8
+            }
4b6aa8
+
4b6aa8
+            sr_report_add_auth(report, key, value);
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        dd_close(dd);
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    char *json_ureport = sr_report_to_json(report);
4b6aa8
+    sr_report_free(report);
4b6aa8
+
4b6aa8
+    return json_ureport;
4b6aa8
+}
4b6aa8
+
4b6aa8
+char *ureport_from_dump_dir(const char *dump_dir_path)
4b6aa8
+{
4b6aa8
+    return ureport_from_dump_dir_ext(dump_dir_path, /*no preferences*/NULL);
4b6aa8
+}
4b6aa8
+
4b6aa8
+static char *new_json_attachment(const char *bthash, const char *type, const char *data)
4b6aa8
+{
4b6aa8
+    struct json_object *attachment = json_object_new_object();
4b6aa8
+    if (!attachment)
4b6aa8
+        die_out_of_memory();
4b6aa8
+
4b6aa8
+    ureport_add_str(attachment, "bthash", bthash);
4b6aa8
+    ureport_add_str(attachment, "type", type);
4b6aa8
+    ureport_add_str(attachment, "data", data);
4b6aa8
+
4b6aa8
+    char *result = xstrdup(json_object_to_json_string(attachment));
4b6aa8
+    json_object_put(attachment);
4b6aa8
+
4b6aa8
+    return result;
4b6aa8
+}
4b6aa8
+
4b6aa8
+struct post_state *ureport_post(const char *json, struct ureport_server_config *config)
4b6aa8
+{
4b6aa8
+    int flags = POST_WANT_BODY | POST_WANT_ERROR_MSG;
4b6aa8
+
4b6aa8
+    if (config->ur_ssl_verify)
4b6aa8
+        flags |= POST_WANT_SSL_VERIFY;
4b6aa8
+
4b6aa8
+    struct post_state *post_state = new_post_state(flags);
4b6aa8
+
4b6aa8
+    if (config->ur_client_cert && config->ur_client_key)
4b6aa8
+    {
4b6aa8
+        post_state->client_cert_path = config->ur_client_cert;
4b6aa8
+        post_state->client_key_path = config->ur_client_key;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    char **headers = xmalloc(sizeof(char *) * (3 + size_map_string(config->ur_http_headers)));
4b6aa8
+    headers[0] = (char *)"Accept: application/json";
4b6aa8
+    headers[1] = (char *)"Connection: close";
4b6aa8
+    headers[2] = NULL;
4b6aa8
+
4b6aa8
+    if (config->ur_http_headers != NULL)
4b6aa8
+    {
4b6aa8
+        unsigned i = 2;
4b6aa8
+        const char *header;
4b6aa8
+        const char *value;
4b6aa8
+        map_string_iter_t iter;
4b6aa8
+        init_map_string_iter(&iter, config->ur_http_headers);
4b6aa8
+        while (next_map_string_iter(&iter, &header, &value))
4b6aa8
+            headers[i++] = xasprintf("%s: %s", header, value);
4b6aa8
+        headers[i] = NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    post_string_as_form_data(post_state, config->ur_url, "application/json",
4b6aa8
+                    (const char **)headers, json);
4b6aa8
+
4b6aa8
+    /* Client authentication failed. Try again without client auth.
4b6aa8
+     * CURLE_SSL_CONNECT_ERROR - cert not found/server doesnt trust the CA
4b6aa8
+     * CURLE_SSL_CERTPROBLEM - malformed certificate/no permission
4b6aa8
+     */
4b6aa8
+    if ((post_state->curl_result == CURLE_SSL_CONNECT_ERROR
4b6aa8
+         || post_state->curl_result == CURLE_SSL_CERTPROBLEM)
4b6aa8
+            && config->ur_client_cert && config->ur_client_key)
4b6aa8
+    {
4b6aa8
+        warn_msg("Authentication failed. Retrying unauthenticated.");
4b6aa8
+        free_post_state(post_state);
4b6aa8
+        post_state = new_post_state(flags);
4b6aa8
+
4b6aa8
+        post_string_as_form_data(post_state, config->ur_url, "application/json",
4b6aa8
+                         (const char **)headers, json);
4b6aa8
+
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    for (unsigned i = size_map_string(config->ur_http_headers); i != 0; --i)
4b6aa8
+        free(headers[i + 1]);
4b6aa8
+    free(headers);
4b6aa8
+
4b6aa8
+    return post_state;
4b6aa8
+}
4b6aa8
+
4b6aa8
+struct post_state *ureport_attach_rhbz(const char *bthash, int rhbz_bug_id,
4b6aa8
+                                       struct ureport_server_config *config)
4b6aa8
+{
4b6aa8
+    char *str_bug_id = xasprintf("%d", rhbz_bug_id);
4b6aa8
+    char *json_attachment = new_json_attachment(bthash, "RHBZ", str_bug_id);
4b6aa8
+    struct post_state *post_state = ureport_post(json_attachment, config);
4b6aa8
+    free(str_bug_id);
4b6aa8
+    free(json_attachment);
4b6aa8
+
4b6aa8
+    return post_state;
4b6aa8
+}
4b6aa8
+
4b6aa8
+struct post_state *ureport_attach_email(const char *bthash, const char *email,
4b6aa8
+                                       struct ureport_server_config *config)
4b6aa8
+{
4b6aa8
+    char *json_attachment = new_json_attachment(bthash, "email", email);
4b6aa8
+    struct post_state *post_state = ureport_post(json_attachment, config);
4b6aa8
+    free(json_attachment);
4b6aa8
+
4b6aa8
+    return post_state;
4b6aa8
+}
4b6aa8
diff --git a/src/lib/ureport.h b/src/lib/ureport.h
4b6aa8
deleted file mode 100644
4b6aa8
index 319aca9..0000000
4b6aa8
--- a/src/lib/ureport.h
4b6aa8
+++ /dev/null
4b6aa8
@@ -1,76 +0,0 @@
4b6aa8
-/*
4b6aa8
-    Copyright (C) 2012  ABRT team
4b6aa8
-    Copyright (C) 2012  RedHat Inc
4b6aa8
-
4b6aa8
-    This program is free software; you can redistribute it and/or modify
4b6aa8
-    it under the terms of the GNU General Public License as published by
4b6aa8
-    the Free Software Foundation; either version 2 of the License, or
4b6aa8
-    (at your option) any later version.
4b6aa8
-
4b6aa8
-    This program is distributed in the hope that it will be useful,
4b6aa8
-    but WITHOUT ANY WARRANTY; without even the implied warranty of
4b6aa8
-    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
4b6aa8
-    GNU General Public License for more details.
4b6aa8
-
4b6aa8
-    You should have received a copy of the GNU General Public License along
4b6aa8
-    with this program; if not, write to the Free Software Foundation, Inc.,
4b6aa8
-    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
4b6aa8
-*/
4b6aa8
-#ifndef UREPORT_H_
4b6aa8
-#define UREPORT_H_
4b6aa8
-
4b6aa8
-#include "problem_data.h"
4b6aa8
-
4b6aa8
-#ifdef __cplusplus
4b6aa8
-extern "C" {
4b6aa8
-#endif
4b6aa8
-
4b6aa8
-/*
4b6aa8
- * uReport generation configuration
4b6aa8
- */
4b6aa8
-struct ureport_preferences
4b6aa8
-{
4b6aa8
-    GList *urp_auth_items;  ///< list of file names included in 'auth' key
4b6aa8
-};
4b6aa8
-
4b6aa8
-/*
4b6aa8
- * uReport server configuration
4b6aa8
- */
4b6aa8
-struct ureport_server_config
4b6aa8
-{
4b6aa8
-    const char *ur_url;   ///< Web service URL
4b6aa8
-    bool ur_ssl_verify;   ///< Verify HOST and PEER certificates
4b6aa8
-    char *ur_client_cert; ///< Path to certificate used for client
4b6aa8
-                          ///< authentication (or NULL)
4b6aa8
-    char *ur_client_key;  ///< Private key for the certificate
4b6aa8
-    map_string_t *ur_http_headers; ///< Additional HTTP headers
4b6aa8
-
4b6aa8
-    struct ureport_preferences ur_prefs; ///< configuration for uReport generation
4b6aa8
-};
4b6aa8
-
4b6aa8
-struct abrt_post_state;
4b6aa8
-
4b6aa8
-#define post_ureport libreport_post_ureport
4b6aa8
-struct post_state *post_ureport(const char *json_ureport,
4b6aa8
-                                struct ureport_server_config *config);
4b6aa8
-
4b6aa8
-#define ureport_attach_rhbz libreport_ureport_attach_rhbz
4b6aa8
-struct post_state *ureport_attach_rhbz(const char *bthash, int rhbz_bug_id,
4b6aa8
-                                       struct ureport_server_config *config);
4b6aa8
-
4b6aa8
-#define ureport_attach_email libreport_ureport_attach_email
4b6aa8
-struct post_state *ureport_attach_email(const char *bthash, const char *email,
4b6aa8
-                                        struct ureport_server_config *config);
4b6aa8
-
4b6aa8
-#define ureport_from_dump_dir libreport_ureport_from_dump_dir
4b6aa8
-char *ureport_from_dump_dir(const char *dump_dir_path);
4b6aa8
-
4b6aa8
-#define ureport_from_dump_dir_ext libreport_ureport_from_dump_dir_ext
4b6aa8
-char *ureport_from_dump_dir_ext(const char *dump_dir_path,
4b6aa8
-                                const struct ureport_preferences *preferences);
4b6aa8
-
4b6aa8
-#ifdef __cplusplus
4b6aa8
-}
4b6aa8
-#endif
4b6aa8
-
4b6aa8
-#endif
4b6aa8
diff --git a/src/plugins/Makefile.am b/src/plugins/Makefile.am
4b6aa8
index 7ec5219..7ec08d7 100644
4b6aa8
--- a/src/plugins/Makefile.am
4b6aa8
+++ b/src/plugins/Makefile.am
4b6aa8
@@ -250,15 +250,17 @@ endif
4b6aa8
 
4b6aa8
 if BUILD_UREPORT
4b6aa8
 reporter_ureport_SOURCES = \
4b6aa8
-    ureport.c
4b6aa8
+    reporter-ureport.c
4b6aa8
 reporter_ureport_CPPFLAGS = \
4b6aa8
     -I$(srcdir)/../include \
4b6aa8
     -I$(srcdir)/../lib \
4b6aa8
+    $(JSON_C_CFLAGS) \
4b6aa8
     $(GLIB_CFLAGS) \
4b6aa8
     $(LIBREPORT_CFLAGS) \
4b6aa8
     -DPLUGINS_CONF_DIR=\"$(REPORT_PLUGINS_CONF_DIR)\" \
4b6aa8
     -D_GNU_SOURCE
4b6aa8
 reporter_ureport_LDADD = \
4b6aa8
+    $(JSON_C_LIBS) \
4b6aa8
     ../lib/libreport.la \
4b6aa8
     ../lib/libreport-web.la
4b6aa8
 endif
4b6aa8
diff --git a/src/plugins/reporter-ureport.c b/src/plugins/reporter-ureport.c
4b6aa8
new file mode 100644
4b6aa8
index 0000000..d827c7d
4b6aa8
--- /dev/null
4b6aa8
+++ b/src/plugins/reporter-ureport.c
4b6aa8
@@ -0,0 +1,777 @@
4b6aa8
+/*
4b6aa8
+    Copyright (C) 2012  ABRT Team
4b6aa8
+    Copyright (C) 2012  RedHat inc.
4b6aa8
+
4b6aa8
+    This program is free software; you can redistribute it and/or modify
4b6aa8
+    it under the terms of the GNU General Public License as published by
4b6aa8
+    the Free Software Foundation; either version 2 of the License, or
4b6aa8
+    (at your option) any later version.
4b6aa8
+
4b6aa8
+    This program is distributed in the hope that it will be useful,
4b6aa8
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
4b6aa8
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
4b6aa8
+    GNU General Public License for more details.
4b6aa8
+
4b6aa8
+    You should have received a copy of the GNU General Public License along
4b6aa8
+    with this program; if not, write to the Free Software Foundation, Inc.,
4b6aa8
+    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
4b6aa8
+*/
4b6aa8
+
4b6aa8
+#include <json.h>
4b6aa8
+#include "internal_libreport.h"
4b6aa8
+#include "ureport.h"
4b6aa8
+#include "libreport_curl.h"
4b6aa8
+
4b6aa8
+#define CONF_FILE_PATH PLUGINS_CONF_DIR"/ureport.conf"
4b6aa8
+
4b6aa8
+#define REPORT_URL_SFX "reports/new/"
4b6aa8
+#define ATTACH_URL_SFX "reports/attach/"
4b6aa8
+#define BTHASH_URL_SFX "reports/bthash/"
4b6aa8
+
4b6aa8
+#define RHSM_CERT_PATH "/etc/pki/consumer/cert.pem"
4b6aa8
+#define RHSM_KEY_PATH "/etc/pki/consumer/key.pem"
4b6aa8
+
4b6aa8
+#define RHAP_PEM_DIR_PATH "/etc/pki/entitlement"
4b6aa8
+#define RHAP_ENT_DATA_BEGIN_TAG "-----BEGIN ENTITLEMENT DATA-----"
4b6aa8
+#define RHAP_ENT_DATA_END_TAG "-----END ENTITLEMENT DATA-----"
4b6aa8
+#define RHAP_SIG_DATA_BEGIN_TAG "-----BEGIN RSA SIGNATURE-----"
4b6aa8
+#define RHAP_SIG_DATA_END_TAG "-----END RSA SIGNATURE-----"
4b6aa8
+
4b6aa8
+#define VALUE_FROM_CONF(opt, var, tr) do { const char *value = getenv("uReport_"opt); \
4b6aa8
+        if (!value) { value = get_map_string_item_or_NULL(settings, opt); } if (value) { var = tr(value); } \
4b6aa8
+    } while(0)
4b6aa8
+
4b6aa8
+static char *puppet_config_print(const char *key)
4b6aa8
+{
4b6aa8
+    char *command = xasprintf("puppet config print %s", key);
4b6aa8
+    char *result = run_in_shell_and_save_output(0, command, NULL, NULL);
4b6aa8
+    free(command);
4b6aa8
+
4b6aa8
+    /* run_in_shell_and_save_output always returns non-NULL */
4b6aa8
+    if (result[0] != '/')
4b6aa8
+        goto error;
4b6aa8
+
4b6aa8
+    char *newline = strchrnul(result, '\n');
4b6aa8
+    if (!newline)
4b6aa8
+        goto error;
4b6aa8
+
4b6aa8
+    *newline = '\0';
4b6aa8
+    return result;
4b6aa8
+error:
4b6aa8
+    free(result);
4b6aa8
+    error_msg_and_die("Unable to determine puppet %s path (puppet not installed?)", key);
4b6aa8
+}
4b6aa8
+
4b6aa8
+static void parse_client_auth_paths(struct ureport_server_config *config, const char *client_auth)
4b6aa8
+{
4b6aa8
+    if (client_auth == NULL)
4b6aa8
+        return;
4b6aa8
+
4b6aa8
+    if (strcmp(client_auth, "") == 0)
4b6aa8
+    {
4b6aa8
+        config->ur_client_cert = NULL;
4b6aa8
+        config->ur_client_key = NULL;
4b6aa8
+        log_notice("Not using client authentication");
4b6aa8
+    }
4b6aa8
+    else if (strcmp(client_auth, "rhsm") == 0)
4b6aa8
+    {
4b6aa8
+        config->ur_client_cert = xstrdup(RHSM_CERT_PATH);
4b6aa8
+        config->ur_client_key = xstrdup(RHSM_KEY_PATH);
4b6aa8
+    }
4b6aa8
+    else if (strcmp(client_auth, "rhsm-entitlement") == 0)
4b6aa8
+    {
4b6aa8
+        GList *certs = get_file_list(RHAP_PEM_DIR_PATH, "pem");
4b6aa8
+        if (g_list_length(certs) != 2)
4b6aa8
+        {
4b6aa8
+            log_notice(RHAP_PEM_DIR_PATH" does not contain unique cert-key files pair");
4b6aa8
+            log_notice("Not using client authentication");
4b6aa8
+            return;
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        const char *cert = NULL;
4b6aa8
+        const char *key = NULL;
4b6aa8
+
4b6aa8
+        file_obj_t *fst = (file_obj_t *)certs->data;
4b6aa8
+        file_obj_t *scn = (file_obj_t *)certs->next->data;
4b6aa8
+
4b6aa8
+        if (strlen(fo_get_filename(fst)) < strlen(fo_get_filename(scn)))
4b6aa8
+        {
4b6aa8
+            cert = fo_get_filename(fst);
4b6aa8
+            key = fo_get_filename(scn);
4b6aa8
+
4b6aa8
+            config->ur_client_cert = xstrdup(fo_get_fullpath(fst));
4b6aa8
+            config->ur_client_key = xstrdup(fo_get_fullpath(scn));
4b6aa8
+        }
4b6aa8
+        else
4b6aa8
+        {
4b6aa8
+            cert = fo_get_filename(scn);
4b6aa8
+            key = fo_get_filename(fst);
4b6aa8
+
4b6aa8
+            config->ur_client_cert = xstrdup(fo_get_fullpath(scn));
4b6aa8
+            config->ur_client_key = xstrdup(fo_get_fullpath(fst));
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        const bool iscomplement = prefixcmp(key, cert) != 0 || strcmp("-key", key + strlen(cert)) != 0;
4b6aa8
+        g_list_free_full(certs, (GDestroyNotify)free_file_obj);
4b6aa8
+
4b6aa8
+        if (iscomplement)
4b6aa8
+        {
4b6aa8
+            log_notice("Key file '%s' isn't complement to cert file '%s'",
4b6aa8
+                    config->ur_client_key, config->ur_client_cert);
4b6aa8
+            log_notice("Not using client authentication");
4b6aa8
+
4b6aa8
+            free(config->ur_client_cert);
4b6aa8
+            free(config->ur_client_key);
4b6aa8
+            config->ur_client_cert = NULL;
4b6aa8
+            config->ur_client_key = NULL;
4b6aa8
+
4b6aa8
+            return;
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        char *certdata = xmalloc_open_read_close(config->ur_client_cert, /*no size limit*/NULL);
4b6aa8
+        if (certdata != NULL)
4b6aa8
+        {
4b6aa8
+            char *ent_data = xstrdup_between(certdata,
4b6aa8
+                    RHAP_ENT_DATA_BEGIN_TAG, RHAP_ENT_DATA_END_TAG);
4b6aa8
+
4b6aa8
+            char *sig_data = xstrdup_between(certdata,
4b6aa8
+                    RHAP_SIG_DATA_BEGIN_TAG, RHAP_SIG_DATA_END_TAG);
4b6aa8
+
4b6aa8
+            if (ent_data != NULL && sig_data != NULL)
4b6aa8
+            {
4b6aa8
+                ent_data = strremovech(ent_data, '\n');
4b6aa8
+                insert_map_string(config->ur_http_headers,
4b6aa8
+                        xstrdup("X-RH-Entitlement-Data"),
4b6aa8
+                        xasprintf(RHAP_ENT_DATA_BEGIN_TAG"%s"RHAP_ENT_DATA_END_TAG, ent_data));
4b6aa8
+
4b6aa8
+                sig_data = strremovech(sig_data, '\n');
4b6aa8
+                insert_map_string(config->ur_http_headers,
4b6aa8
+                        xstrdup("X-RH-Entitlement-Sig"),
4b6aa8
+                        xasprintf(RHAP_SIG_DATA_BEGIN_TAG"%s"RHAP_SIG_DATA_END_TAG, sig_data));
4b6aa8
+            }
4b6aa8
+            else
4b6aa8
+            {
4b6aa8
+                log_notice("Cert file '%s' doesn't contain Entitlement and RSA Signature sections", config->ur_client_cert);
4b6aa8
+                log_notice("Not using HTTP authentication headers");
4b6aa8
+            }
4b6aa8
+
4b6aa8
+            free(sig_data);
4b6aa8
+            free(ent_data);
4b6aa8
+            free(certdata);
4b6aa8
+        }
4b6aa8
+    }
4b6aa8
+    else if (strcmp(client_auth, "puppet") == 0)
4b6aa8
+    {
4b6aa8
+        config->ur_client_cert = puppet_config_print("hostcert");
4b6aa8
+        config->ur_client_key = puppet_config_print("hostprivkey");
4b6aa8
+    }
4b6aa8
+    else
4b6aa8
+    {
4b6aa8
+        char *scratch = xstrdup(client_auth);
4b6aa8
+        config->ur_client_cert = xstrdup(strtok(scratch, ":"));
4b6aa8
+        config->ur_client_key = xstrdup(strtok(NULL, ":"));
4b6aa8
+        free(scratch);
4b6aa8
+        if (config->ur_client_cert == NULL || config->ur_client_key == NULL)
4b6aa8
+            error_msg_and_die("Invalid client authentication specification");
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (config->ur_client_cert && config->ur_client_key)
4b6aa8
+    {
4b6aa8
+        log_notice("Using client certificate: %s", config->ur_client_cert);
4b6aa8
+        log_notice("Using client private key: %s", config->ur_client_key);
4b6aa8
+    }
4b6aa8
+}
4b6aa8
+
4b6aa8
+/*
4b6aa8
+ * Loads uReport configuration from various sources.
4b6aa8
+ *
4b6aa8
+ * Replaces a value of an already configured option only if the
4b6aa8
+ * option was found in a configuration source.
4b6aa8
+ *
4b6aa8
+ * @param config a server configuration to be populated
4b6aa8
+ */
4b6aa8
+static void load_ureport_server_config(struct ureport_server_config *config, map_string_t *settings)
4b6aa8
+{
4b6aa8
+    VALUE_FROM_CONF("URL", config->ur_url, (const char *));
4b6aa8
+    VALUE_FROM_CONF("SSLVerify", config->ur_ssl_verify, string_to_bool);
4b6aa8
+
4b6aa8
+    bool include_auth = false;
4b6aa8
+    VALUE_FROM_CONF("IncludeAuthData", include_auth, string_to_bool);
4b6aa8
+
4b6aa8
+    if (include_auth)
4b6aa8
+    {
4b6aa8
+        const char *auth_items = NULL;
4b6aa8
+        VALUE_FROM_CONF("AuthDataItems", auth_items, (const char *));
4b6aa8
+        config->ur_prefs.urp_auth_items = parse_list(auth_items);
4b6aa8
+
4b6aa8
+        if (config->ur_prefs.urp_auth_items == NULL)
4b6aa8
+            log_warning("IncludeAuthData set to 'yes' but AuthDataItems is empty.");
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    const char *client_auth = NULL;
4b6aa8
+    VALUE_FROM_CONF("SSLClientAuth", client_auth, (const char *));
4b6aa8
+    parse_client_auth_paths(config, client_auth);
4b6aa8
+}
4b6aa8
+
4b6aa8
+struct ureport_server_response {
4b6aa8
+    bool is_error;
4b6aa8
+    char *value;
4b6aa8
+    char *message;
4b6aa8
+    char *bthash;
4b6aa8
+    GList *reported_to_list;
4b6aa8
+    char *solution;
4b6aa8
+};
4b6aa8
+
4b6aa8
+void free_ureport_server_response(struct ureport_server_response *resp)
4b6aa8
+{
4b6aa8
+    if (!resp)
4b6aa8
+        return;
4b6aa8
+
4b6aa8
+    free(resp->solution);
4b6aa8
+    g_list_free_full(resp->reported_to_list, g_free);
4b6aa8
+    free(resp->bthash);
4b6aa8
+    free(resp->message);
4b6aa8
+    free(resp->value);
4b6aa8
+    free(resp);
4b6aa8
+}
4b6aa8
+
4b6aa8
+static char *parse_solution_from_json_list(struct json_object *list, GList **reported_to)
4b6aa8
+{
4b6aa8
+    json_object *list_elem, *struct_elem;
4b6aa8
+    const char *cause, *note, *url;
4b6aa8
+    struct strbuf *solution_buf = strbuf_new();
4b6aa8
+
4b6aa8
+    const unsigned length = json_object_array_length(list);
4b6aa8
+
4b6aa8
+    const char *one_format = _("Your problem seems to be caused by %s\n\n%s\n");
4b6aa8
+    if (length > 1)
4b6aa8
+    {
4b6aa8
+        strbuf_append_str(solution_buf, _("Your problem seems to be caused by one of the following:\n"));
4b6aa8
+        one_format = "\n* %s\n\n%s\n";
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    bool empty = true;
4b6aa8
+    for (unsigned i = 0; i < length; ++i)
4b6aa8
+    {
4b6aa8
+        list_elem = json_object_array_get_idx(list, i);
4b6aa8
+        if (!list_elem)
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        if (!json_object_object_get_ex(list_elem, "cause", &struct_elem))
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        cause = json_object_get_string(struct_elem);
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        if (!json_object_object_get_ex(list_elem, "note", &struct_elem))
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        note = json_object_get_string(struct_elem);
4b6aa8
+        if (!note)
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        empty = false;
4b6aa8
+        strbuf_append_strf(solution_buf, one_format, cause, note);
4b6aa8
+
4b6aa8
+        if (!json_object_object_get_ex(list_elem, "url", &struct_elem))
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        url = json_object_get_string(struct_elem);
4b6aa8
+        if (url)
4b6aa8
+        {
4b6aa8
+            char *reported_to_line = xasprintf("%s: URL=%s", cause, url);
4b6aa8
+            *reported_to = g_list_append(*reported_to, reported_to_line);
4b6aa8
+        }
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (empty)
4b6aa8
+    {
4b6aa8
+        strbuf_free(solution_buf);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    return strbuf_free_nobuf(solution_buf);
4b6aa8
+}
4b6aa8
+
4b6aa8
+/* reported_to json element should be a list of structures
4b6aa8
+{ "reporter": "Bugzilla",
4b6aa8
+  "type": "url",
4b6aa8
+  "value": "https://bugzilla.redhat.com/show_bug.cgi?id=XYZ" } */
4b6aa8
+static GList *parse_reported_to_from_json_list(struct json_object *list)
4b6aa8
+{
4b6aa8
+    int i;
4b6aa8
+    json_object *list_elem, *struct_elem;
4b6aa8
+    const char *reporter, *value, *type;
4b6aa8
+    char *reported_to_line, *prefix;
4b6aa8
+    GList *result = NULL;
4b6aa8
+
4b6aa8
+    for (i = 0; i < json_object_array_length(list); ++i)
4b6aa8
+    {
4b6aa8
+        prefix = NULL;
4b6aa8
+        list_elem = json_object_array_get_idx(list, i);
4b6aa8
+        if (!list_elem)
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        if (!json_object_object_get_ex(list_elem, "reporter", &struct_elem))
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        reporter = json_object_get_string(struct_elem);
4b6aa8
+        if (!reporter)
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        if (!json_object_object_get_ex(list_elem, "value", &struct_elem))
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        value = json_object_get_string(struct_elem);
4b6aa8
+        if (!value)
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        if (!json_object_object_get_ex(list_elem, "type", &struct_elem))
4b6aa8
+            continue;
4b6aa8
+
4b6aa8
+        type = json_object_get_string(struct_elem);
4b6aa8
+        if (type)
4b6aa8
+        {
4b6aa8
+            if (strcasecmp("url", type) == 0)
4b6aa8
+                prefix = xstrdup("URL=");
4b6aa8
+            else if (strcasecmp("bthash", type) == 0)
4b6aa8
+                prefix = xstrdup("BTHASH=");
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        if (!prefix)
4b6aa8
+            prefix = xstrdup("");
4b6aa8
+
4b6aa8
+        reported_to_line = xasprintf("%s: %s%s", reporter, prefix, value);
4b6aa8
+        free(prefix);
4b6aa8
+
4b6aa8
+        result = g_list_append(result, reported_to_line);
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    return result;
4b6aa8
+}
4b6aa8
+
4b6aa8
+/*
4b6aa8
+ * Reponse samples:
4b6aa8
+ * {"error":"field 'foo' is required"}
4b6aa8
+ * {"response":"true"}
4b6aa8
+ * {"response":"false"}
4b6aa8
+ */
4b6aa8
+static struct ureport_server_response *ureport_server_parse_json(json_object *json)
4b6aa8
+{
4b6aa8
+    json_object *obj = NULL;
4b6aa8
+    if (json_object_object_get_ex(json, "error", &obj))
4b6aa8
+    {
4b6aa8
+        struct ureport_server_response *out_response = xzalloc(sizeof(*out_response));
4b6aa8
+        out_response->is_error = true;
4b6aa8
+        /*
4b6aa8
+         * Used to use json_object_to_json_string(obj), but it returns
4b6aa8
+         * the string in quote marks (") - IOW, json-formatted string.
4b6aa8
+         */
4b6aa8
+        out_response->value = xstrdup(json_object_get_string(obj));
4b6aa8
+        return out_response;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (json_object_object_get_ex(json, "result", &obj))
4b6aa8
+    {
4b6aa8
+        struct ureport_server_response *out_response = xzalloc(sizeof(*out_response));
4b6aa8
+        out_response->value = xstrdup(json_object_get_string(obj));
4b6aa8
+
4b6aa8
+        json_object *message = NULL;
4b6aa8
+        if (json_object_object_get_ex(json, "message", &message))
4b6aa8
+            out_response->message = xstrdup(json_object_get_string(message));
4b6aa8
+
4b6aa8
+        json_object *bthash = NULL;
4b6aa8
+        if (json_object_object_get_ex(json, "bthash", &bthash))
4b6aa8
+            out_response->bthash = xstrdup(json_object_get_string(bthash));
4b6aa8
+
4b6aa8
+        json_object *reported_to_list = NULL;
4b6aa8
+        if (json_object_object_get_ex(json, "reported_to", &reported_to_list))
4b6aa8
+            out_response->reported_to_list = parse_reported_to_from_json_list(reported_to_list);
4b6aa8
+
4b6aa8
+        json_object *solutions = NULL;
4b6aa8
+        if (json_object_object_get_ex(json, "solutions", &solutions))
4b6aa8
+            out_response->solution = parse_solution_from_json_list(solutions, &(out_response->reported_to_list));
4b6aa8
+
4b6aa8
+        return out_response;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    return NULL;
4b6aa8
+}
4b6aa8
+
4b6aa8
+static struct ureport_server_response *get_server_response(post_state_t *post_state, struct ureport_server_config *config)
4b6aa8
+{
4b6aa8
+    /* Previously, the condition here was (post_state->errmsg[0] != '\0')
4b6aa8
+     * however when the server asks for optional client authentication and we do not have the certificates,
4b6aa8
+     * then post_state->errmsg contains "NSS: client certificate not found (nickname not specified)" even though
4b6aa8
+     * the request succeeded.
4b6aa8
+     */
4b6aa8
+    if (post_state->curl_result != CURLE_OK)
4b6aa8
+    {
4b6aa8
+        error_msg(_("Failed to upload uReport to the server '%s' with curl: %s"), config->ur_url, post_state->errmsg);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (post_state->http_resp_code == 404)
4b6aa8
+    {
4b6aa8
+        error_msg(_("The URL '%s' does not exist (got error 404 from server)"), config->ur_url);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (post_state->http_resp_code == 500)
4b6aa8
+    {
4b6aa8
+        error_msg(_("The server at '%s' encountered an internal error (got error 500)"), config->ur_url);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (post_state->http_resp_code == 503)
4b6aa8
+    {
4b6aa8
+        error_msg(_("The server at '%s' currently can't handle the request (got error 503)"), config->ur_url);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (post_state->http_resp_code != 202
4b6aa8
+            && post_state->http_resp_code != 400
4b6aa8
+            && post_state->http_resp_code != 413)
4b6aa8
+    {
4b6aa8
+        /* can't print better error message */
4b6aa8
+        error_msg(_("Unexpected HTTP response from '%s': %d"), config->ur_url, post_state->http_resp_code);
4b6aa8
+        log_notice("%s", post_state->body);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    json_object *const json = json_tokener_parse(post_state->body);
4b6aa8
+
4b6aa8
+    if (is_error(json))
4b6aa8
+    {
4b6aa8
+        error_msg(_("Unable to parse response from ureport server at '%s'"), config->ur_url);
4b6aa8
+        log_notice("%s", post_state->body);
4b6aa8
+        json_object_put(json);
4b6aa8
+        return NULL;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    struct ureport_server_response *response = ureport_server_parse_json(json);
4b6aa8
+    json_object_put(json);
4b6aa8
+
4b6aa8
+    if (!response)
4b6aa8
+        error_msg(_("The response from '%s' has invalid format"), config->ur_url);
4b6aa8
+    else if ((post_state->http_resp_code == 202 && response->is_error)
4b6aa8
+                || (post_state->http_resp_code != 202 && !response->is_error))
4b6aa8
+    {
4b6aa8
+        /* HTTP CODE 202 means that call was successful but the response */
4b6aa8
+        /* has an error message */
4b6aa8
+        error_msg(_("Type mismatch has been detected in the response from '%s'"), config->ur_url);
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    return response;
4b6aa8
+}
4b6aa8
+
4b6aa8
+typedef post_state_t *(*attach_handler)(const char *, void *, struct ureport_server_config *);
4b6aa8
+
4b6aa8
+static post_state_t *wrp_ureport_attach_rhbz(const char *ureport_hash, int *rhbz_bug,
4b6aa8
+        struct ureport_server_config *config)
4b6aa8
+{
4b6aa8
+    return ureport_attach_rhbz(ureport_hash, *rhbz_bug, config);
4b6aa8
+}
4b6aa8
+
4b6aa8
+static bool perform_attach(struct ureport_server_config *config, const char *ureport_hash,
4b6aa8
+        attach_handler handler, void *args)
4b6aa8
+{
4b6aa8
+    char *dest_url = concat_path_file(config->ur_url, ATTACH_URL_SFX);
4b6aa8
+    const char *old_url = config->ur_url;
4b6aa8
+    config->ur_url = dest_url;
4b6aa8
+    post_state_t *post_state = handler(ureport_hash, args, config);
4b6aa8
+    config->ur_url = old_url;
4b6aa8
+    free(dest_url);
4b6aa8
+
4b6aa8
+    struct ureport_server_response *resp = get_server_response(post_state, config);
4b6aa8
+    free_post_state(post_state);
4b6aa8
+    /* don't use str_bo_bool() because we require "true" string */
4b6aa8
+    const int result = !resp || resp->is_error || strcmp(resp->value,"true") != 0;
4b6aa8
+
4b6aa8
+    if (resp && resp->is_error)
4b6aa8
+    {
4b6aa8
+        error_msg(_("The server at '%s' responded with an error: '%s'"), config->ur_url, resp->value);
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    free_ureport_server_response(resp);
4b6aa8
+
4b6aa8
+    return result;
4b6aa8
+}
4b6aa8
+
4b6aa8
+int main(int argc, char **argv)
4b6aa8
+{
4b6aa8
+    setlocale(LC_ALL, "");
4b6aa8
+#if ENABLE_NLS
4b6aa8
+    bindtextdomain(PACKAGE, LOCALEDIR);
4b6aa8
+    textdomain(PACKAGE);
4b6aa8
+#endif
4b6aa8
+
4b6aa8
+    abrt_init(argv);
4b6aa8
+
4b6aa8
+    struct ureport_server_config config = {
4b6aa8
+        .ur_url = NULL,
4b6aa8
+        .ur_ssl_verify = true,
4b6aa8
+        .ur_client_cert = NULL,
4b6aa8
+        .ur_client_key = NULL,
4b6aa8
+        .ur_http_headers = NULL,
4b6aa8
+        {
4b6aa8
+            .urp_auth_items = NULL,
4b6aa8
+        },
4b6aa8
+    };
4b6aa8
+
4b6aa8
+    config.ur_http_headers = new_map_string();
4b6aa8
+
4b6aa8
+    enum {
4b6aa8
+        OPT_v = 1 << 0,
4b6aa8
+        OPT_d = 1 << 1,
4b6aa8
+        OPT_u = 1 << 2,
4b6aa8
+        OPT_k = 1 << 3,
4b6aa8
+        OPT_t = 1 << 4,
4b6aa8
+        OPT_i = 1 << 5,
4b6aa8
+    };
4b6aa8
+
4b6aa8
+    int ret = 1; /* "failure" (for now) */
4b6aa8
+    bool insecure = !config.ur_ssl_verify;
4b6aa8
+    const char *conf_file = CONF_FILE_PATH;
4b6aa8
+    const char *arg_server_url = NULL;
4b6aa8
+    const char *client_auth = NULL;
4b6aa8
+    GList *auth_items = NULL;
4b6aa8
+    const char *dump_dir_path = ".";
4b6aa8
+    const char *ureport_hash = NULL;
4b6aa8
+    bool ureport_hash_from_rt = false;
4b6aa8
+    int rhbz_bug = -1;
4b6aa8
+    bool rhbz_bug_from_rt = false;
4b6aa8
+    const char *email_address = NULL;
4b6aa8
+    bool email_address_from_env = false;
4b6aa8
+    struct dump_dir *dd = NULL;
4b6aa8
+    struct options program_options[] = {
4b6aa8
+        OPT__VERBOSE(&g_verbose),
4b6aa8
+        OPT__DUMP_DIR(&dump_dir_path),
4b6aa8
+        OPT_STRING('u', "url", &arg_server_url, "URL", _("Specify server URL")),
4b6aa8
+        OPT_BOOL('k', "insecure", &insecure,
4b6aa8
+                          _("Allow insecure connection to ureport server")),
4b6aa8
+        OPT_STRING('t', "auth", &client_auth, "SOURCE", _("Use client authentication")),
4b6aa8
+        OPT_LIST('i', "auth_items", &auth_items, "AUTH_ITEMS", _("Additional files included in 'auth' key")),
4b6aa8
+        OPT_STRING('c', NULL, &conf_file, "FILE", _("Configuration file")),
4b6aa8
+        OPT_STRING('a', "attach", &ureport_hash, "BTHASH",
4b6aa8
+                          _("bthash of uReport to attach (conflicts with -A)")),
4b6aa8
+        OPT_BOOL('A', "attach-rt", &ureport_hash_from_rt,
4b6aa8
+                          _("attach to a bthash from reported_to (conflicts with -a)")),
4b6aa8
+        OPT_STRING('e', "email", &email_address, "EMAIL",
4b6aa8
+                          _("contact e-mail address (requires -a|-A, conflicts with -E)")),
4b6aa8
+        OPT_BOOL('E', "email-env", &email_address_from_env,
4b6aa8
+                          _("contact e-mail address from environment or configuration file (requires -a|-A, conflicts with -e)")),
4b6aa8
+        OPT_INTEGER('b', "bug-id", &rhbz_bug,
4b6aa8
+                          _("attach RHBZ bug (requires -a|-A, conflicts with -B)")),
4b6aa8
+        OPT_BOOL('B', "bug-id-rt", &rhbz_bug_from_rt,
4b6aa8
+                          _("attach last RHBZ bug from reported_to (requires -a|-A, conflicts with -b)")),
4b6aa8
+        OPT_END(),
4b6aa8
+    };
4b6aa8
+
4b6aa8
+    const char *program_usage_string = _(
4b6aa8
+        "& [-v] [-c FILE] [-u URL] [-k] [-t SOURCE] [-A -a bthash -B -b bug-id -E -e email] [-d DIR]\n"
4b6aa8
+        "& [-v] [-c FILE] [-u URL] [-k] [-t SOURCE] [-i AUTH_ITEMS]\\\n"
4b6aa8
+        "  [-A -a bthash -B -b bug-id -E -e email] [-d DIR]\n"
4b6aa8
+        "\n"
4b6aa8
+        "Upload micro report or add an attachment to a micro report\n"
4b6aa8
+        "\n"
4b6aa8
+        "Reads the default configuration from "CONF_FILE_PATH
4b6aa8
+    );
4b6aa8
+
4b6aa8
+    unsigned opts = parse_opts(argc, argv, program_options, program_usage_string);
4b6aa8
+
4b6aa8
+    map_string_t *settings = new_map_string();
4b6aa8
+    load_conf_file(conf_file, settings, /*skip key w/o values:*/ false);
4b6aa8
+
4b6aa8
+    load_ureport_server_config(&config, settings);
4b6aa8
+
4b6aa8
+    if (opts & OPT_u)
4b6aa8
+        config.ur_url = arg_server_url;
4b6aa8
+    if (opts & OPT_k)
4b6aa8
+        config.ur_ssl_verify = !insecure;
4b6aa8
+    if (opts & OPT_t)
4b6aa8
+        parse_client_auth_paths(&config, client_auth);
4b6aa8
+    if (opts & OPT_i)
4b6aa8
+    {
4b6aa8
+        g_list_free_full(config.ur_prefs.urp_auth_items, free);
4b6aa8
+        config.ur_prefs.urp_auth_items = auth_items;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (!config.ur_url)
4b6aa8
+        error_msg_and_die("You need to specify server URL");
4b6aa8
+
4b6aa8
+    post_state_t *post_state = NULL;
4b6aa8
+
4b6aa8
+    if (ureport_hash && ureport_hash_from_rt)
4b6aa8
+        error_msg_and_die("You need to pass either -a bthash or -A");
4b6aa8
+
4b6aa8
+    if (rhbz_bug >= 0 && rhbz_bug_from_rt)
4b6aa8
+        error_msg_and_die("You need to pass either -b bug-id or -B");
4b6aa8
+
4b6aa8
+    if (email_address && email_address_from_env)
4b6aa8
+        error_msg_and_die("You need to pass either -e bthash or -E");
4b6aa8
+
4b6aa8
+    if (ureport_hash_from_rt || rhbz_bug_from_rt)
4b6aa8
+    {
4b6aa8
+        dd = dd_opendir(dump_dir_path, DD_OPEN_READONLY);
4b6aa8
+        if (!dd)
4b6aa8
+            xfunc_die();
4b6aa8
+
4b6aa8
+        if (ureport_hash_from_rt)
4b6aa8
+        {
4b6aa8
+            report_result_t *ureport_result = find_in_reported_to(dd, "uReport");
4b6aa8
+
4b6aa8
+            if (!ureport_result || !ureport_result->bthash)
4b6aa8
+                error_msg_and_die(_("This problem does not have an uReport assigned."));
4b6aa8
+
4b6aa8
+            /* sorry, this will be leaked */
4b6aa8
+            ureport_hash = xstrdup(ureport_result->bthash);
4b6aa8
+
4b6aa8
+            free_report_result(ureport_result);
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        if (rhbz_bug_from_rt)
4b6aa8
+        {
4b6aa8
+            report_result_t *bz_result = find_in_reported_to(dd, "Bugzilla");
4b6aa8
+
4b6aa8
+            if (!bz_result || !bz_result->url)
4b6aa8
+                error_msg_and_die(_("This problem has not been reported to Bugzilla."));
4b6aa8
+
4b6aa8
+            char *bugid_ptr = strstr(bz_result->url, "show_bug.cgi?id=");
4b6aa8
+            if (!bugid_ptr)
4b6aa8
+                error_msg_and_die(_("Unable to find bug ID in bugzilla URL '%s'"), bz_result->url);
4b6aa8
+            bugid_ptr += strlen("show_bug.cgi?id=");
4b6aa8
+
4b6aa8
+            /* we're just reading int, sscanf works fine */
4b6aa8
+            if (sscanf(bugid_ptr, "%d", &rhbz_bug) != 1)
4b6aa8
+                error_msg_and_die(_("Unable to parse bug ID from bugzilla URL '%s'"), bz_result->url);
4b6aa8
+
4b6aa8
+            free_report_result(bz_result);
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        dd_close(dd);
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (email_address_from_env)
4b6aa8
+    {
4b6aa8
+        VALUE_FROM_CONF("ContactEmail", email_address, (const char *));
4b6aa8
+
4b6aa8
+        if (!email_address)
4b6aa8
+            error_msg_and_die(_("Neither environment variable 'uReport_ContactEmail' nor configuration option 'ContactEmail' is set"));
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    if (ureport_hash)
4b6aa8
+    {
4b6aa8
+        if (rhbz_bug < 0 && !email_address)
4b6aa8
+            error_msg_and_die(_("You need to specify bug ID, contact email or both"));
4b6aa8
+
4b6aa8
+        if (rhbz_bug >= 0)
4b6aa8
+        {
4b6aa8
+            if (perform_attach(&config, ureport_hash, (attach_handler)wrp_ureport_attach_rhbz, (void *)&rhbz_bug))
4b6aa8
+                goto finalize;
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        if (email_address)
4b6aa8
+        {
4b6aa8
+            if (perform_attach(&config, ureport_hash, (attach_handler)ureport_attach_email, (void *)email_address))
4b6aa8
+                goto finalize;
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        ret = 0;
4b6aa8
+        goto finalize;
4b6aa8
+    }
4b6aa8
+    if (!ureport_hash && (rhbz_bug >= 0 || email_address))
4b6aa8
+        error_msg_and_die(_("You need to specify bthash of the uReport to attach."));
4b6aa8
+
4b6aa8
+    /* -b, -a nor -r were specified - upload uReport from dump_dir */
4b6aa8
+    const char *server_url = config.ur_url;
4b6aa8
+    char *dest_url = concat_path_file(config.ur_url, REPORT_URL_SFX);
4b6aa8
+    config.ur_url = dest_url;
4b6aa8
+
4b6aa8
+    char *json_ureport = ureport_from_dump_dir_ext(dump_dir_path, &(config.ur_prefs));
4b6aa8
+    if (!json_ureport)
4b6aa8
+    {
4b6aa8
+        error_msg(_("Not uploading an empty uReport"));
4b6aa8
+        goto format_err;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    post_state = ureport_post(json_ureport, &config);
4b6aa8
+    free(json_ureport);
4b6aa8
+
4b6aa8
+    if (!post_state)
4b6aa8
+    {
4b6aa8
+        error_msg(_("Failed on submitting the problem"));
4b6aa8
+        goto format_err;
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    struct ureport_server_response *response = get_server_response(post_state, &config);
4b6aa8
+
4b6aa8
+    if (!response)
4b6aa8
+        goto format_err;
4b6aa8
+
4b6aa8
+    if (!response->is_error)
4b6aa8
+    {
4b6aa8
+        log_notice("is known: %s", response->value);
4b6aa8
+        ret = 0; /* "success" */
4b6aa8
+
4b6aa8
+        dd = dd_opendir(dump_dir_path, /* flags */ 0);
4b6aa8
+        if (!dd)
4b6aa8
+            xfunc_die();
4b6aa8
+
4b6aa8
+        if (response->bthash)
4b6aa8
+        {
4b6aa8
+            char *msg = xasprintf("uReport: BTHASH=%s", response->bthash);
4b6aa8
+            add_reported_to(dd, msg);
4b6aa8
+            free(msg);
4b6aa8
+
4b6aa8
+            char *bthash_url = concat_path_file(server_url, BTHASH_URL_SFX);
4b6aa8
+            msg = xasprintf("ABRT Server: URL=%s%s", bthash_url, response->bthash);
4b6aa8
+            add_reported_to(dd, msg);
4b6aa8
+            free(msg);
4b6aa8
+            free(bthash_url);
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        if (response->reported_to_list)
4b6aa8
+        {
4b6aa8
+            for (GList *e = response->reported_to_list; e; e = g_list_next(e))
4b6aa8
+                add_reported_to(dd, e->data);
4b6aa8
+        }
4b6aa8
+
4b6aa8
+        if (response->solution)
4b6aa8
+            dd_save_text(dd, FILENAME_NOT_REPORTABLE, response->solution);
4b6aa8
+
4b6aa8
+        dd_close(dd);
4b6aa8
+
4b6aa8
+        /* If a reported problem is not known then emit NEEDMORE */
4b6aa8
+        if (strcmp("true", response->value) == 0)
4b6aa8
+        {
4b6aa8
+            log(_("This problem has already been reported."));
4b6aa8
+            if (response->message)
4b6aa8
+                log(response->message);
4b6aa8
+
4b6aa8
+            ret = EXIT_STOP_EVENT_RUN;
4b6aa8
+        }
4b6aa8
+    }
4b6aa8
+    else
4b6aa8
+    {
4b6aa8
+        error_msg(_("Server responded with an error: '%s'"), response->value);
4b6aa8
+    }
4b6aa8
+
4b6aa8
+    free_ureport_server_response(response);
4b6aa8
+
4b6aa8
+format_err:
4b6aa8
+    free_post_state(post_state);
4b6aa8
+    free(dest_url);
4b6aa8
+
4b6aa8
+finalize:
4b6aa8
+    if (config.ur_prefs.urp_auth_items != auth_items)
4b6aa8
+        g_list_free_full(config.ur_prefs.urp_auth_items, free);
4b6aa8
+
4b6aa8
+    free_map_string(config.ur_http_headers);
4b6aa8
+
4b6aa8
+    free_map_string(settings);
4b6aa8
+    free(config.ur_client_cert);
4b6aa8
+    free(config.ur_client_key);
4b6aa8
+
4b6aa8
+    return ret;
4b6aa8
+}
4b6aa8
diff --git a/src/plugins/ureport.c b/src/plugins/ureport.c
4b6aa8
deleted file mode 100644
4b6aa8
index 9c69cad..0000000
4b6aa8
--- a/src/plugins/ureport.c
4b6aa8
+++ /dev/null
4b6aa8
@@ -1,777 +0,0 @@
4b6aa8
-/*
4b6aa8
-    Copyright (C) 2012  ABRT Team
4b6aa8
-    Copyright (C) 2012  RedHat inc.
4b6aa8
-
4b6aa8
-    This program is free software; you can redistribute it and/or modify
4b6aa8
-    it under the terms of the GNU General Public License as published by
4b6aa8
-    the Free Software Foundation; either version 2 of the License, or
4b6aa8
-    (at your option) any later version.
4b6aa8
-
4b6aa8
-    This program is distributed in the hope that it will be useful,
4b6aa8
-    but WITHOUT ANY WARRANTY; without even the implied warranty of
4b6aa8
-    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
4b6aa8
-    GNU General Public License for more details.
4b6aa8
-
4b6aa8
-    You should have received a copy of the GNU General Public License along
4b6aa8
-    with this program; if not, write to the Free Software Foundation, Inc.,
4b6aa8
-    51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
4b6aa8
-*/
4b6aa8
-
4b6aa8
-#include <json/json.h>
4b6aa8
-#include "internal_libreport.h"
4b6aa8
-#include "ureport.h"
4b6aa8
-#include "libreport_curl.h"
4b6aa8
-
4b6aa8
-#define CONF_FILE_PATH PLUGINS_CONF_DIR"/ureport.conf"
4b6aa8
-
4b6aa8
-#define REPORT_URL_SFX "reports/new/"
4b6aa8
-#define ATTACH_URL_SFX "reports/attach/"
4b6aa8
-#define BTHASH_URL_SFX "reports/bthash/"
4b6aa8
-
4b6aa8
-#define RHSM_CERT_PATH "/etc/pki/consumer/cert.pem"
4b6aa8
-#define RHSM_KEY_PATH "/etc/pki/consumer/key.pem"
4b6aa8
-
4b6aa8
-#define RHAP_PEM_DIR_PATH "/etc/pki/entitlement"
4b6aa8
-#define RHAP_ENT_DATA_BEGIN_TAG "-----BEGIN ENTITLEMENT DATA-----"
4b6aa8
-#define RHAP_ENT_DATA_END_TAG "-----END ENTITLEMENT DATA-----"
4b6aa8
-#define RHAP_SIG_DATA_BEGIN_TAG "-----BEGIN RSA SIGNATURE-----"
4b6aa8
-#define RHAP_SIG_DATA_END_TAG "-----END RSA SIGNATURE-----"
4b6aa8
-
4b6aa8
-#define VALUE_FROM_CONF(opt, var, tr) do { const char *value = getenv("uReport_"opt); \
4b6aa8
-        if (!value) { value = get_map_string_item_or_NULL(settings, opt); } if (value) { var = tr(value); } \
4b6aa8
-    } while(0)
4b6aa8
-
4b6aa8
-static char *puppet_config_print(const char *key)
4b6aa8
-{
4b6aa8
-    char *command = xasprintf("puppet config print %s", key);
4b6aa8
-    char *result = run_in_shell_and_save_output(0, command, NULL, NULL);
4b6aa8
-    free(command);
4b6aa8
-
4b6aa8
-    /* run_in_shell_and_save_output always returns non-NULL */
4b6aa8
-    if (result[0] != '/')
4b6aa8
-        goto error;
4b6aa8
-
4b6aa8
-    char *newline = strchrnul(result, '\n');
4b6aa8
-    if (!newline)
4b6aa8
-        goto error;
4b6aa8
-
4b6aa8
-    *newline = '\0';
4b6aa8
-    return result;
4b6aa8
-error:
4b6aa8
-    free(result);
4b6aa8
-    error_msg_and_die("Unable to determine puppet %s path (puppet not installed?)", key);
4b6aa8
-}
4b6aa8
-
4b6aa8
-static void parse_client_auth_paths(struct ureport_server_config *config, const char *client_auth)
4b6aa8
-{
4b6aa8
-    if (client_auth == NULL)
4b6aa8
-        return;
4b6aa8
-
4b6aa8
-    if (strcmp(client_auth, "") == 0)
4b6aa8
-    {
4b6aa8
-        config->ur_client_cert = NULL;
4b6aa8
-        config->ur_client_key = NULL;
4b6aa8
-        log_notice("Not using client authentication");
4b6aa8
-    }
4b6aa8
-    else if (strcmp(client_auth, "rhsm") == 0)
4b6aa8
-    {
4b6aa8
-        config->ur_client_cert = xstrdup(RHSM_CERT_PATH);
4b6aa8
-        config->ur_client_key = xstrdup(RHSM_KEY_PATH);
4b6aa8
-    }
4b6aa8
-    else if (strcmp(client_auth, "rhsm-entitlement") == 0)
4b6aa8
-    {
4b6aa8
-        GList *certs = get_file_list(RHAP_PEM_DIR_PATH, "pem");
4b6aa8
-        if (g_list_length(certs) != 2)
4b6aa8
-        {
4b6aa8
-            log_notice(RHAP_PEM_DIR_PATH" does not contain unique cert-key files pair");
4b6aa8
-            log_notice("Not using client authentication");
4b6aa8
-            return;
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        const char *cert = NULL;
4b6aa8
-        const char *key = NULL;
4b6aa8
-
4b6aa8
-        file_obj_t *fst = (file_obj_t *)certs->data;
4b6aa8
-        file_obj_t *scn = (file_obj_t *)certs->next->data;
4b6aa8
-
4b6aa8
-        if (strlen(fo_get_filename(fst)) < strlen(fo_get_filename(scn)))
4b6aa8
-        {
4b6aa8
-            cert = fo_get_filename(fst);
4b6aa8
-            key = fo_get_filename(scn);
4b6aa8
-
4b6aa8
-            config->ur_client_cert = xstrdup(fo_get_fullpath(fst));
4b6aa8
-            config->ur_client_key = xstrdup(fo_get_fullpath(scn));
4b6aa8
-        }
4b6aa8
-        else
4b6aa8
-        {
4b6aa8
-            cert = fo_get_filename(scn);
4b6aa8
-            key = fo_get_filename(fst);
4b6aa8
-
4b6aa8
-            config->ur_client_cert = xstrdup(fo_get_fullpath(scn));
4b6aa8
-            config->ur_client_key = xstrdup(fo_get_fullpath(fst));
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        const bool iscomplement = prefixcmp(key, cert) != 0 || strcmp("-key", key + strlen(cert)) != 0;
4b6aa8
-        g_list_free_full(certs, (GDestroyNotify)free_file_obj);
4b6aa8
-
4b6aa8
-        if (iscomplement)
4b6aa8
-        {
4b6aa8
-            log_notice("Key file '%s' isn't complement to cert file '%s'",
4b6aa8
-                    config->ur_client_key, config->ur_client_cert);
4b6aa8
-            log_notice("Not using client authentication");
4b6aa8
-
4b6aa8
-            free(config->ur_client_cert);
4b6aa8
-            free(config->ur_client_key);
4b6aa8
-            config->ur_client_cert = NULL;
4b6aa8
-            config->ur_client_key = NULL;
4b6aa8
-
4b6aa8
-            return;
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        char *certdata = xmalloc_open_read_close(config->ur_client_cert, /*no size limit*/NULL);
4b6aa8
-        if (certdata != NULL)
4b6aa8
-        {
4b6aa8
-            char *ent_data = xstrdup_between(certdata,
4b6aa8
-                    RHAP_ENT_DATA_BEGIN_TAG, RHAP_ENT_DATA_END_TAG);
4b6aa8
-
4b6aa8
-            char *sig_data = xstrdup_between(certdata,
4b6aa8
-                    RHAP_SIG_DATA_BEGIN_TAG, RHAP_SIG_DATA_END_TAG);
4b6aa8
-
4b6aa8
-            if (ent_data != NULL && sig_data != NULL)
4b6aa8
-            {
4b6aa8
-                ent_data = strremovech(ent_data, '\n');
4b6aa8
-                insert_map_string(config->ur_http_headers,
4b6aa8
-                        xstrdup("X-RH-Entitlement-Data"),
4b6aa8
-                        xasprintf(RHAP_ENT_DATA_BEGIN_TAG"%s"RHAP_ENT_DATA_END_TAG, ent_data));
4b6aa8
-
4b6aa8
-                sig_data = strremovech(sig_data, '\n');
4b6aa8
-                insert_map_string(config->ur_http_headers,
4b6aa8
-                        xstrdup("X-RH-Entitlement-Sig"),
4b6aa8
-                        xasprintf(RHAP_SIG_DATA_BEGIN_TAG"%s"RHAP_SIG_DATA_END_TAG, sig_data));
4b6aa8
-            }
4b6aa8
-            else
4b6aa8
-            {
4b6aa8
-                log_notice("Cert file '%s' doesn't contain Entitlement and RSA Signature sections", config->ur_client_cert);
4b6aa8
-                log_notice("Not using HTTP authentication headers");
4b6aa8
-            }
4b6aa8
-
4b6aa8
-            free(sig_data);
4b6aa8
-            free(ent_data);
4b6aa8
-            free(certdata);
4b6aa8
-        }
4b6aa8
-    }
4b6aa8
-    else if (strcmp(client_auth, "puppet") == 0)
4b6aa8
-    {
4b6aa8
-        config->ur_client_cert = puppet_config_print("hostcert");
4b6aa8
-        config->ur_client_key = puppet_config_print("hostprivkey");
4b6aa8
-    }
4b6aa8
-    else
4b6aa8
-    {
4b6aa8
-        char *scratch = xstrdup(client_auth);
4b6aa8
-        config->ur_client_cert = xstrdup(strtok(scratch, ":"));
4b6aa8
-        config->ur_client_key = xstrdup(strtok(NULL, ":"));
4b6aa8
-        free(scratch);
4b6aa8
-        if (config->ur_client_cert == NULL || config->ur_client_key == NULL)
4b6aa8
-            error_msg_and_die("Invalid client authentication specification");
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (config->ur_client_cert && config->ur_client_key)
4b6aa8
-    {
4b6aa8
-        log_notice("Using client certificate: %s", config->ur_client_cert);
4b6aa8
-        log_notice("Using client private key: %s", config->ur_client_key);
4b6aa8
-    }
4b6aa8
-}
4b6aa8
-
4b6aa8
-/*
4b6aa8
- * Loads uReport configuration from various sources.
4b6aa8
- *
4b6aa8
- * Replaces a value of an already configured option only if the
4b6aa8
- * option was found in a configuration source.
4b6aa8
- *
4b6aa8
- * @param config a server configuration to be populated
4b6aa8
- */
4b6aa8
-static void load_ureport_server_config(struct ureport_server_config *config, map_string_t *settings)
4b6aa8
-{
4b6aa8
-    VALUE_FROM_CONF("URL", config->ur_url, (const char *));
4b6aa8
-    VALUE_FROM_CONF("SSLVerify", config->ur_ssl_verify, string_to_bool);
4b6aa8
-
4b6aa8
-    bool include_auth = false;
4b6aa8
-    VALUE_FROM_CONF("IncludeAuthData", include_auth, string_to_bool);
4b6aa8
-
4b6aa8
-    if (include_auth)
4b6aa8
-    {
4b6aa8
-        const char *auth_items = NULL;
4b6aa8
-        VALUE_FROM_CONF("AuthDataItems", auth_items, (const char *));
4b6aa8
-        config->ur_prefs.urp_auth_items = parse_list(auth_items);
4b6aa8
-
4b6aa8
-        if (config->ur_prefs.urp_auth_items == NULL)
4b6aa8
-            log_warning("IncludeAuthData set to 'yes' but AuthDataItems is empty.");
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    const char *client_auth = NULL;
4b6aa8
-    VALUE_FROM_CONF("SSLClientAuth", client_auth, (const char *));
4b6aa8
-    parse_client_auth_paths(config, client_auth);
4b6aa8
-}
4b6aa8
-
4b6aa8
-struct ureport_server_response {
4b6aa8
-    bool is_error;
4b6aa8
-    char *value;
4b6aa8
-    char *message;
4b6aa8
-    char *bthash;
4b6aa8
-    GList *reported_to_list;
4b6aa8
-    char *solution;
4b6aa8
-};
4b6aa8
-
4b6aa8
-void free_ureport_server_response(struct ureport_server_response *resp)
4b6aa8
-{
4b6aa8
-    if (!resp)
4b6aa8
-        return;
4b6aa8
-
4b6aa8
-    free(resp->solution);
4b6aa8
-    g_list_free_full(resp->reported_to_list, g_free);
4b6aa8
-    free(resp->bthash);
4b6aa8
-    free(resp->message);
4b6aa8
-    free(resp->value);
4b6aa8
-    free(resp);
4b6aa8
-}
4b6aa8
-
4b6aa8
-static char *parse_solution_from_json_list(struct json_object *list, GList **reported_to)
4b6aa8
-{
4b6aa8
-    json_object *list_elem, *struct_elem;
4b6aa8
-    const char *cause, *note, *url;
4b6aa8
-    struct strbuf *solution_buf = strbuf_new();
4b6aa8
-
4b6aa8
-    const unsigned length = json_object_array_length(list);
4b6aa8
-
4b6aa8
-    const char *one_format = _("Your problem seems to be caused by %s\n\n%s\n");
4b6aa8
-    if (length > 1)
4b6aa8
-    {
4b6aa8
-        strbuf_append_str(solution_buf, _("Your problem seems to be caused by one of the following:\n"));
4b6aa8
-        one_format = "\n* %s\n\n%s\n";
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    bool empty = true;
4b6aa8
-    for (unsigned i = 0; i < length; ++i)
4b6aa8
-    {
4b6aa8
-        list_elem = json_object_array_get_idx(list, i);
4b6aa8
-        if (!list_elem)
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        if (!json_object_object_get_ex(list_elem, "cause", &struct_elem))
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        cause = json_object_get_string(struct_elem);
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        if (!json_object_object_get_ex(list_elem, "note", &struct_elem))
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        note = json_object_get_string(struct_elem);
4b6aa8
-        if (!note)
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        empty = false;
4b6aa8
-        strbuf_append_strf(solution_buf, one_format, cause, note);
4b6aa8
-
4b6aa8
-        if (!json_object_object_get_ex(list_elem, "url", &struct_elem))
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        url = json_object_get_string(struct_elem);
4b6aa8
-        if (url)
4b6aa8
-        {
4b6aa8
-            char *reported_to_line = xasprintf("%s: URL=%s", cause, url);
4b6aa8
-            *reported_to = g_list_append(*reported_to, reported_to_line);
4b6aa8
-        }
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (empty)
4b6aa8
-    {
4b6aa8
-        strbuf_free(solution_buf);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    return strbuf_free_nobuf(solution_buf);
4b6aa8
-}
4b6aa8
-
4b6aa8
-/* reported_to json element should be a list of structures
4b6aa8
-{ "reporter": "Bugzilla",
4b6aa8
-  "type": "url",
4b6aa8
-  "value": "https://bugzilla.redhat.com/show_bug.cgi?id=XYZ" } */
4b6aa8
-static GList *parse_reported_to_from_json_list(struct json_object *list)
4b6aa8
-{
4b6aa8
-    int i;
4b6aa8
-    json_object *list_elem, *struct_elem;
4b6aa8
-    const char *reporter, *value, *type;
4b6aa8
-    char *reported_to_line, *prefix;
4b6aa8
-    GList *result = NULL;
4b6aa8
-
4b6aa8
-    for (i = 0; i < json_object_array_length(list); ++i)
4b6aa8
-    {
4b6aa8
-        prefix = NULL;
4b6aa8
-        list_elem = json_object_array_get_idx(list, i);
4b6aa8
-        if (!list_elem)
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        if (!json_object_object_get_ex(list_elem, "reporter", &struct_elem))
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        reporter = json_object_get_string(struct_elem);
4b6aa8
-        if (!reporter)
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        if (!json_object_object_get_ex(list_elem, "value", &struct_elem))
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        value = json_object_get_string(struct_elem);
4b6aa8
-        if (!value)
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        if (!json_object_object_get_ex(list_elem, "type", &struct_elem))
4b6aa8
-            continue;
4b6aa8
-
4b6aa8
-        type = json_object_get_string(struct_elem);
4b6aa8
-        if (type)
4b6aa8
-        {
4b6aa8
-            if (strcasecmp("url", type) == 0)
4b6aa8
-                prefix = xstrdup("URL=");
4b6aa8
-            else if (strcasecmp("bthash", type) == 0)
4b6aa8
-                prefix = xstrdup("BTHASH=");
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        if (!prefix)
4b6aa8
-            prefix = xstrdup("");
4b6aa8
-
4b6aa8
-        reported_to_line = xasprintf("%s: %s%s", reporter, prefix, value);
4b6aa8
-        free(prefix);
4b6aa8
-
4b6aa8
-        result = g_list_append(result, reported_to_line);
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    return result;
4b6aa8
-}
4b6aa8
-
4b6aa8
-/*
4b6aa8
- * Reponse samples:
4b6aa8
- * {"error":"field 'foo' is required"}
4b6aa8
- * {"response":"true"}
4b6aa8
- * {"response":"false"}
4b6aa8
- */
4b6aa8
-static struct ureport_server_response *ureport_server_parse_json(json_object *json)
4b6aa8
-{
4b6aa8
-    json_object *obj = NULL;
4b6aa8
-    if (json_object_object_get_ex(json, "error", &obj))
4b6aa8
-    {
4b6aa8
-        struct ureport_server_response *out_response = xzalloc(sizeof(*out_response));
4b6aa8
-        out_response->is_error = true;
4b6aa8
-        /*
4b6aa8
-         * Used to use json_object_to_json_string(obj), but it returns
4b6aa8
-         * the string in quote marks (") - IOW, json-formatted string.
4b6aa8
-         */
4b6aa8
-        out_response->value = xstrdup(json_object_get_string(obj));
4b6aa8
-        return out_response;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (json_object_object_get_ex(json, "result", &obj))
4b6aa8
-    {
4b6aa8
-        struct ureport_server_response *out_response = xzalloc(sizeof(*out_response));
4b6aa8
-        out_response->value = xstrdup(json_object_get_string(obj));
4b6aa8
-
4b6aa8
-        json_object *message = NULL;
4b6aa8
-        if (json_object_object_get_ex(json, "message", &message))
4b6aa8
-            out_response->message = xstrdup(json_object_get_string(message));
4b6aa8
-
4b6aa8
-        json_object *bthash = NULL;
4b6aa8
-        if (json_object_object_get_ex(json, "bthash", &bthash))
4b6aa8
-            out_response->bthash = xstrdup(json_object_get_string(bthash));
4b6aa8
-
4b6aa8
-        json_object *reported_to_list = NULL;
4b6aa8
-        if (json_object_object_get_ex(json, "reported_to", &reported_to_list))
4b6aa8
-            out_response->reported_to_list = parse_reported_to_from_json_list(reported_to_list);
4b6aa8
-
4b6aa8
-        json_object *solutions = NULL;
4b6aa8
-        if (json_object_object_get_ex(json, "solutions", &solutions))
4b6aa8
-            out_response->solution = parse_solution_from_json_list(solutions, &(out_response->reported_to_list));
4b6aa8
-
4b6aa8
-        return out_response;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    return NULL;
4b6aa8
-}
4b6aa8
-
4b6aa8
-static struct ureport_server_response *get_server_response(post_state_t *post_state, struct ureport_server_config *config)
4b6aa8
-{
4b6aa8
-    /* Previously, the condition here was (post_state->errmsg[0] != '\0')
4b6aa8
-     * however when the server asks for optional client authentication and we do not have the certificates,
4b6aa8
-     * then post_state->errmsg contains "NSS: client certificate not found (nickname not specified)" even though
4b6aa8
-     * the request succeeded.
4b6aa8
-     */
4b6aa8
-    if (post_state->curl_result != CURLE_OK)
4b6aa8
-    {
4b6aa8
-        error_msg(_("Failed to upload uReport to the server '%s' with curl: %s"), config->ur_url, post_state->errmsg);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (post_state->http_resp_code == 404)
4b6aa8
-    {
4b6aa8
-        error_msg(_("The URL '%s' does not exist (got error 404 from server)"), config->ur_url);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (post_state->http_resp_code == 500)
4b6aa8
-    {
4b6aa8
-        error_msg(_("The server at '%s' encountered an internal error (got error 500)"), config->ur_url);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (post_state->http_resp_code == 503)
4b6aa8
-    {
4b6aa8
-        error_msg(_("The server at '%s' currently can't handle the request (got error 503)"), config->ur_url);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (post_state->http_resp_code != 202
4b6aa8
-            && post_state->http_resp_code != 400
4b6aa8
-            && post_state->http_resp_code != 413)
4b6aa8
-    {
4b6aa8
-        /* can't print better error message */
4b6aa8
-        error_msg(_("Unexpected HTTP response from '%s': %d"), config->ur_url, post_state->http_resp_code);
4b6aa8
-        log_notice("%s", post_state->body);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    json_object *const json = json_tokener_parse(post_state->body);
4b6aa8
-
4b6aa8
-    if (is_error(json))
4b6aa8
-    {
4b6aa8
-        error_msg(_("Unable to parse response from ureport server at '%s'"), config->ur_url);
4b6aa8
-        log_notice("%s", post_state->body);
4b6aa8
-        json_object_put(json);
4b6aa8
-        return NULL;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    struct ureport_server_response *response = ureport_server_parse_json(json);
4b6aa8
-    json_object_put(json);
4b6aa8
-
4b6aa8
-    if (!response)
4b6aa8
-        error_msg(_("The response from '%s' has invalid format"), config->ur_url);
4b6aa8
-    else if ((post_state->http_resp_code == 202 && response->is_error)
4b6aa8
-                || (post_state->http_resp_code != 202 && !response->is_error))
4b6aa8
-    {
4b6aa8
-        /* HTTP CODE 202 means that call was successful but the response */
4b6aa8
-        /* has an error message */
4b6aa8
-        error_msg(_("Type mismatch has been detected in the response from '%s'"), config->ur_url);
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    return response;
4b6aa8
-}
4b6aa8
-
4b6aa8
-typedef post_state_t *(*attach_handler)(const char *, void *, struct ureport_server_config *);
4b6aa8
-
4b6aa8
-static post_state_t *wrp_ureport_attach_rhbz(const char *ureport_hash, int *rhbz_bug,
4b6aa8
-        struct ureport_server_config *config)
4b6aa8
-{
4b6aa8
-    return ureport_attach_rhbz(ureport_hash, *rhbz_bug, config);
4b6aa8
-}
4b6aa8
-
4b6aa8
-static bool perform_attach(struct ureport_server_config *config, const char *ureport_hash,
4b6aa8
-        attach_handler handler, void *args)
4b6aa8
-{
4b6aa8
-    char *dest_url = concat_path_file(config->ur_url, ATTACH_URL_SFX);
4b6aa8
-    const char *old_url = config->ur_url;
4b6aa8
-    config->ur_url = dest_url;
4b6aa8
-    post_state_t *post_state = handler(ureport_hash, args, config);
4b6aa8
-    config->ur_url = old_url;
4b6aa8
-    free(dest_url);
4b6aa8
-
4b6aa8
-    struct ureport_server_response *resp = get_server_response(post_state, config);
4b6aa8
-    free_post_state(post_state);
4b6aa8
-    /* don't use str_bo_bool() because we require "true" string */
4b6aa8
-    const int result = !resp || resp->is_error || strcmp(resp->value,"true") != 0;
4b6aa8
-
4b6aa8
-    if (resp && resp->is_error)
4b6aa8
-    {
4b6aa8
-        error_msg(_("The server at '%s' responded with an error: '%s'"), config->ur_url, resp->value);
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    free_ureport_server_response(resp);
4b6aa8
-
4b6aa8
-    return result;
4b6aa8
-}
4b6aa8
-
4b6aa8
-int main(int argc, char **argv)
4b6aa8
-{
4b6aa8
-    setlocale(LC_ALL, "");
4b6aa8
-#if ENABLE_NLS
4b6aa8
-    bindtextdomain(PACKAGE, LOCALEDIR);
4b6aa8
-    textdomain(PACKAGE);
4b6aa8
-#endif
4b6aa8
-
4b6aa8
-    abrt_init(argv);
4b6aa8
-
4b6aa8
-    struct ureport_server_config config = {
4b6aa8
-        .ur_url = NULL,
4b6aa8
-        .ur_ssl_verify = true,
4b6aa8
-        .ur_client_cert = NULL,
4b6aa8
-        .ur_client_key = NULL,
4b6aa8
-        .ur_http_headers = NULL,
4b6aa8
-        {
4b6aa8
-            .urp_auth_items = NULL,
4b6aa8
-        },
4b6aa8
-    };
4b6aa8
-
4b6aa8
-    config.ur_http_headers = new_map_string();
4b6aa8
-
4b6aa8
-    enum {
4b6aa8
-        OPT_v = 1 << 0,
4b6aa8
-        OPT_d = 1 << 1,
4b6aa8
-        OPT_u = 1 << 2,
4b6aa8
-        OPT_k = 1 << 3,
4b6aa8
-        OPT_t = 1 << 4,
4b6aa8
-        OPT_i = 1 << 5,
4b6aa8
-    };
4b6aa8
-
4b6aa8
-    int ret = 1; /* "failure" (for now) */
4b6aa8
-    bool insecure = !config.ur_ssl_verify;
4b6aa8
-    const char *conf_file = CONF_FILE_PATH;
4b6aa8
-    const char *arg_server_url = NULL;
4b6aa8
-    const char *client_auth = NULL;
4b6aa8
-    GList *auth_items = NULL;
4b6aa8
-    const char *dump_dir_path = ".";
4b6aa8
-    const char *ureport_hash = NULL;
4b6aa8
-    bool ureport_hash_from_rt = false;
4b6aa8
-    int rhbz_bug = -1;
4b6aa8
-    bool rhbz_bug_from_rt = false;
4b6aa8
-    const char *email_address = NULL;
4b6aa8
-    bool email_address_from_env = false;
4b6aa8
-    struct dump_dir *dd = NULL;
4b6aa8
-    struct options program_options[] = {
4b6aa8
-        OPT__VERBOSE(&g_verbose),
4b6aa8
-        OPT__DUMP_DIR(&dump_dir_path),
4b6aa8
-        OPT_STRING('u', "url", &arg_server_url, "URL", _("Specify server URL")),
4b6aa8
-        OPT_BOOL('k', "insecure", &insecure,
4b6aa8
-                          _("Allow insecure connection to ureport server")),
4b6aa8
-        OPT_STRING('t', "auth", &client_auth, "SOURCE", _("Use client authentication")),
4b6aa8
-        OPT_LIST('i', "auth_items", &auth_items, "AUTH_ITEMS", _("Additional files included in 'auth' key")),
4b6aa8
-        OPT_STRING('c', NULL, &conf_file, "FILE", _("Configuration file")),
4b6aa8
-        OPT_STRING('a', "attach", &ureport_hash, "BTHASH",
4b6aa8
-                          _("bthash of uReport to attach (conflicts with -A)")),
4b6aa8
-        OPT_BOOL('A', "attach-rt", &ureport_hash_from_rt,
4b6aa8
-                          _("attach to a bthash from reported_to (conflicts with -a)")),
4b6aa8
-        OPT_STRING('e', "email", &email_address, "EMAIL",
4b6aa8
-                          _("contact e-mail address (requires -a|-A, conflicts with -E)")),
4b6aa8
-        OPT_BOOL('E', "email-env", &email_address_from_env,
4b6aa8
-                          _("contact e-mail address from environment or configuration file (requires -a|-A, conflicts with -e)")),
4b6aa8
-        OPT_INTEGER('b', "bug-id", &rhbz_bug,
4b6aa8
-                          _("attach RHBZ bug (requires -a|-A, conflicts with -B)")),
4b6aa8
-        OPT_BOOL('B', "bug-id-rt", &rhbz_bug_from_rt,
4b6aa8
-                          _("attach last RHBZ bug from reported_to (requires -a|-A, conflicts with -b)")),
4b6aa8
-        OPT_END(),
4b6aa8
-    };
4b6aa8
-
4b6aa8
-    const char *program_usage_string = _(
4b6aa8
-        "& [-v] [-c FILE] [-u URL] [-k] [-t SOURCE] [-A -a bthash -B -b bug-id -E -e email] [-d DIR]\n"
4b6aa8
-        "& [-v] [-c FILE] [-u URL] [-k] [-t SOURCE] [-i AUTH_ITEMS]\\\n"
4b6aa8
-        "  [-A -a bthash -B -b bug-id -E -e email] [-d DIR]\n"
4b6aa8
-        "\n"
4b6aa8
-        "Upload micro report or add an attachment to a micro report\n"
4b6aa8
-        "\n"
4b6aa8
-        "Reads the default configuration from "CONF_FILE_PATH
4b6aa8
-    );
4b6aa8
-
4b6aa8
-    unsigned opts = parse_opts(argc, argv, program_options, program_usage_string);
4b6aa8
-
4b6aa8
-    map_string_t *settings = new_map_string();
4b6aa8
-    load_conf_file(conf_file, settings, /*skip key w/o values:*/ false);
4b6aa8
-
4b6aa8
-    load_ureport_server_config(&config, settings);
4b6aa8
-
4b6aa8
-    if (opts & OPT_u)
4b6aa8
-        config.ur_url = arg_server_url;
4b6aa8
-    if (opts & OPT_k)
4b6aa8
-        config.ur_ssl_verify = !insecure;
4b6aa8
-    if (opts & OPT_t)
4b6aa8
-        parse_client_auth_paths(&config, client_auth);
4b6aa8
-    if (opts & OPT_i)
4b6aa8
-    {
4b6aa8
-        g_list_free_full(config.ur_prefs.urp_auth_items, free);
4b6aa8
-        config.ur_prefs.urp_auth_items = auth_items;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (!config.ur_url)
4b6aa8
-        error_msg_and_die("You need to specify server URL");
4b6aa8
-
4b6aa8
-    post_state_t *post_state = NULL;
4b6aa8
-
4b6aa8
-    if (ureport_hash && ureport_hash_from_rt)
4b6aa8
-        error_msg_and_die("You need to pass either -a bthash or -A");
4b6aa8
-
4b6aa8
-    if (rhbz_bug >= 0 && rhbz_bug_from_rt)
4b6aa8
-        error_msg_and_die("You need to pass either -b bug-id or -B");
4b6aa8
-
4b6aa8
-    if (email_address && email_address_from_env)
4b6aa8
-        error_msg_and_die("You need to pass either -e bthash or -E");
4b6aa8
-
4b6aa8
-    if (ureport_hash_from_rt || rhbz_bug_from_rt)
4b6aa8
-    {
4b6aa8
-        dd = dd_opendir(dump_dir_path, DD_OPEN_READONLY);
4b6aa8
-        if (!dd)
4b6aa8
-            xfunc_die();
4b6aa8
-
4b6aa8
-        if (ureport_hash_from_rt)
4b6aa8
-        {
4b6aa8
-            report_result_t *ureport_result = find_in_reported_to(dd, "uReport");
4b6aa8
-
4b6aa8
-            if (!ureport_result || !ureport_result->bthash)
4b6aa8
-                error_msg_and_die(_("This problem does not have an uReport assigned."));
4b6aa8
-
4b6aa8
-            /* sorry, this will be leaked */
4b6aa8
-            ureport_hash = xstrdup(ureport_result->bthash);
4b6aa8
-
4b6aa8
-            free_report_result(ureport_result);
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        if (rhbz_bug_from_rt)
4b6aa8
-        {
4b6aa8
-            report_result_t *bz_result = find_in_reported_to(dd, "Bugzilla");
4b6aa8
-
4b6aa8
-            if (!bz_result || !bz_result->url)
4b6aa8
-                error_msg_and_die(_("This problem has not been reported to Bugzilla."));
4b6aa8
-
4b6aa8
-            char *bugid_ptr = strstr(bz_result->url, "show_bug.cgi?id=");
4b6aa8
-            if (!bugid_ptr)
4b6aa8
-                error_msg_and_die(_("Unable to find bug ID in bugzilla URL '%s'"), bz_result->url);
4b6aa8
-            bugid_ptr += strlen("show_bug.cgi?id=");
4b6aa8
-
4b6aa8
-            /* we're just reading int, sscanf works fine */
4b6aa8
-            if (sscanf(bugid_ptr, "%d", &rhbz_bug) != 1)
4b6aa8
-                error_msg_and_die(_("Unable to parse bug ID from bugzilla URL '%s'"), bz_result->url);
4b6aa8
-
4b6aa8
-            free_report_result(bz_result);
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        dd_close(dd);
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (email_address_from_env)
4b6aa8
-    {
4b6aa8
-        VALUE_FROM_CONF("ContactEmail", email_address, (const char *));
4b6aa8
-
4b6aa8
-        if (!email_address)
4b6aa8
-            error_msg_and_die(_("Neither environment variable 'uReport_ContactEmail' nor configuration option 'ContactEmail' is set"));
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    if (ureport_hash)
4b6aa8
-    {
4b6aa8
-        if (rhbz_bug < 0 && !email_address)
4b6aa8
-            error_msg_and_die(_("You need to specify bug ID, contact email or both"));
4b6aa8
-
4b6aa8
-        if (rhbz_bug >= 0)
4b6aa8
-        {
4b6aa8
-            if (perform_attach(&config, ureport_hash, (attach_handler)wrp_ureport_attach_rhbz, (void *)&rhbz_bug))
4b6aa8
-                goto finalize;
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        if (email_address)
4b6aa8
-        {
4b6aa8
-            if (perform_attach(&config, ureport_hash, (attach_handler)ureport_attach_email, (void *)email_address))
4b6aa8
-                goto finalize;
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        ret = 0;
4b6aa8
-        goto finalize;
4b6aa8
-    }
4b6aa8
-    if (!ureport_hash && (rhbz_bug >= 0 || email_address))
4b6aa8
-        error_msg_and_die(_("You need to specify bthash of the uReport to attach."));
4b6aa8
-
4b6aa8
-    /* -b, -a nor -r were specified - upload uReport from dump_dir */
4b6aa8
-    const char *server_url = config.ur_url;
4b6aa8
-    char *dest_url = concat_path_file(config.ur_url, REPORT_URL_SFX);
4b6aa8
-    config.ur_url = dest_url;
4b6aa8
-
4b6aa8
-    char *json_ureport = ureport_from_dump_dir_ext(dump_dir_path, &(config.ur_prefs));
4b6aa8
-    if (!json_ureport)
4b6aa8
-    {
4b6aa8
-        error_msg(_("Not uploading an empty uReport"));
4b6aa8
-        goto format_err;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    post_state = post_ureport(json_ureport, &config);
4b6aa8
-    free(json_ureport);
4b6aa8
-
4b6aa8
-    if (!post_state)
4b6aa8
-    {
4b6aa8
-        error_msg(_("Failed on submitting the problem"));
4b6aa8
-        goto format_err;
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    struct ureport_server_response *response = get_server_response(post_state, &config);
4b6aa8
-
4b6aa8
-    if (!response)
4b6aa8
-        goto format_err;
4b6aa8
-
4b6aa8
-    if (!response->is_error)
4b6aa8
-    {
4b6aa8
-        log_notice("is known: %s", response->value);
4b6aa8
-        ret = 0; /* "success" */
4b6aa8
-
4b6aa8
-        dd = dd_opendir(dump_dir_path, /* flags */ 0);
4b6aa8
-        if (!dd)
4b6aa8
-            xfunc_die();
4b6aa8
-
4b6aa8
-        if (response->bthash)
4b6aa8
-        {
4b6aa8
-            char *msg = xasprintf("uReport: BTHASH=%s", response->bthash);
4b6aa8
-            add_reported_to(dd, msg);
4b6aa8
-            free(msg);
4b6aa8
-
4b6aa8
-            char *bthash_url = concat_path_file(server_url, BTHASH_URL_SFX);
4b6aa8
-            msg = xasprintf("ABRT Server: URL=%s%s", bthash_url, response->bthash);
4b6aa8
-            add_reported_to(dd, msg);
4b6aa8
-            free(msg);
4b6aa8
-            free(bthash_url);
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        if (response->reported_to_list)
4b6aa8
-        {
4b6aa8
-            for (GList *e = response->reported_to_list; e; e = g_list_next(e))
4b6aa8
-                add_reported_to(dd, e->data);
4b6aa8
-        }
4b6aa8
-
4b6aa8
-        if (response->solution)
4b6aa8
-            dd_save_text(dd, FILENAME_NOT_REPORTABLE, response->solution);
4b6aa8
-
4b6aa8
-        dd_close(dd);
4b6aa8
-
4b6aa8
-        /* If a reported problem is not known then emit NEEDMORE */
4b6aa8
-        if (strcmp("true", response->value) == 0)
4b6aa8
-        {
4b6aa8
-            log(_("This problem has already been reported."));
4b6aa8
-            if (response->message)
4b6aa8
-                log(response->message);
4b6aa8
-
4b6aa8
-            ret = EXIT_STOP_EVENT_RUN;
4b6aa8
-        }
4b6aa8
-    }
4b6aa8
-    else
4b6aa8
-    {
4b6aa8
-        error_msg(_("Server responded with an error: '%s'"), response->value);
4b6aa8
-    }
4b6aa8
-
4b6aa8
-    free_ureport_server_response(response);
4b6aa8
-
4b6aa8
-format_err:
4b6aa8
-    free_post_state(post_state);
4b6aa8
-    free(dest_url);
4b6aa8
-
4b6aa8
-finalize:
4b6aa8
-    if (config.ur_prefs.urp_auth_items != auth_items)
4b6aa8
-        g_list_free_full(config.ur_prefs.urp_auth_items, free);
4b6aa8
-
4b6aa8
-    free_map_string(config.ur_http_headers);
4b6aa8
-
4b6aa8
-    free_map_string(settings);
4b6aa8
-    free(config.ur_client_cert);
4b6aa8
-    free(config.ur_client_key);
4b6aa8
-
4b6aa8
-    return ret;
4b6aa8
-}
4b6aa8
-- 
4b6aa8
1.8.3.1
4b6aa8