]> git.lyx.org Git - lyx.git/blob - src/support/environment.cpp
thinko
[lyx.git] / src / support / environment.cpp
1 /**
2  * \file environment.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Angus Leeming
7  * \author João Luis M. Assirati
8  * \author Lars Gullik Bjønnes
9  *
10  * Full author contact details are available in file CREDITS.
11  */
12
13 #include <config.h>
14
15 #include "support/environment.h"
16 #include "support/os.h"
17
18 #include <boost/tokenizer.hpp>
19
20 #include <cstdlib>
21 #include <map>
22 #include <sstream>
23
24 using std::string;
25 using std::vector;
26
27
28 namespace lyx {
29 namespace support {
30
31 string const getEnv(string const & envname)
32 {
33         // f.ex. what about error checking?
34         char const * const ch = getenv(envname.c_str());
35         return ch ? to_utf8(from_local8bit(ch)) : string();
36 }
37
38
39 vector<string> const getEnvPath(string const & name)
40 {
41         typedef boost::char_separator<char> Separator;
42         typedef boost::tokenizer<Separator> Tokenizer;
43
44         string const env_var = getEnv(name);
45         Separator const separator(string(1, os::path_separator()).c_str());
46         Tokenizer const tokens(env_var, separator);
47         Tokenizer::const_iterator it = tokens.begin();
48         Tokenizer::const_iterator const end = tokens.end();
49
50         std::vector<string> vars;
51         for (; it != end; ++it)
52                 vars.push_back(os::internal_path(*it));
53
54         return vars;
55 }
56
57
58 bool setEnv(string const & name, string const & value)
59 {
60         // CHECK Look at and fix this.
61         // f.ex. what about error checking?
62
63         string const encoded = to_local8bit(from_utf8(value));
64 #if defined (HAVE_SETENV)
65         return ::setenv(name.c_str(), encoded.c_str(), true) == 0;
66 #endif
67
68 #if defined (HAVE_PUTENV)
69         static std::map<string, char *> varmap;
70
71         string envstr = name + '=' + encoded;
72         char * newptr = new char[envstr.size() + 1];
73         envstr.copy(newptr, envstr.length());
74         newptr[envstr.length()] = '\0';
75         int const retval = ::putenv(newptr);
76
77         char * oldptr = varmap[name];
78         if (oldptr)
79                 delete oldptr;
80         varmap[name] = newptr;
81         return retval == 0;
82 #endif
83
84 #if !(defined HAVE_SETENV) && !(defined HAVE_PUTENV)
85 #error No environment-setting function has been defined.
86 #endif
87         return false;
88 }
89
90
91 void setEnvPath(string const & name, vector<string> const & env)
92 {
93         char const separator(os::path_separator());
94         std::ostringstream ss;
95         vector<string>::const_iterator const begin = env.begin();
96         vector<string>::const_iterator const end = env.end();
97         vector<string>::const_iterator it = begin;
98         for (; it != end; ++it) {
99                 if (it != begin)
100                         ss << separator;
101                 ss << os::external_path(*it);
102         }
103         setEnv(name, ss.str());
104 }
105
106
107 void prependEnvPath(string const & name, string const & prefix)
108 {
109         vector<string> env_var = getEnvPath(name);
110
111         typedef boost::char_separator<char> Separator;
112         typedef boost::tokenizer<Separator> Tokenizer;
113
114         Separator const separator(string(1, os::path_separator()).c_str());
115
116         // Prepend each new element to the list, removing identical elements
117         // that occur later in the list.
118         Tokenizer const tokens(prefix, separator);
119         vector<string> reversed_tokens(tokens.begin(), tokens.end());
120
121         typedef vector<string>::const_reverse_iterator token_iterator;
122         token_iterator it = reversed_tokens.rbegin();
123         token_iterator const end = reversed_tokens.rend();
124         for (; it != end; ++it) {
125                 vector<string>::iterator remove_it =
126                         std::remove(env_var.begin(), env_var.end(), *it);
127                 env_var.erase(remove_it, env_var.end());
128                 env_var.insert(env_var.begin(), *it);
129         }
130
131         setEnvPath(name, env_var);
132 }
133
134 } // namespace support
135 } // namespace lyx