aboutsummaryrefslogtreecommitdiffstats
path: root/nihil.posix/getenv.ccm
blob: 5967bf72a32f6ae5ecadb16c59383d5d533ced0d (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
// This source code is released into the public domain.
module;

#include <cerrno>
#include <expected>
#include <string>
#include <system_error>
#include <vector>

#include <unistd.h>

#include "nihil.hh"

export module nihil.posix:getenv;

import nihil.error;

namespace nihil {


// Find a variable by the given name in the environment by calling getenv_r() if available,
// or getenv() if not.  In either case the value is copied, so will not be affected by
// future calls to setenv().
export [[nodiscard]] auto getenv(std::string_view varname) -> std::expected<std::string, error>
{
	auto cvarname = std::string(varname);

#ifdef NIHIL_HAVE_GETENV_R
	// Start with a buffer of this size, and double it every iteration.
	constexpr auto bufinc = std::size_t{1024};

	auto buf = std::vector<char>(bufinc);
	for (;;) {
		auto const ret = ::getenv_r(cvarname.c_str(), buf.data(), buf.size());

		if (ret == 0)
			return {std::string(buf.data())};

		if (ret == -1 && errno == ERANGE) {
			buf.resize(buf.size() * 2);
			continue;
		}

		return std::unexpected(error(std::errc(errno)));
	}
#else  // NIHIL_HAVE_GETENV_R
	errno = 0;
	auto *v = ::getenv(cvarname.c_str()); // NOLINT

	if (v != nullptr)
		return {std::string(v)};

	if (errno != 0)
		return std::unexpected(error(std::errc(errno)));

	return std::unexpected(error(std::errc::no_such_file_or_directory));
#endif // NIHIL_HAVE_GETENV_R
}

} // namespace nihil