| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248 | #include <curl/curl.h>#include <gmock/gmock.h>#include <gtest/gtest.h>#include <cstddef>#include <functional>#include <memory>#include <stdexcept>#include <string>#include <vector>#include "prometheus/counter.h"#include "prometheus/detail/future_std.h"#include "prometheus/exposer.h"#include "prometheus/family.h"#include "prometheus/registry.h"namespace prometheus {namespace {using namespace testing;class IntegrationTest : public testing::Test { public:  void SetUp() override {    exposer_ = detail::make_unique<Exposer>("127.0.0.1:0");    auto ports = exposer_->GetListeningPorts();    base_url_ = std::string("http://127.0.0.1:") + std::to_string(ports.at(0));  }  struct Resonse {    long code = 0;    std::string body;    std::string contentType;  };  std::function<void(CURL*)> fetchPrePerform_;  Resonse FetchMetrics(std::string metrics_path) {    auto curl = std::shared_ptr<CURL>(curl_easy_init(), curl_easy_cleanup);    if (!curl) {      throw std::runtime_error("failed to initialize libcurl");    }    const auto url = base_url_ + metrics_path;    Resonse response;    curl_easy_setopt(curl.get(), CURLOPT_URL, url.c_str());    curl_easy_setopt(curl.get(), CURLOPT_WRITEDATA, &response.body);    curl_easy_setopt(curl.get(), CURLOPT_WRITEFUNCTION, WriteCallback);    if (fetchPrePerform_) {      fetchPrePerform_(curl.get());    }    CURLcode curl_error = curl_easy_perform(curl.get());    if (curl_error != CURLE_OK) {      throw std::runtime_error("failed to perform HTTP request");    }    curl_easy_getinfo(curl.get(), CURLINFO_RESPONSE_CODE, &response.code);    char* ct = nullptr;    curl_easy_getinfo(curl.get(), CURLINFO_CONTENT_TYPE, &ct);    if (ct) {      response.contentType = ct;    }    return response;  }  std::shared_ptr<Registry> RegisterSomeCounter(const std::string& name,                                                const std::string& path) {    const auto registry = std::make_shared<Registry>();    BuildCounter().Name(name).Register(*registry).Add({}).Increment();    exposer_->RegisterCollectable(registry, path);    return registry;  };  std::unique_ptr<Exposer> exposer_;  std::string base_url_;  std::string default_metrics_path_ = "/metrics"; private:  static size_t WriteCallback(void* contents, size_t size, size_t nmemb,                              void* userp) {    auto response = reinterpret_cast<std::string*>(userp);    size_t realsize = size * nmemb;    response->append(reinterpret_cast<const char*>(contents), realsize);    return realsize;  }};TEST_F(IntegrationTest, doesNotExposeAnythingOnDefaultPath) {  const auto metrics = FetchMetrics(default_metrics_path_);  EXPECT_GE(metrics.code, 400);}TEST_F(IntegrationTest, exposeSingleCounter) {  const std::string counter_name = "example_total";  auto registry = RegisterSomeCounter(counter_name, default_metrics_path_);  const auto metrics = FetchMetrics(default_metrics_path_);  ASSERT_EQ(metrics.code, 200);  EXPECT_THAT(metrics.body, HasSubstr(counter_name));}TEST_F(IntegrationTest, exposesCountersOnDifferentUrls) {  const std::string first_metrics_path = "/first";  const std::string second_metrics_path = "/second";  const std::string first_counter_name = "first_total";  const std::string second_counter_name = "second_total";  const auto first_registry =      RegisterSomeCounter(first_counter_name, first_metrics_path);  const auto second_registry =      RegisterSomeCounter(second_counter_name, second_metrics_path);  // all set-up  const auto first_metrics = FetchMetrics(first_metrics_path);  const auto second_metrics = FetchMetrics(second_metrics_path);  // check results  ASSERT_EQ(first_metrics.code, 200);  ASSERT_EQ(second_metrics.code, 200);  EXPECT_THAT(first_metrics.body, HasSubstr(first_counter_name));  EXPECT_THAT(second_metrics.body, HasSubstr(second_counter_name));  EXPECT_THAT(first_metrics.body, Not(HasSubstr(second_counter_name)));  EXPECT_THAT(second_metrics.body, Not(HasSubstr(first_counter_name)));}TEST_F(IntegrationTest, unexposeRegistry) {  const std::string counter_name = "some_counter_total";  const auto registry =      RegisterSomeCounter(counter_name, default_metrics_path_);  exposer_->RemoveCollectable(registry, default_metrics_path_);  const auto metrics = FetchMetrics(default_metrics_path_);  ASSERT_EQ(metrics.code, 200);  EXPECT_THAT(metrics.body, Not(HasSubstr(counter_name)));}TEST_F(IntegrationTest, acceptOptionalCompression) {  const std::string counter_name = "example_total";  auto registry = RegisterSomeCounter(counter_name, default_metrics_path_);  fetchPrePerform_ = [](CURL* curl) {    curl_easy_setopt(curl, CURLOPT_ACCEPT_ENCODING, "");  };  const auto metrics = FetchMetrics(default_metrics_path_);  ASSERT_EQ(metrics.code, 200);  EXPECT_THAT(metrics.body, HasSubstr(counter_name));}#if 0  // https://github.com/civetweb/civetweb/issues/954TEST_F(IntegrationTest, shouldRejectRequestWithoutAuthorization) {  const std::string counter_name = "example_total";  auto registry = RegisterSomeCounter(counter_name, default_metrics_path_);  exposer_->RegisterAuth(      [](const std::string& user, const std::string& password) {        return user == "test_user" && password == "test_password";      },      "Some Auth Realm", default_metrics_path_);  const auto metrics = FetchMetrics(default_metrics_path_);  ASSERT_EQ(metrics.code, 401);}#endifTEST_F(IntegrationTest, shouldPerformProperAuthentication) {  const std::string counter_name = "example_total";  auto registry = RegisterSomeCounter(counter_name, default_metrics_path_);  const auto my_username = "test_user";  const auto my_password = "test_password";  fetchPrePerform_ = [my_username, my_password](CURL* curl) {    curl_easy_setopt(curl, CURLOPT_HTTPAUTH, CURLAUTH_BASIC);    curl_easy_setopt(curl, CURLOPT_USERNAME, my_username);    curl_easy_setopt(curl, CURLOPT_PASSWORD, my_password);  };  exposer_->RegisterAuth(      [my_username, my_password](const std::string& user,                                 const std::string& password) {        return user == my_username && password == my_password;      },      "Some Auth Realm", default_metrics_path_);  const auto metrics = FetchMetrics(default_metrics_path_);  ASSERT_EQ(metrics.code, 200);  EXPECT_THAT(metrics.body, HasSubstr(counter_name));}TEST_F(IntegrationTest, shouldDealWithExpiredCollectables) {  const std::string first_counter_name = "first_total";  const std::string second_counter_name = "second_total";  const auto registry =      RegisterSomeCounter(first_counter_name, default_metrics_path_);  auto disappearing_registry =      RegisterSomeCounter(second_counter_name, default_metrics_path_);  disappearing_registry.reset();  // all set-up  const auto metrics = FetchMetrics(default_metrics_path_);  // check results  ASSERT_EQ(metrics.code, 200);  EXPECT_THAT(metrics.body, HasSubstr(first_counter_name));  EXPECT_THAT(metrics.body, Not(HasSubstr(second_counter_name)));}TEST_F(IntegrationTest, shouldSendBodyAsUtf8) {  const std::string counter_name = "example_total";  auto registry = RegisterSomeCounter(counter_name, default_metrics_path_);  const auto metrics = FetchMetrics(default_metrics_path_);  // check content type  ASSERT_EQ(metrics.code, 200);  EXPECT_THAT(metrics.contentType, HasSubstr("utf-8"));}}  // namespace}  // namespace prometheus
 |