/****************************************************************************** * Icinga 2 * * Copyright (C) 2012 Icinga Development Team (http://www.icinga.org/) * * * * 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 "i2-config.h" using std::ifstream; using namespace icinga; /** * Constructor for the ConfigCompiler class. * * @param path The path of the configuration file (or another name that * identifies the source of the configuration text). * @param input Input stream for the configuration file. * @param includeHandler Handler function for #include directives. */ ConfigCompiler::ConfigCompiler(const String& path, istream *input, HandleIncludeFunc includeHandler) : m_Path(path), m_Input(input), m_HandleInclude(includeHandler) { InitializeScanner(); } /** * Destructor for the ConfigCompiler class. */ ConfigCompiler::~ConfigCompiler(void) { DestroyScanner(); } /** * Reads data from the input stream. Used internally by the lexer. * * @param buffer Where to store data. * @param max_size The maximum number of bytes to read from the stream. * @returns The actual number of bytes read. */ size_t ConfigCompiler::ReadInput(char *buffer, size_t max_size) { m_Input->read(buffer, max_size); return static_cast(m_Input->gcount()); } /** * Retrieves the scanner object. * * @returns The scanner object. */ void *ConfigCompiler::GetScanner(void) const { return m_Scanner; } /** * Retrieves the result from the compiler. * * @returns A list of configuration items. */ vector ConfigCompiler::GetResult(void) const { return m_Result; } /** * Retrieves the path for the input file. * * @returns The path. */ String ConfigCompiler::GetPath(void) const { return m_Path; } /** * Handles an include directive by calling the include handler callback * function. * * @param include The path from the include directive. */ void ConfigCompiler::HandleInclude(const String& include) { String path = Utility::DirName(GetPath()) + "/" + include; vector items = m_HandleInclude(path); std::copy(items.begin(), items.end(), back_inserter(m_Result)); } /** * Compiles a stream. * * @param path A name identifying the stream. * @param stream The input stream. * @returns Configuration items. */ vector ConfigCompiler::CompileStream(const String& path, istream *stream) { ConfigCompiler ctx(path, stream); ctx.Compile(); return ctx.GetResult(); } /** * Compiles a file. * * @param path The path. * @returns Configuration items. */ vector ConfigCompiler::CompileFile(const String& path) { ifstream stream; stream.open(path.CStr(), ifstream::in); if (!stream) throw_exception(invalid_argument("Could not open config file: " + path)); Logger::Write(LogInformation, "config", "Compiling config file: " + path); return CompileStream(path, &stream); } /** * Compiles a snippet of text. * * @param path A name identifying the text. * @param text The text. * @returns Configuration items. */ vector ConfigCompiler::CompileText(const String& path, const String& text) { stringstream stream(text); return CompileStream(path, &stream); } /** * Default include handler. Includes the file and returns a list of * configuration items. * * @param include The path from the include directive. * @returns A list of configuration objects. */ vector ConfigCompiler::HandleFileInclude(const String& include) { /* TODO: implement wildcard includes */ return CompileFile(include); } /** * Adds an object to the result. * * @param object The configuration item. */ void ConfigCompiler::AddObject(const ConfigItem::Ptr& object) { m_Result.push_back(object); }