blob: d989455a25a823e1243c828b01e7f640b89d2bc7 [file] [log] [blame]
Steve DiBenedettobb75b552014-02-08 12:12:11 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -07003 * Copyright (c) 2014 Regents of the University of California,
4 * Arizona Board of Regents,
5 * Colorado State University,
6 * University Pierre & Marie Curie, Sorbonne University,
7 * Washington University in St. Louis,
8 * Beijing Institute of Technology
9 *
10 * This file is part of NFD (Named Data Networking Forwarding Daemon).
11 * See AUTHORS.md for complete list of NFD authors and contributors.
12 *
13 * NFD is free software: you can redistribute it and/or modify it under the terms
14 * of the GNU General Public License as published by the Free Software Foundation,
15 * either version 3 of the License, or (at your option) any later version.
16 *
17 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
18 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
19 * PURPOSE. See the GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License along with
22 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
23 **/
Steve DiBenedettobb75b552014-02-08 12:12:11 -070024
Steve DiBenedettobb75b552014-02-08 12:12:11 -070025#include "config-file.hpp"
Alexander Afanasyev613e2a92014-04-15 13:36:58 -070026#include "logger.hpp"
Steve DiBenedettobb75b552014-02-08 12:12:11 -070027
28#include <boost/property_tree/info_parser.hpp>
Davide Pesavento52a18f92014-04-10 00:55:01 +020029#include <fstream>
Steve DiBenedettobb75b552014-02-08 12:12:11 -070030
31namespace nfd {
32
Steve DiBenedetto34c95f72014-04-17 20:56:00 -060033void
34ConfigFile::throwErrorOnUnknownSection(const std::string& filename,
35 const std::string& sectionName,
36 const ConfigSection& section,
37 bool isDryRun)
38{
39 std::string msg = "Error processing configuration file ";
40 msg += filename;
41 msg += ": no module subscribed for section \"" + sectionName + "\"";
42
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -070043 BOOST_THROW_EXCEPTION(ConfigFile::Error(msg));
Steve DiBenedetto34c95f72014-04-17 20:56:00 -060044}
45
46void
47ConfigFile::ignoreUnknownSection(const std::string& filename,
48 const std::string& sectionName,
49 const ConfigSection& section,
50 bool isDryRun)
51{
52 // do nothing
53}
54
Yanbiao Li698f4fe2015-08-19 16:30:16 -070055bool
56ConfigFile::parseYesNo(const ConfigSection::const_iterator& i,
57 const std::string& optionName,
58 const std::string& sectionName)
59{
60 const std::string value = i->second.get_value<std::string>();
61 if (value == "yes") {
62 return true;
63 }
64
65 if (value == "no") {
66 return false;
67 }
68
69 BOOST_THROW_EXCEPTION(ConfigFile::Error("Invalid value for option \"" +
70 optionName + "\" in \"" +
71 sectionName + "\" section"));
72}
73
Steve DiBenedetto34c95f72014-04-17 20:56:00 -060074ConfigFile::ConfigFile(UnknownConfigSectionHandler unknownSectionCallback)
75 : m_unknownSectionCallback(unknownSectionCallback)
Steve DiBenedettobb75b552014-02-08 12:12:11 -070076{
Steve DiBenedettobb75b552014-02-08 12:12:11 -070077}
78
79void
80ConfigFile::addSectionHandler(const std::string& sectionName,
Steve DiBenedetto1a3c6732014-03-13 06:44:05 -060081 ConfigSectionHandler subscriber)
Steve DiBenedettobb75b552014-02-08 12:12:11 -070082{
83 m_subscriptions[sectionName] = subscriber;
84}
85
86void
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -060087ConfigFile::parse(const std::string& filename, bool isDryRun)
Steve DiBenedettobb75b552014-02-08 12:12:11 -070088{
89 std::ifstream inputFile;
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -060090 inputFile.open(filename.c_str());
91 if (!inputFile.good() || !inputFile.is_open())
Steve DiBenedettobb75b552014-02-08 12:12:11 -070092 {
93 std::string msg = "Failed to read configuration file: ";
94 msg += filename;
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -070095 BOOST_THROW_EXCEPTION(Error(msg));
Steve DiBenedettobb75b552014-02-08 12:12:11 -070096 }
97 parse(inputFile, isDryRun, filename);
98 inputFile.close();
99}
100
101void
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -0600102ConfigFile::parse(const std::string& input, bool isDryRun, const std::string& filename)
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700103{
104 std::istringstream inputStream(input);
105 parse(inputStream, isDryRun, filename);
106}
107
108
109void
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -0600110ConfigFile::parse(std::istream& input, bool isDryRun, const std::string& filename)
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700111{
112 try
113 {
114 boost::property_tree::read_info(input, m_global);
115 }
116 catch (const boost::property_tree::info_parser_error& error)
117 {
118 std::stringstream msg;
119 msg << "Failed to parse configuration file";
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -0600120 msg << " " << filename;
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700121 msg << " " << error.message() << " line " << error.line();
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -0700122 BOOST_THROW_EXCEPTION(Error(msg.str()));
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700123 }
124
125 process(isDryRun, filename);
126}
127
128void
Alexander Afanasyevc0273e32015-01-06 13:05:50 -0800129ConfigFile::parse(const ConfigSection& config, bool isDryRun, const std::string& filename)
130{
131 m_global = config;
132 process(isDryRun, filename);
133}
134
135void
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -0600136ConfigFile::process(bool isDryRun, const std::string& filename)
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700137{
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -0600138 BOOST_ASSERT(!filename.empty());
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700139 // NFD_LOG_DEBUG("processing..." << ((isDryRun)?("dry run"):("")));
140
141 if (m_global.begin() == m_global.end())
142 {
Steve DiBenedetto34c95f72014-04-17 20:56:00 -0600143 std::string msg = "Error processing configuration file: ";
Steve DiBenedetto84da5bf2014-03-11 14:51:29 -0600144 msg += filename;
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700145 msg += " no data";
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -0700146 BOOST_THROW_EXCEPTION(Error(msg));
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700147 }
148
149 for (ConfigSection::const_iterator i = m_global.begin(); i != m_global.end(); ++i)
150 {
151 const std::string& sectionName = i->first;
152 const ConfigSection& section = i->second;
153
154 SubscriptionTable::iterator subscriberIt = m_subscriptions.find(sectionName);
155 if (subscriberIt != m_subscriptions.end())
156 {
Steve DiBenedetto1a3c6732014-03-13 06:44:05 -0600157 ConfigSectionHandler subscriber = subscriberIt->second;
158 subscriber(section, isDryRun, filename);
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700159 }
160 else
161 {
Steve DiBenedetto34c95f72014-04-17 20:56:00 -0600162 m_unknownSectionCallback(filename, sectionName, section, isDryRun);
Steve DiBenedettobb75b552014-02-08 12:12:11 -0700163 }
164 }
165}
166
167}