Skip to content

Commit

Permalink
Move env::copy yo utility and fix encoding
Browse files Browse the repository at this point in the history
  • Loading branch information
AntoinePrv committed Nov 10, 2023
1 parent ed27c71 commit 24fa539
Show file tree
Hide file tree
Showing 7 changed files with 106 additions and 63 deletions.
4 changes: 2 additions & 2 deletions libmamba/include/mamba/core/activation.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@
#ifndef MAMBA_CORE_ACTIVATION_HPP
#define MAMBA_CORE_ACTIVATION_HPP

#include <map>
#include <string>
#include <unordered_map>
#include <utility>
#include <vector>

Expand Down Expand Up @@ -101,7 +101,7 @@ namespace mamba
bool m_stack = false;
ActivationType m_action;

std::map<std::string, std::string> m_env;
std::unordered_map<std::string, std::string> m_env;
};

class PosixActivator : public Activator
Expand Down
2 changes: 0 additions & 2 deletions libmamba/include/mamba/core/environment.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@
#ifndef MAMBA_CORE_ENVIRONMENT_HPP
#define MAMBA_CORE_ENVIRONMENT_HPP

#include <map>
#include <string>
#include <vector>

Expand All @@ -20,7 +19,6 @@ namespace mamba::env

auto which(const std::string& exe, const std::string& override_path = "") -> fs::u8path;
auto which(const std::string& exe, const std::vector<fs::u8path>& search_paths) -> fs::u8path;
auto copy() -> std::map<std::string, std::string>;
auto platform() -> std::string;
auto home_directory() -> fs::u8path;
auto user_config_dir() -> fs::u8path;
Expand Down
5 changes: 5 additions & 0 deletions libmamba/include/mamba/util/environment.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,16 @@

#include <optional>
#include <string>
#include <unordered_map>

namespace mamba::util
{
auto get_env(const std::string& key) -> std::optional<std::string>;

void set_env(const std::string& key, const std::string& value);

void unset_env(const std::string& key);

auto get_full_env() -> std::unordered_map<std::string, std::string>;
}
#endif
2 changes: 1 addition & 1 deletion libmamba/src/core/activation.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ namespace mamba

Activator::Activator(const Context& context)
: m_context(context)
, m_env(env::copy())
, m_env(util::get_full_env())
{
}

Expand Down
42 changes: 0 additions & 42 deletions libmamba/src/core/environment.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -94,48 +94,6 @@ namespace mamba::env
return ""; // empty path
}

std::map<std::string, std::string> copy()
{
std::map<std::string, std::string> m;
#ifndef _WIN32
int i = 1;
const char* c = *environ;
for (; c; i++)
{
std::string_view s(c);
auto pos = s.find("=");
m[std::string(s.substr(0, pos))] = (pos != s.npos) ? std::string(s.substr(pos + 1)) : "";
c = *(environ + i);
}
#else

// inspired by
// https://github.com/gennaroprota/breath/blob/0709a9f0fe4e745b1d9fc44ab65d92853820b515
// /breath/environment/brt/dep/syst/windows/get_environment_map.cpp#L38-L80
char* start = GetEnvironmentStrings();
if (start == nullptr)
{
throw std::runtime_error("GetEnvironmentStrings() failed");
}

char* current = start;
while (*current != '\0')
{
std::string_view s = current;
auto pos = s.find("=");
assert(pos != std::string_view::npos);
std::string key = util::to_upper(s.substr(0, pos));
if (!key.empty())
{
std::string_view value = (pos != s.npos) ? s.substr(pos + 1) : "";
m[std::string(key)] = value;
}
current += s.size() + 1;
}
#endif
return m;
}

std::string platform()
{
#ifndef _WIN32
Expand Down
70 changes: 70 additions & 0 deletions libmamba/src/util/environment.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@

#ifdef _WIN32

#include <cassert>
#include <mutex>
#include <stdexcept>

Expand Down Expand Up @@ -92,6 +93,51 @@ namespace mamba::util
{
set_env(key, "");
}

auto get_full_env() -> std::unordered_map<std::string, std::string>
{
static constexpr auto npos = std::wstring_view::npos;

auto env = std::unordered_map<std::string, std::string>();

struct Environ
{
Environ()
{
ptr = GetEnvironmentStringsW();
if (ptr == nullptr)
{
throw std::runtime_error("Fail to get environment");
}
}

~Environ()
{
FreeEnvironmentStringsW(ptr);
}

wchar_t* ptr = nullptr;
} environ = {};

wchar_t* current = environ.ptr;
while (*current != '\0')
{
const auto expr = std::wstring_view(current);
const auto pos = expr.find(L'=');
assert(pos != npos);
std::string key = to_upper(windows_encoding_to_utf8(expr.substr(0, pos)));
if (!key.empty())
{
std::string value = windows_encoding_to_utf8(
(pos != npos) ? expr.substr(pos + 1) : L""
);
env.emplace(std::move(key), std::move(value));
}
current += expr.size() + 1;
}

return env;
}
}

#else // #ifdef _WIN32
Expand All @@ -103,6 +149,11 @@ namespace mamba::util

#include "mamba/util/environment.hpp"

extern "C"
{
extern char** environ; // Unix defined
}

namespace mamba::util
{
auto get_env(const std::string& key) -> std::optional<std::string>
Expand Down Expand Up @@ -133,6 +184,25 @@ namespace mamba::util
throw std::runtime_error(fmt::format(R"(Could not unset environment variable "{}")", key));
}
}

auto get_full_env() -> std::unordered_map<std::string, std::string>
{
// To keep the same signature between Unix and Windows (which is more convenient
// to work with), we have to copy strings.
// A more advanced slution could wrap a platform specific solution in a common return type.
auto env = std::unordered_map<std::string, std::string>();
for (std::size_t i = 0; environ[i]; ++i)
{
const auto expr = std::string_view(environ[i]);
const auto pos = expr.find('=');
env.emplace(
expr.substr(0, pos),
(pos != expr.npos) ? std::string(expr.substr(pos + 1)) : ""

);
}
return env;
}
}

#endif // #ifdef _WIN32
44 changes: 28 additions & 16 deletions libmamba/tests/src/util/test_environment.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -8,34 +8,46 @@

#include "mamba/util/environment.hpp"

using namespace mamba;
using namespace mamba::util;

TEST_SUITE("util::environment")
{
TEST_CASE("get_env")
{
CHECK_FALSE(util::get_env("VAR_THAT_DOES_NOT_EXIST_XYZ").has_value());
CHECK(util::get_env("PATH").has_value());
CHECK_FALSE(get_env("VAR_THAT_DOES_NOT_EXIST_XYZ").has_value());
CHECK(get_env("PATH").has_value());
}

TEST_CASE("set_env")
{
util::set_env("VAR_THAT_DOES_NOT_EXIST_XYZ", "VALUE");
CHECK_EQ(util::get_env("VAR_THAT_DOES_NOT_EXIST_XYZ"), "VALUE");
util::set_env(u8"VAR_私のにほんごわへたです", u8"😀");
CHECK_EQ(util::get_env(u8"VAR_私のにほんごわへたです"), u8"😀");
util::set_env(u8"VAR_私のにほんごわへたです", u8"hello");
CHECK_EQ(util::get_env(u8"VAR_私のにほんごわへたです"), u8"hello");
set_env("VAR_THAT_DOES_NOT_EXIST_XYZ", "VALUE");
CHECK_EQ(get_env("VAR_THAT_DOES_NOT_EXIST_XYZ"), "VALUE");
set_env(u8"VAR_私のにほんごわへたです", u8"😀");
CHECK_EQ(get_env(u8"VAR_私のにほんごわへたです"), u8"😀");
set_env(u8"VAR_私のにほんごわへたです", u8"hello");
CHECK_EQ(get_env(u8"VAR_私のにほんごわへたです"), u8"hello");
}

TEST_CASE("unset_env")
{
CHECK_FALSE(util::get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
util::unset_env("VAR_THAT_DOES_NOT_EXIST_ZZZ");
CHECK_FALSE(util::get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
util::set_env("VAR_THAT_DOES_NOT_EXIST_ZZZ", "VALUE");
CHECK(util::get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
util::unset_env("VAR_THAT_DOES_NOT_EXIST_ZZZ");
CHECK_FALSE(util::get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
CHECK_FALSE(get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
unset_env("VAR_THAT_DOES_NOT_EXIST_ZZZ");
CHECK_FALSE(get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
set_env("VAR_THAT_DOES_NOT_EXIST_ZZZ", "VALUE");
CHECK(get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
unset_env("VAR_THAT_DOES_NOT_EXIST_ZZZ");
CHECK_FALSE(get_env("VAR_THAT_DOES_NOT_EXIST_ZZZ").has_value());
}

TEST_CASE("get_full_env")
{
auto environ = get_full_env();
CHECK_GT(environ.size(), 0);
CHECK_EQ(environ.count("VAR_THAT_MUST_NOT_EXIST_XYZ"), 0);
CHECK_EQ(environ.count("PATH"), 1);

set_env(u8"VAR_私のにほHelloわへたです", u8"😀");
environ = get_full_env();
CHECK_EQ(environ.at(u8"VAR_私のにほHelloわへたです"), u8"😀");
}
}

0 comments on commit 24fa539

Please sign in to comment.