/* MiddleWare.cpp Copyright (C) 2009 Zdenek Prikryl (zprikryl@redhat.com) Copyright (C) 2009 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. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "MiddleWare.h" #include "DebugDump.h" #include "Settings.h" CMiddleWare::CMiddleWare(const std::string& pPlugisConfDir, const std::string& pPlugisLibDir, const std::string& pMiddleWareConfFile) : m_pPluginManager(NULL), m_bOpenGPGCheck(true) { m_pPluginManager = new CPluginManager(pPlugisConfDir, pPlugisLibDir); if (m_pPluginManager == NULL) { throw std::string("Not enought memory."); } m_pPluginManager->LoadPlugins(); LoadSettings(pMiddleWareConfFile); set_enabled_plugins_t::iterator it_p; for (it_p = m_setEnabledPlugins.begin(); it_p != m_setEnabledPlugins.end(); it_p++) { m_pPluginManager->RegisterPlugin(*it_p); } } CMiddleWare::~CMiddleWare() { m_pPluginManager->UnLoadPlugins(); delete m_pPluginManager; } void CMiddleWare::LoadSettings(const std::string& pPath) { map_settings_t settings; load_settings(pPath, settings); if (settings.find("BlackList") != settings.end()) { parse_settings(settings["BlackList"], m_setBlackList); } if (settings.find("EnabledPlugins") != settings.end()) { parse_settings(settings["EnabledPlugins"], m_setEnabledPlugins); } if (settings.find("OpenGPGPublicKeys") != settings.end()) { parse_settings(settings["OpenGPGPublicKeys"], m_setOpenGPGKeys); set_opengpg_keys_t::iterator it_k; for (it_k = m_setOpenGPGKeys.begin(); it_k != m_setOpenGPGKeys.end(); it_k++) { m_RPMInfo.LoadOpenGPGPublicKey(*it_k); } } if (settings.find("EnableOpenGPG") != settings.end()) { m_bOpenGPGCheck = settings["EnableOpenGPG"] == "yes"; } if (settings.find("Database") != settings.end()) { m_sDatabase = settings["Database"]; if (m_setEnabledPlugins.find(m_sDatabase) == m_setEnabledPlugins.end()) { throw std::string("Database plugin '"+m_sDatabase+"' isn't enabled."); } } else { throw std::string("No database plugin is selected."); } set_enabled_plugins_t::iterator it_p; for (it_p = m_setEnabledPlugins.begin(); it_p != m_setEnabledPlugins.end(); it_p++) { if (settings.find(*it_p) != settings.end()) { set_reporters_t reporters; parse_settings(settings[*it_p], reporters); m_mapPlugin2Reporters[*it_p] = reporters; } } } void CMiddleWare::DebugDump2Report(const std::string& pDebugDumpDir, crash_report_t& pCrashReport) { CDebugDump dd; dd.Open(pDebugDumpDir); dd.LoadText(FILENAME_UUID, pCrashReport.m_sUUID); dd.LoadText(FILENAME_ARCHITECTURE, pCrashReport.m_sArchitecture); dd.LoadText(FILENAME_KERNEL, pCrashReport.m_sKernel); dd.LoadText(FILENAME_PACKAGE, pCrashReport.m_sPackage); dd.LoadText(FILENAME_EXECUTABLE, pCrashReport.m_sExecutable); dd.LoadText(FILENAME_CMDLINE, pCrashReport.m_sCmdLine); dd.LoadText(FILENAME_RELEASE, pCrashReport.m_sRelease); if (dd.Exist(FILENAME_TEXTDATA1)) { dd.LoadText(FILENAME_TEXTDATA1, pCrashReport.m_sTextData1); } if (dd.Exist(FILENAME_TEXTDATA2)) { dd.LoadText(FILENAME_TEXTDATA2, pCrashReport.m_sTextData2); } if (dd.Exist(FILENAME_BINARYDATA1)) { pCrashReport.m_sBinaryData1 = pDebugDumpDir + "/" + FILENAME_BINARYDATA1; } if (dd.Exist(FILENAME_BINARYDATA2)) { pCrashReport.m_sBinaryData2 = pDebugDumpDir + "/" + FILENAME_BINARYDATA2; } dd.Close(); } void CMiddleWare::RegisterPlugin(const std::string& pName) { m_pPluginManager->RegisterPlugin(pName); } void CMiddleWare::UnRegisterPlugin(const std::string& pName) { m_pPluginManager->UnRegisterPlugin(pName); } std::string CMiddleWare::GetLocalUUIDLanguage(const std::string& pLanguage, const std::string& pDebugDumpDir) { CLanguage* language = m_pPluginManager->GetLanguage(pLanguage); return language->GetLocalUUID(pDebugDumpDir); } std::string CMiddleWare::GetGlobalUUIDLanguage(const std::string& pLanguage, const std::string& pDebugDumpDir) { CLanguage* language = m_pPluginManager->GetLanguage(pLanguage); return language->GetGlobalUUID(pDebugDumpDir); } void CMiddleWare::CreateReportLanguage(const std::string& pLanguage, const std::string& pDebugDumpDir) { CLanguage* language = m_pPluginManager->GetLanguage(pLanguage); return language->CreateReport(pDebugDumpDir); } std::string CMiddleWare::GetLocalUUIDApplication(const std::string& pApplication, const std::string& pDebugDumpDir) { CApplication* application = m_pPluginManager->GetApplication(pApplication); return application->GetLocalUUID(pDebugDumpDir); } std::string CMiddleWare::GetGlobalUUIDApplication(const std::string& pApplication, const std::string& pDebugDumpDir) { CApplication* application = m_pPluginManager->GetApplication(pApplication); return application->GetGlobalUUID(pDebugDumpDir); } void CMiddleWare::CreateReportApplication(const std::string& pApplication, const std::string& pDebugDumpDir) { CApplication* application = m_pPluginManager->GetApplication(pApplication); return application->CreateReport(pDebugDumpDir); } void CMiddleWare::CreateReport(const std::string& pUUID, const std::string& pUID, crash_report_t& pCrashReport) { CDatabase* database = m_pPluginManager->GetDatabase(m_sDatabase); database_row_t row; database->Connect(); row = database->GetUUIDData(pUUID, pUID); database->DisConnect(); if (pUUID == "" || row.m_sUUID != pUUID) { throw std::string("CMiddleWare::CreateReport(): UUID '"+pUUID+"' is not in database."); } std::string appLan; std::string UUID; CDebugDump dd; dd.Open(row.m_sDebugDumpDir); if (dd.Exist(FILENAME_APPLICATION)) { dd.LoadText(FILENAME_APPLICATION, appLan); CreateReportApplication(appLan, row.m_sDebugDumpDir); UUID = GetGlobalUUIDApplication(appLan, row.m_sDebugDumpDir); } if (dd.Exist(FILENAME_LANGUAGE)) { dd.LoadText(FILENAME_LANGUAGE, appLan); CreateReportLanguage(appLan, row.m_sDebugDumpDir); UUID = GetGlobalUUIDLanguage(appLan, row.m_sDebugDumpDir); } dd.SaveText(FILENAME_UUID, UUID); dd.Close(); DebugDump2Report(row.m_sDebugDumpDir, pCrashReport); pCrashReport.m_sMWID = appLan + ";" + pUID + ";" + pUUID ; } void CMiddleWare::Report(const crash_report_t& pCrashReport) { std::string::size_type pos1 = 0; std::string::size_type pos2 = pCrashReport.m_sMWID.find(";", pos1); std::string lanAppPlugin = pCrashReport.m_sMWID.substr(pos1, pos2); pos1 = pos2 + 1; pos2 = pCrashReport.m_sMWID.find(";", pos1); std::string UID = pCrashReport.m_sMWID.substr(pos1, pos2 - pos1); pos1 = pos2 + 1; std::string UUID = pCrashReport.m_sMWID.substr(pos1);; CDatabase* database = m_pPluginManager->GetDatabase(m_sDatabase); database->Connect(); database->SetReported(UUID, UID); database->DisConnect(); if (m_mapPlugin2Reporters.find(lanAppPlugin) != m_mapPlugin2Reporters.end()) { set_reporters_t::iterator it_r; for (it_r = m_mapPlugin2Reporters[lanAppPlugin].begin(); it_r != m_mapPlugin2Reporters[lanAppPlugin].end(); it_r++) { CReporter* reporter = m_pPluginManager->GetReporter(*it_r); reporter->Report(pCrashReport); } } } void CMiddleWare::DeleteDebugDump(const std::string& pUUID, const std::string& pUID) { database_row_t row; CDatabase* database = m_pPluginManager->GetDatabase(m_sDatabase); database->Connect(); row = database->GetUUIDData(pUUID, pUID); database->Delete(pUUID, pUID); database->DisConnect(); CDebugDump dd; dd.Open(row.m_sDebugDumpDir); dd.Delete(); dd.Close(); } int CMiddleWare::SaveDebugDump(const std::string& pDebugDumpDir, crash_info_t& pCrashInfo) { CDatabase* database = m_pPluginManager->GetDatabase(m_sDatabase); std::string UUID; std::string UID; std::string package; std::string description; std::string executable; std::string time; CDebugDump dd; dd.Open(pDebugDumpDir); dd.LoadText(FILENAME_EXECUTABLE, executable); package = m_RPMInfo.GetPackage(executable, description); if (executable != "kernel") { std::string packageName = package.substr(0, package.rfind("-", package.rfind("-") - 1)); if (packageName == "" || (m_setBlackList.find(packageName) != m_setBlackList.end())) { dd.Delete(); dd.Close(); return 0; } if (m_bOpenGPGCheck) { if (!m_RPMInfo.CheckFingerprint(packageName) || !m_RPMInfo.CheckHash(packageName, executable)) { dd.Delete(); dd.Close(); return 0; } } } else { package = "kernel"; description = "The Linux kernel" description += "\n\n"; description += "The kernel contains the Linux kernel (vmlinuz), the core of any" "Linux operating system. The kernel handles the basic functions" "of the operating system: memory allocation, process allocation, device" "input and output, etc." } dd.SaveText(FILENAME_PACKAGE, package); dd.SaveText(FILENAME_DESCRIPTION, description); if (dd.Exist(FILENAME_APPLICATION)) { std::string application; dd.LoadText(FILENAME_APPLICATION, application); UUID = GetLocalUUIDApplication(application, pDebugDumpDir); } if (dd.Exist(FILENAME_LANGUAGE)) { std::string language; dd.LoadText(FILENAME_LANGUAGE, language); UUID = GetLocalUUIDLanguage(language, pDebugDumpDir); } if (UUID == "") { throw std::string("CMiddleWare::SaveDebugDumpToDataBase(): Wrong UUID."); } dd.LoadText(FILENAME_TIME, time); dd.LoadText(FILENAME_UID, UID); database_row_t row; database->Connect(); database->Insert(UUID, UID, pDebugDumpDir, time); row = database->GetUUIDData(UUID, UID); database->DisConnect(); if (row.m_sReported == "1") { dd.Delete(); dd.Close(); return 0; } if (row.m_sCount != "1") { dd.Delete(); } dd.Close(); pCrashInfo.m_sUUID = UUID; pCrashInfo.m_sUID = UID; pCrashInfo.m_sCount = row.m_sCount; pCrashInfo.m_sExecutable = executable; pCrashInfo.m_sPackage = package; pCrashInfo.m_sTime = row.m_sTime; pCrashInfo.m_sReported = "0"; return 1; } vector_crash_infos_t CMiddleWare::GetCrashInfos(const std::string& pUID) { CDatabase* database = m_pPluginManager->GetDatabase(m_sDatabase); vector_database_rows_t rows; database->Connect(); rows = database->GetUIDData(pUID); database->DisConnect(); vector_crash_infos_t infos; std::string data; int ii; for (ii = 0; ii < rows.size(); ii++) { crash_info_t info; CDebugDump dd; info.m_sUUID = rows[ii].m_sUUID; info.m_sUID = rows[ii].m_sUID; info.m_sCount = rows[ii].m_sCount; info.m_sTime = rows[ii].m_sTime; info.m_sReported = rows[ii].m_sReported; dd.Open(rows[ii].m_sDebugDumpDir); dd.LoadText(FILENAME_EXECUTABLE, data); info.m_sExecutable = data; dd.LoadText(FILENAME_PACKAGE, data); info.m_sPackage = data; dd.LoadText(FILENAME_DESCRIPTION, data); info.m_sDescription = data; dd.Close(); infos.push_back(info); } return infos; }