// Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root. #include #include #include #include #include #include using namespace config; using vespalib::Portal; using vespalib::NullCryptoEngine; auto create_server() { return Portal::create(std::make_shared(), 0); } class HTTPStatus : public vespalib::Portal::GetHandler { private: Portal::SP _server; std::string _reply; bool _fail; Portal::Token::UP _root; void get(Portal::GetRequest request) override { if (_fail) { request.respond_with_error(500, "Error"); } else { request.respond_with_content("application/json", _reply); } }; public: HTTPStatus(std::string reply) : _server(create_server()), _reply(reply), _fail(false), _root(_server->bind("/", *this)) {} HTTPStatus(bool fail) : _server(create_server()), _reply(""), _fail(fail), _root(_server->bind("/", *this)) {} int getListenPort() { return _server->listen_port(); } ~HTTPStatus() { _root.reset(); }; }; class Status { public: ConfigStatus::Flags flags; std::unique_ptr status; Status(int http_port, const ConfigStatus::Flags& cfg_flags, const std::vector& model_hosts) : flags(cfg_flags) { flags.verbose = true; ConfigSet set; ConfigContext::SP ctx(new ConfigContext(set)); cloud::config::ModelConfigBuilder builder; cloud::config::ModelConfigBuilder::Hosts::Services::Ports port; port.number = http_port; port.tags = "http state"; cloud::config::ModelConfigBuilder::Hosts::Services service; service.name = "qrserver"; service.type = "qrserver"; service.configid = "qrserver/cluster.default"; service.clustertype = "qrserver"; service.clustername = "default"; service.ports.push_back(port); for (auto& mhost : model_hosts) { cloud::config::ModelConfigBuilder::Hosts host; host.services.push_back(service); host.name = mhost; builder.hosts.push_back(host); } set.addBuilder("admin/model", &builder); config::ConfigUri uri("admin/model", ctx); std::unique_ptr s(new ConfigStatus(flags, uri)); status = std::move(s); } Status(int http_port) : Status(http_port, ConfigStatus::Flags(), {{"localhost"}}) {} ~Status() { } }; std::string ok_json_at_gen_1() { return "{\"config\": { \"all\": { \"generation\": 1 } }}"; } TEST_FF("all ok", HTTPStatus(ok_json_at_gen_1()), Status(f1.getListenPort())) { ASSERT_EQUAL(0, f2.status->action()); } TEST_FF("generation too old", HTTPStatus(std::string("{\"config\": { \"all\": { \"generation\": 0 } }}")), Status(f1.getListenPort())) { ASSERT_EQUAL(1, f2.status->action()); } TEST_FF("bad json", HTTPStatus(std::string("{")), Status(f1.getListenPort())) { ASSERT_EQUAL(1, f2.status->action()); } TEST_FF("http failure", HTTPStatus(true), Status(f1.getListenPort())) { ASSERT_EQUAL(1, f2.status->action()); } TEST_F("queried host set can be constrained", HTTPStatus(ok_json_at_gen_1())) { HostFilter filter({"localhost"}); std::vector hosts( {"localhost", "no-such-host.foo.yahoo.com"}); Status status(f1.getListenPort(), ConfigStatus::Flags(filter), hosts); // Non-existing host should never be contacted. ASSERT_EQUAL(0, status.status->action()); } TEST_MAIN() { TEST_RUN_ALL(); }