diff options
Diffstat (limited to 'src/plugins/KerneloopsReporter.cpp')
| -rw-r--r-- | src/plugins/KerneloopsReporter.cpp | 143 |
1 files changed, 143 insertions, 0 deletions
diff --git a/src/plugins/KerneloopsReporter.cpp b/src/plugins/KerneloopsReporter.cpp new file mode 100644 index 00000000..ae459737 --- /dev/null +++ b/src/plugins/KerneloopsReporter.cpp @@ -0,0 +1,143 @@ +/* + Copyright (C) 2010 ABRT team + Copyright (C) 2010 RedHat Inc + + 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; either version 2 of the License, or + (at your option) any later version. + + 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. + + Authors: + Anton Arapov <anton@redhat.com> + Arjan van de Ven <arjan@linux.intel.com> + */ + +#include "abrtlib.h" +#include "comm_layer_inner.h" +#include "abrt_exception.h" +#include "KerneloopsReporter.h" + +using namespace std; + +CKerneloopsReporter::CKerneloopsReporter() +{ + m_pSettings["SubmitURL"] = "http://submit.kerneloops.org/submitoops.php"; +} + +CKerneloopsReporter::~CKerneloopsReporter() +{ +} + +void CKerneloopsReporter::SetSettings(const map_plugin_settings_t& pSettings) +{ + /* Can't simply do this: + + m_pSettings = pSettings; + + * - it will erase keys which aren't present in pSettings. + * Example: if Bugzilla.conf doesn't have "Login = foo", + * then there's no pSettings["Login"] and m_pSettings = pSettings + * will nuke default m_pSettings["Login"] = "", + * making GUI think that we have no "Login" key at all + * and thus never overriding it - even if it *has* an override! + */ + + map_plugin_settings_t::iterator it = m_pSettings.begin(); + while (it != m_pSettings.end()) + { + map_plugin_settings_t::const_iterator override = pSettings.find(it->first); + if (override != pSettings.end()) + { + VERB3 log(" kerneloops settings[%s]='%s'", it->first.c_str(), it->second.c_str()); + it->second = override->second; + } + it++; + } +} + +string CKerneloopsReporter::Report(const map_crash_data_t& crash_data, + const map_plugin_settings_t& settings, + const char *args) +{ + /* abrt-action-kerneloops [-s] -c /etc/arbt/Kerneloops.conf -c - -d pCrashData.dir NULL */ + char *argv[9]; + char **pp = argv; + *pp++ = (char*)"abrt-action-kerneloops"; + +//We want to consume output, so don't redirect to syslog. +// if (logmode & LOGMODE_SYSLOG) +// *pp++ = (char*)"-s"; +//TODO: the actions<->daemon interaction will be changed anyway... + + *pp++ = (char*)"-c"; + *pp++ = (char*)(PLUGINS_CONF_DIR"/Kerneloops."PLUGINS_CONF_EXTENSION); + *pp++ = (char*)"-c"; + *pp++ = (char*)"-"; + *pp++ = (char*)"-d"; + *pp++ = (char*)get_crash_data_item_content_or_NULL(crash_data, CD_DUMPDIR); + *pp = NULL; + int pipefds[2]; + pid_t pid = fork_execv_on_steroids(EXECFLG_INPUT + EXECFLG_OUTPUT + EXECFLG_ERR2OUT, + argv, + pipefds, + /* unsetenv_vec: */ NULL, + /* dir: */ NULL, + /* uid(unused): */ 0 + ); + + /* Write the configuration to stdin */ + map_plugin_settings_t::const_iterator it = settings.begin(); + while (it != settings.end()) + { + full_write_str(pipefds[1], it->first.c_str()); + full_write_str(pipefds[1], "="); + full_write_str(pipefds[1], it->second.c_str()); + full_write_str(pipefds[1], "\n"); + it++; + } + close(pipefds[1]); + + FILE *fp = fdopen(pipefds[0], "r"); + if (!fp) + die_out_of_memory(); + + /* Consume log from stdout */ + string bug_status; + char *buf; + while ((buf = xmalloc_fgetline(fp)) != NULL) + { + if (strncmp(buf, "STATUS:", 7) == 0) + bug_status = buf + 7; + else + if (strncmp(buf, "EXCEPT:", 7) == 0) + { + CABRTException e(EXCEP_PLUGIN, "%s", buf + 7); + free(buf); + fclose(fp); + waitpid(pid, NULL, 0); + throw e; + } + update_client("%s", buf); + free(buf); + } + + fclose(fp); /* this also closes pipefds[0] */ + /* wait for child to actually exit, and prevent leaving a zombie behind */ + waitpid(pid, NULL, 0); + + return bug_status; +} + +PLUGIN_INFO(REPORTER, + CKerneloopsReporter, + "KerneloopsReporter", + "0.0.1", + _("Sends kernel oops information to kerneloops.org"), + "anton@redhat.com", + "http://people.redhat.com/aarapov", + PLUGINS_LIB_DIR"/KerneloopsReporter.glade"); |
