| // Copyright 2012 The Chromium Authors | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "base/environment.h" | 
 |  | 
 | #include <memory> | 
 | #include <optional> | 
 | #include <string> | 
 | #include <string_view> | 
 | #include <utility> | 
 |  | 
 | #include "base/containers/heap_array.h" | 
 | #include "base/memory/ptr_util.h" | 
 | #include "base/strings/cstring_view.h" | 
 | #include "base/strings/string_util.h" | 
 | #include "build/build_config.h" | 
 |  | 
 | #if BUILDFLAG(IS_WIN) | 
 | #include <windows.h> | 
 |  | 
 | #include "base/strings/utf_string_conversions.h" | 
 | #elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) | 
 | #include <stdlib.h> | 
 | #endif | 
 |  | 
 | namespace base { | 
 |  | 
 | namespace { | 
 |  | 
 | class EnvironmentImpl : public Environment { | 
 |  public: | 
 |   std::optional<std::string> GetVar(cstring_view variable_name) override { | 
 |     auto result = GetVarImpl(variable_name); | 
 |     if (result.has_value()) { | 
 |       return result; | 
 |     } | 
 |  | 
 |     // Some commonly used variable names are uppercase while others | 
 |     // are lowercase, which is inconsistent. Let's try to be helpful | 
 |     // and look for a variable name with the reverse case. | 
 |     // I.e. HTTP_PROXY may be http_proxy for some users/systems. | 
 |     char first_char = variable_name[0]; | 
 |     std::string alternate_case_var; | 
 |     if (IsAsciiLower(first_char)) { | 
 |       alternate_case_var = ToUpperASCII(variable_name); | 
 |     } else if (IsAsciiUpper(first_char)) { | 
 |       alternate_case_var = ToLowerASCII(variable_name); | 
 |     } else { | 
 |       return std::nullopt; | 
 |     } | 
 |     return GetVarImpl(alternate_case_var); | 
 |   } | 
 |  | 
 |   bool SetVar(cstring_view variable_name, | 
 |               const std::string& new_value) override { | 
 |     return SetVarImpl(variable_name, new_value); | 
 |   } | 
 |  | 
 |   bool UnSetVar(cstring_view variable_name) override { | 
 |     return UnSetVarImpl(variable_name); | 
 |   } | 
 |  | 
 |  private: | 
 |   std::optional<std::string> GetVarImpl(cstring_view variable_name) { | 
 | #if BUILDFLAG(IS_WIN) | 
 |     std::wstring wide_name = UTF8ToWide(variable_name); | 
 |     // Documented to be the maximum environment variable size in characters. | 
 |     static constexpr size_t kMaxLength = 32767; | 
 |     auto value = base::HeapArray<wchar_t>::Uninit(kMaxLength); | 
 |     const DWORD value_length = | 
 |         ::GetEnvironmentVariable(wide_name.c_str(), value.data(), kMaxLength); | 
 |     if (value_length == 0 || value_length >= kMaxLength) { | 
 |       return std::nullopt;  // Ignore errors and excessively large values. | 
 |     } | 
 |     return WideToUTF8(std::wstring_view(value.data(), value_length)); | 
 | #elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) | 
 |     const char* env_value = getenv(variable_name.c_str()); | 
 |     if (!env_value) { | 
 |       return std::nullopt; | 
 |     } | 
 |     return std::string(env_value); | 
 | #endif | 
 |   } | 
 |  | 
 |   bool SetVarImpl(cstring_view variable_name, const std::string& new_value) { | 
 | #if BUILDFLAG(IS_WIN) | 
 |     // On success, a nonzero value is returned. | 
 |     return !!SetEnvironmentVariable(UTF8ToWide(variable_name).c_str(), | 
 |                                     UTF8ToWide(new_value).c_str()); | 
 | #elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) | 
 |     // On success, zero is returned. | 
 |     return !setenv(variable_name.c_str(), new_value.c_str(), 1); | 
 | #endif | 
 |   } | 
 |  | 
 |   bool UnSetVarImpl(cstring_view variable_name) { | 
 | #if BUILDFLAG(IS_WIN) | 
 |     // On success, a nonzero value is returned. | 
 |     return !!SetEnvironmentVariable(UTF8ToWide(variable_name).c_str(), nullptr); | 
 | #elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) | 
 |     // On success, zero is returned. | 
 |     return !unsetenv(variable_name.c_str()); | 
 | #endif | 
 |   } | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | Environment::~Environment() = default; | 
 |  | 
 | // static | 
 | std::unique_ptr<Environment> Environment::Create() { | 
 |   return std::make_unique<EnvironmentImpl>(); | 
 | } | 
 |  | 
 | bool Environment::HasVar(cstring_view variable_name) { | 
 |   return GetVar(variable_name).has_value(); | 
 | } | 
 |  | 
 | }  // namespace base |