/****************************************************************************** * Icinga 2 * * Copyright (C) 2012-2018 Icinga Development Team (https://icinga.com/) * * * * This program is free software; you can redistribute it and/or * * modify it under the terms of the GNU General Public License * * as published by the Free Software Foundation; 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 St, Fifth Floor, Boston, MA 02110-1301, USA. * ******************************************************************************/ #include "plugins/thresholds.hpp" #include #include #include using namespace boost::algorithm; threshold::threshold() : set(false) {} threshold::threshold(const double v, const double c, bool l , bool p ) { lower = v; upper = c; legal = l; perc = p; } threshold::threshold(const std::wstring& stri) { if (stri.empty()) throw std::invalid_argument("Threshold must not be empty"); std::wstring str = stri; //kill whitespace boost::algorithm::trim(str); bool low = (str.at(0) == L'!'); if (low) str = std::wstring(str.begin() + 1, str.end()); bool pc = false; if (str.at(0) == L'[' && str.at(str.length() - 1) == L']') {//is range str = std::wstring(str.begin() + 1, str.end() - 1); std::vector svec; boost::split(svec, str, boost::is_any_of(L"-")); if (svec.size() != 2) throw std::invalid_argument("Threshold range requires two arguments"); std::wstring str1 = svec.at(0), str2 = svec.at(1); if (str1.at(str1.length() - 1) == L'%' && str2.at(str2.length() - 1) == L'%') { pc = true; str1 = std::wstring(str1.begin(), str1.end() - 1); str2 = std::wstring(str2.begin(), str2.end() - 1); } try { boost::algorithm::trim(str1); lower = boost::lexical_cast(str1); boost::algorithm::trim(str2); upper = boost::lexical_cast(str2); legal = !low; perc = pc; set = true; } catch (const boost::bad_lexical_cast&) { throw std::invalid_argument("Unknown Threshold type"); } } else { //not range if (str.at(str.length() - 1) == L'%') { pc = true; str = std::wstring(str.begin(), str.end() - 1); } try { boost::algorithm::trim(str); lower = upper = boost::lexical_cast(str); legal = !low; perc = pc; set = true; } catch (const boost::bad_lexical_cast&) { throw std::invalid_argument("Unknown Threshold type"); } } } //return TRUE if the threshold is broken bool threshold::rend(const double val, const double max) { double upperAbs = upper; double lowerAbs = lower; if (perc) { upperAbs = upper / 100.0 * max; lowerAbs = lower / 100.0 * max; } if (!set) return set; if (lowerAbs == upperAbs) return val > upperAbs == legal; else return (val < lowerAbs || upperAbs < val) != legal; } //returns a printable string of the threshold std::wstring threshold::pString(const double max) { if (!set) return L""; //transform percentages to abolute values double lowerAbs = lower; double upperAbs = upper; if (perc) { lowerAbs = lower / 100.0 * max; upperAbs = upper / 100.0 * max; } std::wstring s, lowerStr = removeZero(lowerAbs), upperStr = removeZero(upperAbs); if (lower != upper) { s.append(L"[").append(lowerStr).append(L"-") .append(upperStr).append(L"]"); } else s.append(lowerStr); return s; } threshold threshold::toSeconds(const Tunit& fromUnit) { if (!set) return *this; double lowerAbs = lower; double upperAbs = upper; switch (fromUnit) { case TunitMS: lowerAbs = lowerAbs / 1000; upperAbs = upperAbs / 1000; break; case TunitS: lowerAbs = lowerAbs ; upperAbs = upperAbs ; break; case TunitM: lowerAbs = lowerAbs * 60; upperAbs = upperAbs * 60; break; case TunitH: lowerAbs = lowerAbs * 60 * 60; upperAbs = upperAbs * 60 * 60; break; } return threshold(lowerAbs, upperAbs, legal, perc); } std::wstring removeZero(double val) { std::wstring ret = boost::lexical_cast(val); std::wstring::size_type pos = ret.length(); if (ret.find_first_of(L".") == std::string::npos) return ret; for (std::wstring::reverse_iterator rit = ret.rbegin(); rit != ret.rend(); ++rit) { if (*rit == L'.') { return ret.substr(0, pos - 1); } if (*rit != L'0') { return ret.substr(0, pos); } pos--; } return L"0"; } std::vector splitMultiOptions(const std::wstring& str) { std::vector sVec; boost::split(sVec, str, boost::is_any_of(L",")); return sVec; } Bunit parseBUnit(const std::wstring& str) { std::wstring wstr = to_upper_copy(str); if (wstr == L"B") return BunitB; if (wstr == L"KB") return BunitkB; if (wstr == L"MB") return BunitMB; if (wstr == L"GB") return BunitGB; if (wstr == L"TB") return BunitTB; throw std::invalid_argument("Unknown unit type"); } std::wstring BunitStr(const Bunit& unit) { switch (unit) { case BunitB: return L"B"; case BunitkB: return L"kB"; case BunitMB: return L"MB"; case BunitGB: return L"GB"; case BunitTB: return L"TB"; } return NULL; } Tunit parseTUnit(const std::wstring& str) { std::wstring wstr = to_lower_copy(str); if (wstr == L"ms") return TunitMS; if (wstr == L"s") return TunitS; if (wstr == L"m") return TunitM; if (wstr == L"h") return TunitH; throw std::invalid_argument("Unknown unit type"); } std::wstring TunitStr(const Tunit& unit) { switch (unit) { case TunitMS: return L"ms"; case TunitS: return L"s"; case TunitM: return L"m"; case TunitH: return L"h"; } return NULL; } void printErrorInfo(unsigned long err) { if (!err) err = GetLastError(); LPWSTR mBuf = NULL; if (!FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, err, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPWSTR)&mBuf, 0, NULL)) std::wcout << "Failed to format error message, last error was: " << err << '\n'; else { boost::trim_right(std::wstring(mBuf)); std::wcout << mBuf << std::endl; } } std::wstring formatErrorInfo(unsigned long err) { std::wostringstream out; if (!err) err = GetLastError(); LPWSTR mBuf = NULL; if (!FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, err, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPWSTR)&mBuf, 0, NULL)) out << "Failed to format error message, last error was: " << err; else { std::wstring tempOut = std::wstring(mBuf); boost::trim_right(tempOut); out << tempOut; } return out.str(); } std::wstring stateToString(const state& state) { switch (state) { case OK: return L"OK"; case WARNING: return L"WARNING"; case CRITICAL: return L"CRITICAL"; default: return L"UNKNOWN"; } }