diff --git a/.github/workflows/cmake.yml b/.github/workflows/cmake.yml index 5c7be52..eaed021 100644 --- a/.github/workflows/cmake.yml +++ b/.github/workflows/cmake.yml @@ -8,23 +8,20 @@ jobs: matrix: python-version: ["3.8", "3.9", "3.10", "3.11"] runs-on: ubuntu-latest - container: ghcr.io/klebert-engineering/manylinux-cpp17-py${{ matrix.python-version }}:2023.2 + container: ghcr.io/klebert-engineering/manylinux-cpp17-py${{ matrix.python-version }}-x86_64:2024.1 + env: + ACTIONS_ALLOW_USE_UNSECURE_NODE_VERSION: true steps: - - uses: actions/checkout@v2 - with: - submodules: recursive - - name: Cache Conan packages - uses: actions/cache@v2 - with: - path: ~/.conan/data - key: ${{ runner.os }}-conan-${{ hashFiles('**/conanfile.txt') }} - restore-keys: | - ${{ runner.os }}-conan- + - name: Which Node.js? + run: | + echo "Node at $(which node): $(node -v); npm: $(npm -v)" + - name: Checkout code + uses: actions/checkout@v2 - name: Configure run: | python3 -m venv venv && . ./venv/bin/activate - pip install -U setuptools wheel pip conan==2.2.1 - mkdir build && cd build && cmake -DCMAKE_BUILD_TYPE=Release .. + pip install -U setuptools wheel pip conan==2.5.0 + mkdir build && cd build && cmake -DCMAKE_BUILD_TYPE=Release -DZSWAG_ENABLE_TESTING=ON .. - name: Build working-directory: build run: | @@ -38,7 +35,7 @@ jobs: working-directory: build run: | . ../venv/bin/activate - ctest -C Release --verbose --no-test=fail + ctest -C Release --verbose --no-tests=error - name: Deploy uses: actions/upload-artifact@v2 with: @@ -65,7 +62,7 @@ jobs: with: python-version: ${{ matrix.python-version }} architecture: x64 - - run: python -m pip install setuptools wheel conan==2.2.1 + - run: python -m pip install setuptools wheel conan==2.5.0 - run: mkdir build - name: Build (macOS) if: matrix.os == 'macos-13' @@ -76,7 +73,8 @@ jobs: cmake -DPython3_ROOT_DIR=$pythonLocation \ -DPython3_FIND_FRAMEWORK=LAST \ -DCMAKE_BUILD_TYPE=Release \ - -DHTTPLIB_USE_BROTLI_IF_AVAILABLE=OFF .. + -DHTTPLIB_USE_BROTLI_IF_AVAILABLE=OFF \ + -DZSWAG_ENABLE_TESTING=ON .. cmake --build . mv bin/wheel bin/wheel-auditme # Same as on Linux mkdir bin/wheel && mv bin/wheel-auditme/zswag*.whl bin/wheel @@ -88,7 +86,7 @@ jobs: working-directory: build run: | echo "cmake -DPython3_ROOT_DIR=$env:pythonLocation" - cmake "-DPython3_ROOT_DIR=$env:pythonLocation" -DPython3_FIND_REGISTRY=LAST -DHTTPLIB_USE_ZLIB_IF_AVAILABLE=OFF -DCMAKE_BUILD_TYPE=Release .. + cmake "-DPython3_ROOT_DIR=$env:pythonLocation" -DPython3_FIND_REGISTRY=LAST -DCMAKE_BUILD_TYPE=Release -DZSWAG_ENABLE_TESTING=ON .. cmake --build . --config Release - name: Deploy uses: actions/upload-artifact@v2 @@ -98,4 +96,4 @@ jobs: - name: Test working-directory: build run: | - ctest -C Release --verbose --no-test=fail + ctest -C Release --verbose --no-tests=error diff --git a/CMakeLists.txt b/CMakeLists.txt index fe4e9d3..b440f97 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -23,13 +23,14 @@ project(zswag) set(CMAKE_CXX_STANDARD 17) set(CMAKE_CXX_STANDARD_REQUIRED ON) -set(ZSWAG_VERSION 1.6.7post1) +set(ZSWAG_VERSION 1.7.0) option(ZSWAG_BUILD_WHEELS "Enable zswag whl-output to WHEEL_DEPLOY_DIRECTORY." ON) option(ZSWAG_KEYCHAIN_SUPPORT "Enable zswag keychain support." ON) option(ZSWAG_ENABLE_TESTING "Enable testing for the project" OFF) if (CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_SOURCE_DIR) + message (STATUS "Testing will be enabled as zswag is the top-level project.") set (ZSWAG_ENABLE_TESTING ON CACHE BOOL "By default, enable testing if this is the main project") endif() @@ -79,7 +80,7 @@ if (ZSWAG_BUILD_WHEELS) endif() endif() -if (NOT TARGET yaml-cpp) +if (NOT TARGET yaml-cpp::yaml-cpp) FetchContent_Declare(yaml-cpp GIT_REPOSITORY "https://github.com/jbeder/yaml-cpp.git" GIT_TAG "yaml-cpp-0.7.0" @@ -103,7 +104,7 @@ if (NOT TARGET speedyj) FetchContent_MakeAvailable(speedyj) endif() -if (NOT TARGET Catch2) +if (NOT TARGET Catch2::Catch2) FetchContent_Declare(Catch2 GIT_REPOSITORY "https://github.com/catchorg/Catch2.git" GIT_TAG "v3.4.0" @@ -111,7 +112,11 @@ if (NOT TARGET Catch2) FetchContent_MakeAvailable(Catch2) endif() -if (NOT TARGET httplib) +if (NOT TARGET httplib::httplib) + if (NOT TARGET ZLIB::ZLIB) + find_package(ZLIB CONFIG REQUIRED) + endif () + set (HTTPLIB_IS_USING_ZLIB TRUE) FetchContent_Declare(httplib GIT_REPOSITORY "https://github.com/yhirose/cpp-httplib.git" GIT_TAG "v0.15.3" @@ -119,8 +124,9 @@ if (NOT TARGET httplib) FetchContent_MakeAvailable(httplib) target_compile_definitions(httplib INTERFACE - CPPHTTPLIB_OPENSSL_SUPPORT) - target_link_libraries(httplib INTERFACE OpenSSL::SSL) + CPPHTTPLIB_OPENSSL_SUPPORT) + target_link_libraries( + httplib INTERFACE OpenSSL::SSL ZLIB::ZLIB) endif() if(ZSWAG_BUILD_WHEELS AND NOT TARGET pybind11) @@ -160,55 +166,6 @@ if (ZSWAG_BUILD_WHEELS) add_dependencies(wheel zswag-server-wheel) endif() -############## -# deploy openssl libs - -if (WIN32) - set(OPENSSL_DEPLOY_DIR "${ZSWAG_DEPLOY_DIR}/${CMAKE_BUILD_TYPE}") -else() - set(OPENSSL_DEPLOY_DIR "${ZSWAG_DEPLOY_DIR}") -endif() - -message(STATUS "Deploying to ${OPENSSL_DEPLOY_DIR}") -message(STATUS "OpenSSL include dir: ${OPENSSL_INCLUDE_DIR}") - -if(APPLE) - set(OPENSSL_LIB_DIR "${OPENSSL_INCLUDE_DIR}/../lib") - set(OPENSSL_LIBS - "${OPENSSL_LIB_DIR}/libcrypto.3.dylib" - "${OPENSSL_LIB_DIR}/libssl.3.dylib" - ) -elseif(MSVC) - set(OPENSSL_LIB_DIR "${OPENSSL_INCLUDE_DIR}/../bin") - set(OPENSSL_LIBS - "${OPENSSL_LIB_DIR}/libcrypto-3-x64.dll" - "${OPENSSL_LIB_DIR}/libssl-3-x64.dll" - ) -elseif(UNIX AND NOT APPLE) - set(OPENSSL_LIB_DIR "${OPENSSL_INCLUDE_DIR}/../lib") - set(OPENSSL_LIBS - "${OPENSSL_LIB_DIR}/libcrypto.so.3" - "${OPENSSL_LIB_DIR}/libssl.so.3" - ) -endif() - -foreach(file_i ${OPENSSL_LIBS}) - get_filename_component(filename ${file_i} NAME) - add_custom_command( - OUTPUT "${OPENSSL_DEPLOY_DIR}/${filename}" - COMMAND ${CMAKE_COMMAND} -E copy_if_different - "${file_i}" - "${OPENSSL_DEPLOY_DIR}/${filename}" - DEPENDS "${file_i}" - COMMENT "Copying ${file_i} to ${OPENSSL_DEPLOY_DIR}" - ) - list(APPEND COPIED_OPENSSL_LIBS "${OPENSSL_DEPLOY_DIR}/${filename}") -endforeach(file_i) - -add_custom_target(copy_openssl_libs ALL DEPENDS ${COPIED_OPENSSL_LIBS}) - -add_dependencies(zswagcl copy_openssl_libs) - diff --git a/README.md b/README.md index 09cda10..2bbabd4 100644 --- a/README.md +++ b/README.md @@ -581,26 +581,28 @@ variable. The YAML file contains a list of HTTP-related configs that are applied to HTTP requests based on a regular expression which is matched against the requested URL. -For example, the following entry would match all requests due to the `.*` -url-match-pattern: +For example, the following entry would match all requests due to the `*` +url-match-pattern for the `scope` field: ```yaml -- url: .* - basic-auth: - user: johndoe - keychain: keychain-service-string - proxy: - host: localhost - port: 8888 - user: test - keychain: ... - cookies: - key: value - headers: - key: value - query: - key: value - api-key: value +http-settings: + # Under http-settings, a list of settings is defined for specific URL scopes. + - scope: * # URL scope - e.g. https://*.nds.live/* or *.google.com. + basic-auth: # Basic auth credentials for matching requests. + user: johndoe + keychain: keychain-service-string + proxy: # Proxy settings for matching requests. + host: localhost + port: 8888 + user: test + keychain: ... + cookies: # Additional Cookies for matching requests. + key: value + headers: # Additional Headers for matching requests. + key: value + query: # Additional Query parameters for matching requests. + key: value + api-key: value # API Key as required by OpenAPI config - see description below. ``` **Note:** For `proxy` configs, the credentials are optional. diff --git a/conanfile.py b/conanfile.py new file mode 100644 index 0000000..bc79822 --- /dev/null +++ b/conanfile.py @@ -0,0 +1,21 @@ +from conan import ConanFile +from conan.tools.cmake import CMakeDeps + +class ZswagRecipe(ConanFile): + name = "zswag" + settings = "os", "arch", "compiler", "build_type" + generators = "CMakeDeps" + + # Specify options + default_options = { + "openssl*:shared": False + } + + def requirements(self): + self.requires("openssl/3.2.0") + self.requires("keychain/1.3.0") + self.requires("spdlog/1.11.0") + self.requires("pybind11/2.10.4") + self.requires("zlib/1.2.13") + # keychain and libsecret have a conflict here. + self.requires("glib/2.78.3", override=True) diff --git a/conanfile.txt b/conanfile.txt deleted file mode 100644 index 837e863..0000000 --- a/conanfile.txt +++ /dev/null @@ -1,11 +0,0 @@ -[requires] -openssl/3.2.0 -keychain/1.2.1 -spdlog/1.11.0 -pybind11/2.10.4 - -[generators] -CMakeDeps - -[options] -openssl*:shared=True diff --git a/libs/httpcl/include/httpcl/http-client.hpp b/libs/httpcl/include/httpcl/http-client.hpp index 7d75b92..95596e4 100644 --- a/libs/httpcl/include/httpcl/http-client.hpp +++ b/libs/httpcl/include/httpcl/http-client.hpp @@ -37,9 +37,7 @@ class IHttpClient Error(Result result, std::string const& message) : std::runtime_error(message) , result(std::move(result)) - { - log().error(message); - } + {} }; virtual ~IHttpClient() = default; diff --git a/libs/httpcl/include/httpcl/http-settings.hpp b/libs/httpcl/include/httpcl/http-settings.hpp index f62a0e5..14f55bf 100644 --- a/libs/httpcl/include/httpcl/http-settings.hpp +++ b/libs/httpcl/include/httpcl/http-settings.hpp @@ -5,6 +5,9 @@ #include #include #include +#include +#include +#include "yaml-cpp/yaml.h" namespace httpcl @@ -41,6 +44,10 @@ struct Config std::string keychain; }; + std::optional scope; + std::regex urlPattern; + std::string urlPatternString; + std::map cookies; std::optional auth; std::optional proxy; @@ -67,7 +74,7 @@ struct Config }; /** - * Loads settings from HTTP_SETTINGS_FILE. + * Loads/stores settings from/to HTTP_SETTINGS_FILE. * Allows returning config for a specific URL. */ struct Settings @@ -85,7 +92,17 @@ struct Settings /** * Map from URL pattern to some config values. */ - std::map settings; + std::vector settings; + YAML::Node document; + mutable std::shared_mutex mutex; + std::chrono::steady_clock::time_point lastRead; + + /** + * Prompt settings instance to re-parse the HTTP settings file, + * by calling updateTimestamp with std::chrono::steady_clock::now(). + */ + static void updateTimestamp(std::chrono::steady_clock::time_point time); + static std::atomic lastUpdated; }; struct secret diff --git a/libs/httpcl/src/http-settings.cpp b/libs/httpcl/src/http-settings.cpp index 03f0f5b..ffaa723 100644 --- a/libs/httpcl/src/http-settings.cpp +++ b/libs/httpcl/src/http-settings.cpp @@ -112,9 +112,51 @@ struct convert } namespace { -YAML::Node configToNode(Config const& config, std::string const& url=".*") { + +std::string convertToRegex(const std::string& scope) { + std::string regexPattern = "^"; + for (char c : scope) { + switch (c) { + case '*': + regexPattern += ".*"; + break; + case '.': + regexPattern += "\\."; + break; + case '\\': + regexPattern += "\\\\"; + break; + case '^': + case '$': + case '|': + case '(': + case ')': + case '[': + case ']': + case '{': + case '}': + case '?': + case '+': + case '-': + case '!': + regexPattern += '\\'; + regexPattern += c; + break; + default: + regexPattern += c; + break; + } + } + regexPattern += ".*$"; + return regexPattern; +} + +YAML::Node configToNode(Config const& config) { YAML::Node result; - result["url"] = url; + if (config.scope) + result["scope"] = *config.scope; + else + result["url"] = config.urlPatternString; if (!config.cookies.empty()) result["cookies"] = config.cookies; @@ -139,16 +181,21 @@ YAML::Node configToNode(Config const& config, std::string const& url=".*") { return result; } -std::pair configFromNode(YAML::Node const& node) +Config configFromNode(YAML::Node const& node) { - std::string urlPattern; Config conf; - if (auto entryParam = node["url"]) - urlPattern = entryParam.as(); - else - throw std::runtime_error( - "HTTP Settings: Missing 'url' field in: " + YAML::Dump(node)); + if (auto entryParam = node["url"]) { + conf.urlPattern = conf.urlPatternString = entryParam.as(); + } + else { + if (auto entryParamScope = node["scope"]) + conf.scope = entryParamScope.as(); + else + conf.scope = "*"; + conf.urlPatternString = convertToRegex(*conf.scope); + conf.urlPattern = conf.urlPatternString; + } if (auto cookies = node["cookies"]) conf.cookies = cookies.as>(); @@ -172,7 +219,7 @@ std::pair configFromNode(YAML::Node const& node) if (auto apiKey = node["api-key"]) conf.apiKey = apiKey.as(); - return {std::move(conf), std::move(urlPattern)}; + return conf; } } @@ -285,8 +332,16 @@ Settings::Settings() load(); } +std::atomic Settings::lastUpdated{std::chrono::steady_clock::now()}; + +void Settings::updateTimestamp(std::chrono::steady_clock::time_point time) { + lastUpdated.store(time, std::memory_order_relaxed); +} + void Settings::load() { + std::unique_lock lock(mutex); + lastRead = std::chrono::steady_clock::now(); settings.clear(); auto cookieJar = std::getenv("HTTP_SETTINGS_FILE"); @@ -302,12 +357,20 @@ void Settings::load() try { log().debug("Loading HTTP settings from '{}'...", cookieJar); - auto node = YAML::LoadFile(cookieJar); + document = YAML::LoadFile(cookieJar); + YAML::Node httpSettingsNode; uint32_t idx = 0; - for (auto const& entry : node.as>()) { - auto [conf, urlPattern] = configFromNode(entry); - settings[urlPattern] = std::move(conf); + if (document.IsMap()) { + httpSettingsNode = document["http-settings"]; + } + else { + // Keep supporting the old format, where the root structure is a settings array. + httpSettingsNode = document; + } + + for (auto const& entry : httpSettingsNode.as>()) { + settings.emplace_back(configFromNode(entry)); ++idx; } @@ -330,9 +393,16 @@ void Settings::store() try { auto node = YAML::Node(); - for (const auto& [key, config] : settings) + for (const auto& config : settings) node.push_back(configToNode(config)); + if (document && document.IsMap()) { + document["http-settings"] = node; + } + else { + document = node; + } + log().debug("Saving HTTP settings to '{}'...", cookieJar); std::ofstream os(cookieJar); os << node; @@ -345,20 +415,31 @@ void Settings::store() Config::Config(const std::string& yamlConf) { YAML::Node parsedYaml = YAML::Load(yamlConf); - *this = configFromNode(parsedYaml).first; + *this = configFromNode(parsedYaml); } Config Settings::operator[] (const std::string &url) const { - Config result; + std::shared_lock lock(mutex); + if (lastRead < lastUpdated.load(std::memory_order_relaxed)) { + lock.unlock(); + try { + const_cast(this)->load(); + } + catch (...) { + // If an exception occurred during load(), it was already + // logged in all likelihood. + } + lock.lock(); + } - for (auto const& [pattern, config] : settings) + Config result; + for (auto const& config : settings) { - if (!std::regex_match(url, std::regex(pattern))) + if (!std::regex_match(url, config.urlPattern)) continue; result |= config; } - return result; } @@ -389,7 +470,7 @@ void Config::apply(httplib::Client &cl) const // Proxy Settings if (proxy) { - cl.set_proxy(proxy->host.c_str(), proxy->port); + cl.set_proxy(proxy->host, proxy->port); auto password = proxy->password; if (!proxy->keychain.empty()) @@ -397,7 +478,7 @@ void Config::apply(httplib::Client &cl) const if (!proxy->user.empty()) cl.set_proxy_basic_auth( - proxy->user.c_str(), password.c_str()); + proxy->user, password); } cl.set_default_headers(httpLibHeaders); diff --git a/libs/pyzswagcl/CMakeLists.txt b/libs/pyzswagcl/CMakeLists.txt index 2c0e8f2..43f8a94 100644 --- a/libs/pyzswagcl/CMakeLists.txt +++ b/libs/pyzswagcl/CMakeLists.txt @@ -19,26 +19,10 @@ target_compile_features(pyzswagcl INTERFACE cxx_std_17) -if (MSVC) - # Required because cpp-httplib speaks https via OpenSSL - set (DEPLOY_FILES - "${OPENSSL_INCLUDE_DIR}/../bin/libcrypto-3-x64.dll" - "${OPENSSL_INCLUDE_DIR}/../bin/libssl-3-x64.dll") -endif() - -if (APPLE) - # Required because cpp-httplib speaks https via OpenSSL - set (DEPLOY_FILES - "${OPENSSL_INCLUDE_DIR}/../lib/libcrypto.3.dylib" - "${OPENSSL_INCLUDE_DIR}/../lib/libssl.3.dylib") -endif() - add_wheel(pyzswagcl AUTHOR "Navigation Data Standard e.V." URL "https://github.com/ndsev/zswag" VERSION "${ZSWAG_VERSION}" DESCRIPTION "Python bindings for the zswag client library." TARGET_DEPENDENCIES - zswagcl speedyj httpcl - DEPLOY_FILES - ${DEPLOY_FILES}) + zswagcl speedyj httpcl) diff --git a/libs/pyzswagcl/py-openapi-client.cpp b/libs/pyzswagcl/py-openapi-client.cpp index 7b66116..cb06d99 100644 --- a/libs/pyzswagcl/py-openapi-client.cpp +++ b/libs/pyzswagcl/py-openapi-client.cpp @@ -65,9 +65,9 @@ namespace void PyOpenApiClient::bind(py::module_& m) { auto serviceClient = py::class_(m, "OAClient") - .def(py::init, std::optional>(), + .def(py::init, std::optional, std::optional>(), "url"_a, "is_local_file"_a = false, "config"_a = httpcl::Config(), - "api_key"_a = std::optional(), "bearer"_a = std::optional()) + "api_key"_a = std::optional(), "bearer"_a = std::optional(), "server_index"_a = std::optional()) // zserio >= 2.3.0 .def("call_method", &PyOpenApiClient::callMethod, "method_name"_a, "request"_a, "unused"_a) @@ -83,7 +83,8 @@ PyOpenApiClient::PyOpenApiClient(std::string const& openApiUrl, bool isLocalFile, httpcl::Config const& config, std::optional apiKey, - std::optional bearer) + std::optional bearer, + std::optional serverIndex) { auto httpConfig = config; // writable copy if (apiKey) @@ -96,11 +97,14 @@ PyOpenApiClient::PyOpenApiClient(std::string const& openApiUrl, std::ifstream fs(openApiUrl); return parseOpenAPIConfig(fs); } - else - return fetchOpenAPIConfig(openApiUrl, *httpClient, httpConfig); + return fetchOpenAPIConfig(openApiUrl, *httpClient, httpConfig); }(); - client_ = std::make_unique(openApiConfig, httpConfig, std::move(httpClient)); + client_ = std::make_unique( + openApiConfig, + httpConfig, + std::move(httpClient), + serverIndex ? *serverIndex : 0); } std::vector PyOpenApiClient::callMethod( diff --git a/libs/pyzswagcl/py-openapi-client.h b/libs/pyzswagcl/py-openapi-client.h index 42dc11e..93ec934 100644 --- a/libs/pyzswagcl/py-openapi-client.h +++ b/libs/pyzswagcl/py-openapi-client.h @@ -18,7 +18,8 @@ class PyOpenApiClient bool isLocalFile, httpcl::Config const& config, std::optional apiKey, - std::optional bearer); + std::optional bearer, + std::optional serverIndex); std::vector callMethod( const std::string& methodName, diff --git a/libs/pyzswagcl/py-zswagcl.cpp b/libs/pyzswagcl/py-zswagcl.cpp index 0a83773..afca395 100644 --- a/libs/pyzswagcl/py-zswagcl.cpp +++ b/libs/pyzswagcl/py-zswagcl.cpp @@ -122,11 +122,17 @@ PYBIND11_MODULE(pyzswagcl, m) auto it = self.methodPath.find(methodName); if (it != self.methodPath.end()) return it->second; - else - throw std::runtime_error( - "Could not find OpenAPI config for method name "s+methodName); + throw std::runtime_error( + "Could not find OpenAPI config for method name "s+methodName); }, py::is_operator(), py::return_value_policy::reference_internal, "method_name"_a) .def_readonly("content", &OpenAPIConfig::content) + .def_property_readonly("servers", [](const OpenAPIConfig& self) -> std::vector + { + std::vector result; + for (auto const& uri : self.servers) + result.emplace_back(uri.build()); + return result; + }, py::return_value_policy::automatic) ; m.def("parse_openapi_config", [](std::string const& path){ diff --git a/libs/zswag/reflect.py b/libs/zswag/reflect.py index bf5ffa9..471617d 100644 --- a/libs/zswag/reflect.py +++ b/libs/zswag/reflect.py @@ -176,7 +176,7 @@ def str_to_bytes(s: str, fmt: OAParamFormat) -> bytes: # Convert a single passed parameter value to it's correct type def parse_param_value(param: OAParam, target_type: Type, value: str) -> Any: - # Check if its an enum ... + # Check if it's an enum ... if issubclass(target_type, Enum): return target_type(parse_param_value(param, int, value)) # Check if the parameter format is native string conversion diff --git a/libs/zswag/test/calc/api.yaml b/libs/zswag/test/calc/api.yaml index d7d0ff4..42eea66 100644 --- a/libs/zswag/test/calc/api.yaml +++ b/libs/zswag/test/calc/api.yaml @@ -273,4 +273,5 @@ paths: summary: '' security: - HeaderAuth: [] -servers: [] +servers: + - url: / diff --git a/libs/zswag/test/calc/client.py b/libs/zswag/test/calc/client.py index e6b984b..d47eac7 100644 --- a/libs/zswag/test/calc/client.py +++ b/libs/zswag/test/calc/client.py @@ -21,7 +21,7 @@ def run_test(aspect, request, fn, expect, auth_args): try: print(f"[py-test-client] Test#{counter}: {aspect}", flush=True) print(f"[py-test-client] -> Instantiating client.", flush=True) - oa_client = OAClient(f"http://{host}:{port}/openapi.json", **auth_args) + oa_client = OAClient(f"http://{host}:{port}/openapi.json", **auth_args, server_index=0) # Just make sure that OpenAPI JSON content is parsable assert oa_client.config().content and json.loads(oa_client.config().content) client = api.Calculator.Client(oa_client) diff --git a/libs/zswag/test/client.cpp b/libs/zswag/test/client.cpp index 8b32d18..7df0032 100644 --- a/libs/zswag/test/client.cpp +++ b/libs/zswag/test/client.cpp @@ -19,7 +19,7 @@ int main (int argc, char* argv[]) { spdlog::info("[cpp-test-client] Starting integration tests with {}", specUrl); - auto runTest = [&] (auto const& fn, auto expect, std::string const& aspect, std::function const& authFun) + auto runTest = [&] (auto const& fn, auto expect, std::string const& aspect, std::function const& authFun) { ++testCounter; spdlog::info("[cpp-test-client] Executing test #{}: {} ...", testCounter, aspect); @@ -28,9 +28,11 @@ int main (int argc, char* argv[]) { spdlog::info("[cpp-test-client] => Instantiating client."); auto httpClient = std::make_unique(); auto openApiConfig = fetchOpenAPIConfig(specUrl, *httpClient); - httpcl::Config authHttpConf; + // See https://github.com/spec-first/connexion/issues/1139 + openApiConfig.servers.insert(openApiConfig.servers.begin(), URIComponents::fromStrPath("/bad/path/we/dont/access")); + Config authHttpConf; authFun(authHttpConf); - auto oaClient = OAClient(openApiConfig, std::move(httpClient), authHttpConf); + auto oaClient = OAClient(openApiConfig, std::move(httpClient), authHttpConf, 1); spdlog::info("[cpp-test-client] => Running request."); calculator::Calculator::Client calcClient(oaClient); auto response = fn(calcClient); @@ -49,13 +51,13 @@ int main (int argc, char* argv[]) { calculator::BaseAndExponent req(calculator::I32(2), calculator::I32(3), 0, "", .0, std::vector{}); return calcClient.powerMethod(req); }, 8., "Pass fields in path and header", - [](httpcl::Config& conf){}); + [](Config& conf){}); runTest([](calculator::Calculator::Client& calcClient){ calculator::Integers req(std::vector{100, -200, 400}); return calcClient.intSumMethod(req); }, 300., "Pass hex-encoded array in query", - [](httpcl::Config& conf){ + [](Config& conf){ conf.headers.insert({"Authorization", "Bearer 123"}); }); @@ -63,8 +65,8 @@ int main (int argc, char* argv[]) { calculator::Bytes req(std::vector{8, 16, 32, 64}); return calcClient.byteSumMethod(req); }, 120., "Pass base64url-encoded byte array in path", - [](httpcl::Config& conf){ - conf.auth = httpcl::Config::BasicAuthentication{ + [](Config& conf){ + conf.auth = Config::BasicAuthentication{ "u", "pw", "" }; }); @@ -73,7 +75,7 @@ int main (int argc, char* argv[]) { calculator::Integers req(std::vector{1, 2, 3, 4}); return calcClient.intMulMethod(req); }, 24., "Pass base64-encoded long array in path", - [](httpcl::Config& conf){ + [](Config& conf){ conf.query.insert({"api-key", "42"}); }); @@ -81,7 +83,7 @@ int main (int argc, char* argv[]) { calculator::Doubles req(std::vector{34.5, 2.}); return calcClient.floatMulMethod(req); }, 69., "Pass float array in query.", - [](httpcl::Config& conf){ + [](Config& conf){ conf.cookies.insert({"api-cookie", "42"}); }); @@ -89,7 +91,7 @@ int main (int argc, char* argv[]) { calculator::Bools req(std::vector{true, false}); return calcClient.bitMulMethod(req); }, false, "Pass bool array in query (expect false).", - [](httpcl::Config& conf){ + [](Config& conf){ conf.apiKey = "42"; }); @@ -97,7 +99,7 @@ int main (int argc, char* argv[]) { calculator::Bools req(std::vector{true, true}); return calcClient.bitMulMethod(req); }, true, "Pass bool array in query (expect true).", - [](httpcl::Config& conf){ + [](Config& conf){ conf.headers.insert({"X-Generic-Token", "42"}); }); @@ -105,7 +107,7 @@ int main (int argc, char* argv[]) { calculator::Double req(1.); return calcClient.identityMethod(req); }, 1., "Pass request as blob in body", - [](httpcl::Config& conf){ + [](Config& conf){ conf.cookies.insert({"api-cookie", "42"}); }); @@ -113,7 +115,7 @@ int main (int argc, char* argv[]) { calculator::Strings req(std::vector{"foo", "bar"}); return calcClient.concatMethod(req); }, std::string("foobar"), "Pass base64-encoded strings.", - [](httpcl::Config& conf){ + [](Config& conf){ conf.headers.insert({"Authorization", "Bearer 123"}); }); @@ -121,7 +123,7 @@ int main (int argc, char* argv[]) { calculator::EnumWrapper req(calculator::Enum::TEST_ENUM_0); return calcClient.nameMethod(req); }, std::string("TEST_ENUM_0"), "Pass enum.", - [](httpcl::Config& conf){ + [](Config& conf){ conf.apiKey = "42"; }); diff --git a/libs/zswagcl/include/zswagcl/oaclient.hpp b/libs/zswagcl/include/zswagcl/oaclient.hpp index 66b6b5b..3b00a33 100644 --- a/libs/zswagcl/include/zswagcl/oaclient.hpp +++ b/libs/zswagcl/include/zswagcl/oaclient.hpp @@ -12,9 +12,10 @@ class OAClient : public ::zserio::IServiceClient { public: OAClient( - zswagcl::OpenAPIConfig config, + OpenAPIConfig config, std::unique_ptr client, - httpcl::Config httpConfig = {}); + httpcl::Config httpConfig = {}, + uint32_t serverIndex = 0); std::vector callMethod( zserio::StringView methodName, diff --git a/libs/zswagcl/include/zswagcl/private/openapi-client.hpp b/libs/zswagcl/include/zswagcl/private/openapi-client.hpp index 212fa83..96a910a 100644 --- a/libs/zswagcl/include/zswagcl/private/openapi-client.hpp +++ b/libs/zswagcl/include/zswagcl/private/openapi-client.hpp @@ -20,7 +20,8 @@ class OpenAPIClient OpenAPIClient(OpenAPIConfig config, httpcl::Config httpConfig, - std::unique_ptr client); + std::unique_ptr client, + uint32_t serverIndex = 0); ~OpenAPIClient(); /** @@ -41,6 +42,7 @@ class OpenAPIClient private: std::unique_ptr client_; httpcl::Settings settings_; + httpcl::URIComponents server_; }; } diff --git a/libs/zswagcl/include/zswagcl/private/openapi-config.hpp b/libs/zswagcl/include/zswagcl/private/openapi-config.hpp index 90902bd..2597f97 100644 --- a/libs/zswagcl/include/zswagcl/private/openapi-config.hpp +++ b/libs/zswagcl/include/zswagcl/private/openapi-config.hpp @@ -183,7 +183,7 @@ struct OpenAPIConfig /** * URI parts. */ - httpcl::URIComponents uri; + std::vector servers; /** * Map from service method name to path configuration. diff --git a/libs/zswagcl/src/oaclient.cpp b/libs/zswagcl/src/oaclient.cpp index b539d3d..85a1e1f 100644 --- a/libs/zswagcl/src/oaclient.cpp +++ b/libs/zswagcl/src/oaclient.cpp @@ -9,8 +9,9 @@ namespace zswagcl OAClient::OAClient(zswagcl::OpenAPIConfig config, std::unique_ptr client, - httpcl::Config httpConfig) - : client_(std::move(config), std::move(httpConfig), std::move(client)) + httpcl::Config httpConfig, + uint32_t serverIndex) + : client_(std::move(config), std::move(httpConfig), std::move(client), serverIndex) {} template diff --git a/libs/zswagcl/src/openapi-client.cpp b/libs/zswagcl/src/openapi-client.cpp index da6c658..d8dce3f 100644 --- a/libs/zswagcl/src/openapi-client.cpp +++ b/libs/zswagcl/src/openapi-client.cpp @@ -117,17 +117,24 @@ void checkSecurityAlternativesAndApplyApiKey(OpenAPIConfig::SecurityAlternatives OpenAPIClient::OpenAPIClient(OpenAPIConfig config, httpcl::Config httpConfig, - std::unique_ptr client) + std::unique_ptr client, + uint32_t serverIndex) : config_(std::move(config)) + , httpConfig_(std::move(httpConfig)) , client_(std::move(client)) - , httpConfig_(httpConfig) { - httpcl::log().debug("Instantiating OpenApiClient for node at '{}'", config_.uri.build()); + if (serverIndex >= config_.servers.size()) + throw httpcl::logRuntimeError( + fmt::format( + "The server index {} is out of bounds (servers.size()={}).", + serverIndex, + config_.servers.size())); + server_ = config_.servers[serverIndex]; + httpcl::log().debug("Instantiating OpenApiClient for node at '{}'", server_.build()); assert(client_); } -OpenAPIClient::~OpenAPIClient() -{} +OpenAPIClient::~OpenAPIClient() = default; std::string OpenAPIClient::call(const std::string& methodIdent, const std::functionsecond; - httpcl::URIComponents uri(config_.uri); + auto uri = server_; uri.appendPath(resolvePath(method, paramCb)); std::string builtUri = uri.build(); std::string debugContext = stx::format("[{} {}]", method.httpMethod, uri.buildPath()); diff --git a/libs/zswagcl/src/openapi-parser.cpp b/libs/zswagcl/src/openapi-parser.cpp index a81af2b..b396104 100644 --- a/libs/zswagcl/src/openapi-parser.cpp +++ b/libs/zswagcl/src/openapi-parser.cpp @@ -235,6 +235,22 @@ static void parseMethodParameter(YAMLScope const& parameterNode, OpenAPIConfig::Path& path) { auto nameNode = parameterNode.mandatoryChild("name"); + + if (!parameterNode[ZSERIO_REQUEST_PART]) { + // Ignore parameters which do not have x-zserio-request-part, but + // output a warning for such parameters if they are not optional. + // By default, OpenAPI treats all request parameters as optional. + // You can add required: true to mark a parameter as required. + if (auto requiredNode = parameterNode["required"]) { + if (requiredNode.as()) + httpcl::log().warn( + "The parameter {} does not have x-zserio-request-part and is not optional." + "Ensure that it is filled by passing additional HTTP settings.", + parameterNode.str()); + } + return; + } + auto& parameter = path.parameters[nameNode.as()]; parameter.ident = nameNode.as(); @@ -385,9 +401,9 @@ static void parseServer(const YAMLScope& serverNode, if (urlStr.empty()) { // Ignore empty URLs. } else if (urlStr.front() == '/') { - config.uri = httpcl::URIComponents::fromStrPath(urlStr); + config.servers.emplace_back(httpcl::URIComponents::fromStrPath(urlStr)); } else { - config.uri = httpcl::URIComponents::fromStrRfc3986(urlStr); + config.servers.emplace_back(httpcl::URIComponents::fromStrRfc3986(urlStr)); } } } @@ -452,11 +468,16 @@ OpenAPIConfig fetchOpenAPIConfig(const std::string& url, httpcl::log().debug("{} Parsing OpenAPI spec", debugContext); auto config = parseOpenAPIConfig(ss); - if (config.uri.scheme.empty()) - config.uri.scheme = uriParts.scheme; - if (config.uri.host.empty()) { - config.uri.host = uriParts.host; - config.uri.port = uriParts.port; + // Add a default server and add missing server uri parts. + if (config.servers.empty()) + config.servers.emplace_back(); + for (auto& server : config.servers) { + if (server.scheme.empty()) + server.scheme = uriParts.scheme; + if (server.host.empty()) { + server.host = uriParts.host; + server.port = uriParts.port; + } } httpcl::log().debug("{} Parsed spec has {} methods.", debugContext, config.methodPath.size()); diff --git a/requirements.txt b/requirements.txt index 55735de..b7ad7b0 100644 --- a/requirements.txt +++ b/requirements.txt @@ -2,5 +2,4 @@ connexion~=2.14.2 requests zserio<3.0.0 pyyaml -pyzswagcl==1.6.7post1 openapi-spec-validator diff --git a/setup.py b/setup.py index a423d15..33a8d5d 100644 --- a/setup.py +++ b/setup.py @@ -10,7 +10,7 @@ long_description = fh.read() required_url = [] -required = [] +required = [f"pyzswagcl=={VERSION}"] with open("requirements.txt", "r") as freq: for line in freq.read().split(): if "://" in line: