X-Git-Url: https://git.teslayout.com/public/public/public/?a=blobdiff_plain;f=source%2Fcore%2Futil%2FEnvironment.cpp;h=b9ed096b20823485c45606cc93cc78ce7d835709;hb=39033fd99e58e994a5e98c1060dcc79e0d81f9c9;hp=7b51fa1d751666f908422f7e63441407e0bb2bdb;hpb=bebea4009ed5a273fbf9ed3644a2140a8f477f99;p=anna.git diff --git a/source/core/util/Environment.cpp b/source/core/util/Environment.cpp index 7b51fa1..b9ed096 100644 --- a/source/core/util/Environment.cpp +++ b/source/core/util/Environment.cpp @@ -1,8 +1,8 @@ -// ANNA - Anna is Not 'N' Anymore +// ANNA - Anna is Not Nothingness Anymore // // (c) Copyright 2005-2014 Eduardo Ramos Testillano & Francisco Ruiz Rayo // -// https://bitbucket.org/testillano/anna +// http://redmine.teslayout.com/projects/anna-suite // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions @@ -14,7 +14,7 @@ // copyright notice, this list of conditions and the following disclaimer // in the documentation and/or other materials provided with the // distribution. -// * Neither the name of Google Inc. nor the names of its +// * Neither the name of the copyright holder nor the names of its // contributors may be used to endorse or promote products derived from // this software without specific prior written permission. // @@ -39,91 +39,109 @@ #include #include #include +#include -#include -#include - - -#include +#include // getenv / setenv +#include using namespace std; using namespace anna; +extern int errno; - -void Environment::initialize() throw() { +void Environment::initialize(char **envp) throw() { LOGMETHOD(TraceMethod tm("Environment", "initialize", ANNA_FILE_LOCATION)); - // clear data - a_managedVars.clear(); + a_vars.clear(); - // Register: - namespace po = boost::program_options; - po::options_description desc("Options"); + if(!envp) return; - FILE *fp; - char c_var[256]; - std::string var; + // register data + std::string assignment, var, val; - /* Open the command for reading. */ - fp = popen("env | cut -d'=' -f1", "r"); - if (fp == NULL) { - Logger::error("Failed to get environment variables list", ANNA_FILE_LOCATION); - return; - } + while(*envp) { + assignment = *envp; + std::size_t equalPos = assignment.find("="); - /* Read the output a line at a time - output it. */ - while (fgets(c_var, sizeof(c_var)-1, fp) != NULL) { - var = c_var; - boost::trim(var); - desc.add_options()(var.c_str(), var.c_str()); - a_managedVars[var] = ""; // temporary + if(equalPos != string::npos) { // protection + var = assignment.substr(0, equalPos - 1); + val = assignment.substr(equalPos, assignment.size() - 1); + a_vars[var] = val; + } + + envp++; } +} - /* close */ - pclose(fp); - - // Parsing: - po::variables_map vm; - try { - po::store(po::parse_environment(desc, [](const std::string& variable) { return variable; }), vm); // can throw - - std::map::const_iterator it; - std::string var, val; - for (it = a_managedVars.begin(); it != a_managedVars.end(); it++) { - var = (*it).first; - if (vm.count(var.c_str())) { // protection - val = vm[var.c_str()].as(); - a_managedVars[var] = val; - } +std::string Environment::getValue(const char* variableName, bool exceptionIfMissing) throw(RuntimeException) { + if(!variableName) + throw RuntimeException("Invalid NULL variable name!", ANNA_FILE_LOCATION); + + std::string var = variableName; + return getValue(var, exceptionIfMissing); +} + +std::string Environment::getValue(const std::string &variableName, bool exceptionIfMissing) throw(RuntimeException) { + std::string result = ""; + std::map::const_iterator it = a_vars.find(variableName); + + if(it == a_vars.end()) { + char *current = getenv(variableName.c_str()); + + if(!current) { + std::string msg = "The variable '"; + msg += variableName; + msg += "' is not defined in the environment."; + LOGDEBUG(Logger::debug(msg, ANNA_FILE_LOCATION)); + + if(exceptionIfMissing) throw RuntimeException(msg, ANNA_FILE_LOCATION); + } else { + // assignment + a_vars[variableName] = current; + result = current; } + } else { + result = it->second; + } + + return result; +} - po::notify(vm); - } catch (po::error& e) { - Logger::error(e.what(), ANNA_FILE_LOCATION); +void Environment::setVariable(const std::string &name, const std::string &value, bool overwrite) throw(RuntimeException) { + if(name == "") throw RuntimeException("Must provide non-empty variable name", ANNA_FILE_LOCATION); + + if(setenv(name.c_str(), value.c_str(), overwrite ? 1 : 0) != 0) { + std::string msg = "Cannot set the environment variable '"; + msg += name; + msg += "=\""; + msg += value; + msg += "\"'. The errno is "; + msg += anna::functions::asString(errno); + throw RuntimeException(msg, ANNA_FILE_LOCATION); } + + // optimization + if(overwrite) + a_vars[name] = value; } -std::string Environment::getValue (const char* variableName, bool exceptionIfMissing) const throw(RuntimeException) { - std::string result = ""; - if (!variableName) - throw RuntimeException("Invalid NULL variable name!", ANNA_FILE_LOCATION); +void Environment::unsetVariable(const std::string &name) throw(RuntimeException) { + if(name == "") throw RuntimeException("Must provide non-empty variable name", ANNA_FILE_LOCATION); - std::map::const_iterator it = a_managedVars.find(variableName); - if (it == a_managedVars.end()) { - std::string msg = "The variable '"; - msg += variableName; - msg += "' is not defined in the environment."; - LOGDEBUG(Logger::debug(msg, ANNA_FILE_LOCATION)); - if (exceptionIfMissing) - throw RuntimeException(msg, ANNA_FILE_LOCATION); - return ""; + if(unsetenv(name.c_str()) != 0) { + std::string msg = "Cannot unset the environment variable named '"; + msg += name; + msg += "'. The errno is "; + msg += anna::functions::asString(errno); + throw RuntimeException(msg, ANNA_FILE_LOCATION); } - return it->second; + std::map::iterator it = a_vars.find(name); + + if(it != a_vars.end()) a_vars.erase(it); }