| #include "config_parser.hpp" |
| #include "xyz/openbmc_project/Common/error.hpp" |
| #include <phosphor-logging/log.hpp> |
| #include <phosphor-logging/elog-errors.hpp> |
| |
| #include <fstream> |
| #include <string> |
| #include <algorithm> |
| #include <unordered_map> |
| #include <regex> |
| #include <list> |
| |
| namespace phosphor |
| { |
| namespace network |
| { |
| namespace config |
| { |
| |
| using namespace phosphor::logging; |
| using namespace sdbusplus::xyz::openbmc_project::Common::Error; |
| |
| Parser::Parser(const fs::path& filePath) |
| { |
| setFile(filePath); |
| } |
| |
| |
| KeyValues Parser::getSection(const std::string& section) |
| { |
| auto it = sections.find(section); |
| if (it == sections.end()) |
| { |
| log<level::ERR>("ConfigParser: Section not found", |
| entry("SECTION=%s",section)); |
| elog<InternalFailure>(); |
| } |
| return it->second; |
| } |
| |
| std::vector<std::string> Parser::getValues(const std::string& section, |
| const std::string& key) |
| { |
| std::vector<std::string> values; |
| auto keyValues = getSection(section); |
| auto it = keyValues.find(key); |
| if (it == keyValues.end()) |
| { |
| log<level::ERR>("ConfigParser: Key not found", |
| entry("KEY=%s",key)); |
| elog<InternalFailure>(); |
| } |
| for (; it != keyValues.end() && key == it->first; it++) |
| { |
| values.push_back(it->second); |
| } |
| return values; |
| } |
| |
| |
| bool Parser::isValueExist(const std::string& section, const std::string& key, |
| const std::string& value) |
| { |
| try |
| { |
| auto values = getValues(section, key); |
| auto it = std::find(values.begin(), values.end(), value); |
| return it != std::end(values) ? true : false; |
| } |
| catch (InternalFailure& e) |
| { |
| commit<InternalFailure>(); |
| } |
| return false; |
| } |
| |
| void Parser::setValue(const std::string& section, const std::string& key, |
| const std::string& value) |
| { |
| KeyValues values; |
| auto it = sections.find(section); |
| if (it != sections.end()) |
| { |
| values = std::move(it->second); |
| } |
| values.insert(std::make_pair(key, value)); |
| |
| if (it != sections.end()) |
| { |
| it->second = std::move(values); |
| } |
| else |
| { |
| sections.insert(std::make_pair(section, std::move(values))); |
| } |
| } |
| |
| #if 0 |
| void Parser::print() |
| { |
| for (auto section : sections) |
| { |
| std::cout << "[" << section.first << "]\n\n"; |
| for (auto keyValue : section.second) |
| { |
| std::cout << keyValue.first << "=" << keyValue.second << "\n"; |
| } |
| } |
| } |
| #endif |
| |
| void Parser::setFile(const fs::path& filePath) |
| { |
| this->filePath = filePath; |
| std::fstream stream; |
| stream.open(filePath.string(), std::fstream::in); |
| |
| if (!stream.is_open()) |
| { |
| return; |
| } |
| //clear all the section data. |
| sections.clear(); |
| parse(stream); |
| stream.close(); |
| } |
| |
| void Parser::parse(std::istream& in) |
| { |
| static const std::regex commentRegex |
| { |
| R"x(\s*[;#])x" |
| }; |
| static const std::regex sectionRegex |
| { |
| R"x(\s*\[([^\]]+)\])x" |
| }; |
| static const std::regex valueRegex |
| { |
| R"x(\s*(\S[^ \t=]*)\s*=\s*(\S+)\s*$)x" |
| }; |
| std::string section; |
| std::smatch pieces; |
| for (std::string line; std::getline(in, line);) |
| { |
| if (line.empty() || std::regex_match(line, pieces, commentRegex)) |
| { |
| // skip comment lines and blank lines |
| } |
| else if (std::regex_match(line, pieces, sectionRegex)) |
| { |
| if (pieces.size() == 2) |
| { |
| section = pieces[1].str(); |
| } |
| } |
| else if (std::regex_match(line, pieces, valueRegex)) |
| { |
| if (pieces.size() == 3) |
| { |
| setValue(section, pieces[1].str(), pieces[2].str()); |
| } |
| } |
| } |
| } |
| |
| }//namespace config |
| }//namespace network |
| }//namespace phosphor |