xref: /netbsd-src/external/bsd/kyua-cli/dist/utils/env.cpp (revision 6b3a42af15b5e090c339512c790dd68f3d11a9d8)
1*6b3a42afSjmmv // Copyright 2010 Google Inc.
2*6b3a42afSjmmv // All rights reserved.
3*6b3a42afSjmmv //
4*6b3a42afSjmmv // Redistribution and use in source and binary forms, with or without
5*6b3a42afSjmmv // modification, are permitted provided that the following conditions are
6*6b3a42afSjmmv // met:
7*6b3a42afSjmmv //
8*6b3a42afSjmmv // * Redistributions of source code must retain the above copyright
9*6b3a42afSjmmv //   notice, this list of conditions and the following disclaimer.
10*6b3a42afSjmmv // * Redistributions in binary form must reproduce the above copyright
11*6b3a42afSjmmv //   notice, this list of conditions and the following disclaimer in the
12*6b3a42afSjmmv //   documentation and/or other materials provided with the distribution.
13*6b3a42afSjmmv // * Neither the name of Google Inc. nor the names of its contributors
14*6b3a42afSjmmv //   may be used to endorse or promote products derived from this software
15*6b3a42afSjmmv //   without specific prior written permission.
16*6b3a42afSjmmv //
17*6b3a42afSjmmv // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*6b3a42afSjmmv // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*6b3a42afSjmmv // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20*6b3a42afSjmmv // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21*6b3a42afSjmmv // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22*6b3a42afSjmmv // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23*6b3a42afSjmmv // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*6b3a42afSjmmv // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*6b3a42afSjmmv // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*6b3a42afSjmmv // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27*6b3a42afSjmmv // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*6b3a42afSjmmv 
29*6b3a42afSjmmv #include "utils/env.hpp"
30*6b3a42afSjmmv 
31*6b3a42afSjmmv #if defined(HAVE_CONFIG_H)
32*6b3a42afSjmmv #  include "config.h"
33*6b3a42afSjmmv #endif
34*6b3a42afSjmmv 
35*6b3a42afSjmmv #include <cerrno>
36*6b3a42afSjmmv #include <cstdlib>
37*6b3a42afSjmmv #include <cstring>
38*6b3a42afSjmmv #include <stdexcept>
39*6b3a42afSjmmv 
40*6b3a42afSjmmv #include "utils/format/macros.hpp"
41*6b3a42afSjmmv #include "utils/logging/macros.hpp"
42*6b3a42afSjmmv #include "utils/optional.ipp"
43*6b3a42afSjmmv 
44*6b3a42afSjmmv using utils::none;
45*6b3a42afSjmmv using utils::optional;
46*6b3a42afSjmmv 
47*6b3a42afSjmmv 
48*6b3a42afSjmmv extern "C" {
49*6b3a42afSjmmv     extern char** environ;
50*6b3a42afSjmmv }
51*6b3a42afSjmmv 
52*6b3a42afSjmmv 
53*6b3a42afSjmmv /// Gets all environment variables.
54*6b3a42afSjmmv ///
55*6b3a42afSjmmv /// \return A mapping of (name, value) pairs describing the environment
56*6b3a42afSjmmv /// variables.
57*6b3a42afSjmmv std::map< std::string, std::string >
getallenv(void)58*6b3a42afSjmmv utils::getallenv(void)
59*6b3a42afSjmmv {
60*6b3a42afSjmmv     std::map< std::string, std::string > allenv;
61*6b3a42afSjmmv     for (char** envp = environ; *envp != NULL; envp++) {
62*6b3a42afSjmmv         const std::string oneenv = *envp;
63*6b3a42afSjmmv         const std::string::size_type pos = oneenv.find('=');
64*6b3a42afSjmmv         const std::string name = oneenv.substr(0, pos);
65*6b3a42afSjmmv         const std::string value = oneenv.substr(pos + 1);
66*6b3a42afSjmmv 
67*6b3a42afSjmmv         PRE(allenv.find(name) == allenv.end());
68*6b3a42afSjmmv         allenv[name] = value;
69*6b3a42afSjmmv     }
70*6b3a42afSjmmv     return allenv;
71*6b3a42afSjmmv }
72*6b3a42afSjmmv 
73*6b3a42afSjmmv 
74*6b3a42afSjmmv /// Gets the value of an environment variable.
75*6b3a42afSjmmv ///
76*6b3a42afSjmmv /// \param name The name of the environment variable to query.
77*6b3a42afSjmmv ///
78*6b3a42afSjmmv /// \return The value of the environment variable if it is defined, or none
79*6b3a42afSjmmv /// otherwise.
80*6b3a42afSjmmv optional< std::string >
getenv(const std::string & name)81*6b3a42afSjmmv utils::getenv(const std::string& name)
82*6b3a42afSjmmv {
83*6b3a42afSjmmv     const char* value = std::getenv(name.c_str());
84*6b3a42afSjmmv     if (value == NULL) {
85*6b3a42afSjmmv         LD(F("Environment variable '%s' is not defined") % name);
86*6b3a42afSjmmv         return none;
87*6b3a42afSjmmv     } else {
88*6b3a42afSjmmv         LD(F("Environment variable '%s' is '%s'") % name % value);
89*6b3a42afSjmmv         return utils::make_optional(std::string(value));
90*6b3a42afSjmmv     }
91*6b3a42afSjmmv }
92*6b3a42afSjmmv 
93*6b3a42afSjmmv 
94*6b3a42afSjmmv /// Gets the value of an environment variable with a default fallback.
95*6b3a42afSjmmv ///
96*6b3a42afSjmmv /// \param name The name of the environment variable to query.
97*6b3a42afSjmmv /// \param default_value The value to return if the variable is not defined.
98*6b3a42afSjmmv ///
99*6b3a42afSjmmv /// \return The value of the environment variable.
100*6b3a42afSjmmv std::string
getenv_with_default(const std::string & name,const std::string & default_value)101*6b3a42afSjmmv utils::getenv_with_default(const std::string& name,
102*6b3a42afSjmmv                            const std::string& default_value)
103*6b3a42afSjmmv {
104*6b3a42afSjmmv     const char* value = std::getenv(name.c_str());
105*6b3a42afSjmmv     if (value == NULL) {
106*6b3a42afSjmmv         LD(F("Environment variable '%s' is not defined; using default '%s'") %
107*6b3a42afSjmmv            name % default_value);
108*6b3a42afSjmmv         return default_value;
109*6b3a42afSjmmv     } else {
110*6b3a42afSjmmv         LD(F("Environment variable '%s' is '%s'") % name % value);
111*6b3a42afSjmmv         return value;
112*6b3a42afSjmmv     }
113*6b3a42afSjmmv }
114*6b3a42afSjmmv 
115*6b3a42afSjmmv 
116*6b3a42afSjmmv /// Sets the value of an environment variable.
117*6b3a42afSjmmv ///
118*6b3a42afSjmmv /// \param name The name of the environment variable to set.
119*6b3a42afSjmmv /// \param val The value to set the environment variable to.  May be empty.
120*6b3a42afSjmmv ///
121*6b3a42afSjmmv /// \throw std::runtime_error If there is an error setting the environment
122*6b3a42afSjmmv ///     variable.
123*6b3a42afSjmmv void
setenv(const std::string & name,const std::string & val)124*6b3a42afSjmmv utils::setenv(const std::string& name, const std::string& val)
125*6b3a42afSjmmv {
126*6b3a42afSjmmv     LD(F("Setting environment variable '%s' to '%s'") % name % val);
127*6b3a42afSjmmv #if defined(HAVE_SETENV)
128*6b3a42afSjmmv     if (::setenv(name.c_str(), val.c_str(), 1) == -1) {
129*6b3a42afSjmmv         const int original_errno = errno;
130*6b3a42afSjmmv         throw std::runtime_error(F("Failed to set environment variable '%s' to "
131*6b3a42afSjmmv                                    "'%s': %s") %
132*6b3a42afSjmmv                                  name % val % std::strerror(original_errno));
133*6b3a42afSjmmv     }
134*6b3a42afSjmmv #elif defined(HAVE_PUTENV)
135*6b3a42afSjmmv     if (::putenv((F("%s=%s") % name % val).c_str()) == -1) {
136*6b3a42afSjmmv         const int original_errno = errno;
137*6b3a42afSjmmv         throw std::runtime_error(F("Failed to set environment variable '%s' to "
138*6b3a42afSjmmv                                    "'%s': %s") %
139*6b3a42afSjmmv                                  name % val % std::strerror(original_errno));
140*6b3a42afSjmmv     }
141*6b3a42afSjmmv #else
142*6b3a42afSjmmv #   error "Don't know how to set an environment variable."
143*6b3a42afSjmmv #endif
144*6b3a42afSjmmv }
145*6b3a42afSjmmv 
146*6b3a42afSjmmv 
147*6b3a42afSjmmv /// Unsets an environment variable.
148*6b3a42afSjmmv ///
149*6b3a42afSjmmv /// \param name The name of the environment variable to unset.
150*6b3a42afSjmmv ///
151*6b3a42afSjmmv /// \throw std::runtime_error If there is an error unsetting the environment
152*6b3a42afSjmmv ///     variable.
153*6b3a42afSjmmv void
unsetenv(const std::string & name)154*6b3a42afSjmmv utils::unsetenv(const std::string& name)
155*6b3a42afSjmmv {
156*6b3a42afSjmmv     LD(F("Unsetting environment variable '%s'") % name);
157*6b3a42afSjmmv #if defined(HAVE_UNSETENV)
158*6b3a42afSjmmv     if (::unsetenv(name.c_str()) == -1) {
159*6b3a42afSjmmv         const int original_errno = errno;
160*6b3a42afSjmmv         throw std::runtime_error(F("Failed to unset environment variable "
161*6b3a42afSjmmv                                    "'%s'") %
162*6b3a42afSjmmv                                  name % std::strerror(original_errno));
163*6b3a42afSjmmv     }
164*6b3a42afSjmmv #elif defined(HAVE_PUTENV)
165*6b3a42afSjmmv     if (::putenv((F("%s=") % name).c_str()) == -1) {
166*6b3a42afSjmmv         const int original_errno = errno;
167*6b3a42afSjmmv         throw std::runtime_error(F("Failed to unset environment variable "
168*6b3a42afSjmmv                                    "'%s'") %
169*6b3a42afSjmmv                                  name % std::strerror(original_errno));
170*6b3a42afSjmmv     }
171*6b3a42afSjmmv #else
172*6b3a42afSjmmv #   error "Don't know how to unset an environment variable."
173*6b3a42afSjmmv #endif
174*6b3a42afSjmmv }
175