/* * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * You may obtain a copy of the Licence at * http://www.gnu.org/licences/lgpl.html * * As a special exception this file can also be included in modules * with other source code as long as that source code has been * released under an Open Source Initiative certificed licence. * More information about OSI certification can be found at: * http://www.opensource.org * * This module 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 Lesser General Public Licence for more details. * * This module was created by members of the firebird development * team. All individual contributions remain the Copyright (C) of * those individuals and all rights are reserved. Contributors to * this file are either listed below or can be obtained from a CVS * history command. * * Created by: Mark O'Donohue * * Contributor(s): */ #include "firebird.h" #include "../../common/config/config_file.h" #include "../jrd/os/fbsyslog.h" #ifdef HAVE_STDLIB_H #include #endif #include #include // Invalid or missing CONF_FILE may lead to severe errors // in applications. That's why for regular SERVER builds // it's better to exit with appropriate diags rather continue // with missing / wrong configuration. #if (! defined(BOOT_BUILD)) && (! defined(EMBEDDED)) && (! defined(SUPERCLIENT)) #define EXIT_ON_NO_CONF #define INFORM_ON_NO_CONF #else #undef EXIT_ON_NO_CONF #undef INFORM_ON_NO_CONF #endif // config_file works with OS case-sensitivity typedef Firebird::PathName string; /****************************************************************************** * * Allow case-insensitive comparison */ bool ConfigFile::key_compare::operator()(const string& x, const string& y) const { return x < y; } /****************************************************************************** * * Strip leading spaces */ void ConfigFile::stripLeadingWhiteSpace(string& s) { if (!s.size()) { return; } const string::size_type startPos = s.find_first_not_of(" \t\r"); if (startPos == string::npos) { s.erase(); // nothing but air } else if (startPos) { s = s.substr(startPos); } } /****************************************************************************** * * Strip trailing spaces */ void ConfigFile::stripTrailingWhiteSpace(string& s) { if (!s.size()) { return; } string::size_type endPos = s.find_last_not_of(" \t\r"); if (endPos != string::npos) { // Note that endPos is the index to the last non-ws char // why we have to inc. it ++endPos; s = s.substr(0, endPos); } } /****************************************************************************** * * Strip any comments */ void ConfigFile::stripComments(string& s) { // Note that this is only a hack. It won't work in case inputLine // contains hash-marks embedded in quotes! Not that I know if we // should care about that case. const string::size_type commentPos = s.find('#'); if (commentPos != string::npos) { s = s.substr(0, commentPos); } } /****************************************************************************** * * Check whether the given key exists or not */ bool ConfigFile::doesKeyExist(const string& key) { checkLoadConfig(); string data = getString(key); return !data.empty(); } /****************************************************************************** * * Return string value corresponding the given key */ string ConfigFile::getString(const string& key) { checkLoadConfig(); mymap_t::iterator lookup; lookup = parameters.find(key); if (lookup != parameters.end()) { return lookup->second; } return string(); } /****************************************************************************** * * Parse key */ string ConfigFile::parseKeyFrom(const string& inputLine, string::size_type& endPos) { endPos = inputLine.find_first_of("=\t"); if (endPos == string::npos) { return inputLine; } return inputLine.substr(0, endPos); } /****************************************************************************** * * Parse value */ string ConfigFile::parseValueFrom(string inputLine, string::size_type initialPos) { if (initialPos == string::npos) { return string(); } // skip leading white spaces unsigned int startPos = inputLine.find_first_not_of("= \t", initialPos); if (startPos == string::npos) { return string(); } stripTrailingWhiteSpace(inputLine); return inputLine.substr(startPos); } /****************************************************************************** * * Load file, if necessary */ void ConfigFile::checkLoadConfig() { if (!isLoadedFlg) { loadConfig(); } } /****************************************************************************** * * Load file immediately */ void ConfigFile::loadConfig() { isLoadedFlg = true; parameters.clear(); FILE* ifile = fopen(configFile.c_str(), "rt"); #ifdef EXIT_ON_NO_CONF int BadLinesCount = 0; #endif if (!ifile) { // config file does not exist #ifdef INFORM_ON_NO_CONF string Msg = "Missing configuration file: " + configFile; #ifdef EXIT_ON_NO_CONF if (fExitOnError) Msg += ", exiting"; #endif Firebird::Syslog::Record(fExitOnError ? Firebird::Syslog::Error : Firebird::Syslog::Warning, Msg.ToString()); #ifdef EXIT_ON_NO_CONF if (fExitOnError) exit(1); #endif #endif //INFORM_ON_NO_CONF return; } string inputLine; while (!feof(ifile)) { inputLine.LoadFromFile(ifile); stripComments(inputLine); stripLeadingWhiteSpace(inputLine); if (!inputLine.size()) { continue; // comment-line or empty line } if (inputLine.find('=') == string::npos) { Firebird::string Msg = (configFile + ": illegal line \"" + inputLine + "\"").ToString(); Firebird::Syslog::Record(fExitOnError ? Firebird::Syslog::Error : Firebird::Syslog::Warning, Msg); #ifdef EXIT_ON_NO_CONF BadLinesCount++; #endif continue; } string::size_type endPos; string key = parseKeyFrom(inputLine, endPos); stripTrailingWhiteSpace(key); // TODO: here we must check for correct parameter spelling ! string value = parseValueFrom(inputLine, endPos); // parameters.insert(pair(key, value)); // Just to display yet another template function parameters.insert(std::make_pair(key, value)); } #ifdef EXIT_ON_NO_CONF if (BadLinesCount && fExitOnError) { exit(1); } #endif }