// Copyright 2022 Huawei Cloud Computing Technology Co., Ltd. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef INCLUDED_SRC_BUILDTOOL_EXECUTION_API_REMOTE_CONFIG_HPP #define INCLUDED_SRC_BUILDTOOL_EXECUTION_API_REMOTE_CONFIG_HPP #include #include #include #include #include #include #include #include #include #include #include #include "gsl/gsl" #include "src/buildtool/logging/logger.hpp" #include "src/utils/cpp/type_safe_arithmetic.hpp" // Port struct PortTag : type_safe_arithmetic_tag {}; using Port = type_safe_arithmetic; [[nodiscard]] static auto ParsePort(int const port_num) noexcept -> std::optional { try { static constexpr int kMaxPortNumber{ std::numeric_limits::max()}; if (port_num >= 0 and port_num <= kMaxPortNumber) { return gsl::narrow_cast(port_num); } } catch (std::out_of_range const& e) { Logger::Log(LogLevel::Error, "Port raised out_of_range exception."); } return std::nullopt; } [[nodiscard]] static auto ParsePort(std::string const& port) noexcept -> std::optional { try { auto port_num = std::stoi(port); return ParsePort(port_num); } catch (std::invalid_argument const& e) { Logger::Log(LogLevel::Error, "Port raised invalid_argument exception."); } return std::nullopt; } class RemoteExecutionConfig { public: struct ServerAddress { std::string host{}; Port port{}; [[nodiscard]] auto ToJson() const noexcept -> nlohmann::json { return nlohmann::json{ fmt::format("{}:{}", host, static_cast(port))}; } }; // Obtain global instance [[nodiscard]] static auto Instance() noexcept -> RemoteExecutionConfig& { static RemoteExecutionConfig config; return config; } // Set remote execution and cache address, unsets if parsing `address` fails [[nodiscard]] static auto SetRemoteAddress( std::string const& address) noexcept -> bool { auto& inst = Instance(); return static_cast(inst.remote_address_ = inst.cache_address_ = ParseAddress(address)); } // Set remote-execution dispatch property list [[nodiscard]] static auto SetRemoteExecutionDispatch( const std::filesystem::path& filename) noexcept -> bool; // Set specific cache address, unsets if parsing `address` fails [[nodiscard]] static auto SetCacheAddress( std::string const& address) noexcept -> bool { return static_cast(Instance().cache_address_ = ParseAddress(address)); } // Add platform property from string of form "key:val" [[nodiscard]] static auto AddPlatformProperty( std::string const& property) noexcept -> bool { if (auto pair = ParseProperty(property)) { Instance().platform_properties_[std::move(pair->first)] = std::move(pair->second); return true; } return false; } // Remote execution address, if set [[nodiscard]] static auto RemoteAddress() noexcept -> std::optional { return Instance().remote_address_; } // Cache address, if set [[nodiscard]] static auto CacheAddress() noexcept -> std::optional { return Instance().cache_address_; } // Instance dispatch list [[nodiscard]] static auto DispatchList() noexcept -> std::vector< std::pair, ServerAddress>> { return Instance().dispatch_; } [[nodiscard]] static auto PlatformProperties() noexcept -> std::map { return Instance().platform_properties_; } private: // Server address of remote execution. std::optional remote_address_{}; // Server dispatch data std::vector, ServerAddress>> dispatch_{}; // Server address of cache endpoint for rebuild. std::optional cache_address_{}; // Platform properties for execution. std::map platform_properties_{}; [[nodiscard]] static auto ParseAddress(std::string const& address) noexcept -> std::optional { std::istringstream iss(address); std::string host; std::string port; if (not std::getline(iss, host, ':') or not std::getline(iss, port, ':')) { return std::nullopt; } auto port_num = ParsePort(port); if (not host.empty() and port_num) { return ServerAddress{std::move(host), *port_num}; } return std::nullopt; } [[nodiscard]] static auto ParseProperty( std::string const& property) noexcept -> std::optional> { std::istringstream pss(property); std::string key; std::string val; if (not std::getline(pss, key, ':') or not std::getline(pss, val, ':')) { return std::nullopt; } return std::make_pair(key, val); } }; #endif // INCLUDED_SRC_BUILDTOOL_EXECUTION_API_REMOTE_CONFIG_HPP