#include "ReportBuilder.h" #include #include #include #include #include #include #include #include #include "JSONString.h" namespace DB { namespace { bool isASCIIString(const std::string & str) { return std::all_of(str.begin(), str.end(), isASCII); } String jsonString(const String & str, FormatSettings & settings) { WriteBufferFromOwnString buffer; writeJSONString(str, buffer, settings); return std::move(buffer.str()); } } ReportBuilder::ReportBuilder(const std::string & server_version_) : server_version(server_version_) , hostname(getFQDNOrHostName()) , num_cores(getNumberOfPhysicalCPUCores()) , num_threads(std::thread::hardware_concurrency()) , ram(getMemoryAmount()) { } static std::string getCurrentTime() { return DateLUT::instance().timeToString(time(nullptr)); } std::string ReportBuilder::buildFullReport( const PerformanceTestInfo & test_info, std::vector & stats, const std::vector & queries_to_run) const { FormatSettings settings; JSONString json_output; json_output.set("hostname", hostname); json_output.set("num_cores", num_cores); json_output.set("num_threads", num_threads); json_output.set("ram", ram); json_output.set("server_version", server_version); json_output.set("time", getCurrentTime()); json_output.set("test_name", test_info.test_name); json_output.set("path", test_info.path); if (!test_info.substitutions.empty()) { JSONString json_parameters(2); /// here, 2 is the size of \t padding for (auto & [parameter, values] : test_info.substitutions) { std::ostringstream array_string; array_string << "["; for (size_t i = 0; i != values.size(); ++i) { array_string << jsonString(values[i], settings); if (i != values.size() - 1) { array_string << ", "; } } array_string << ']'; json_parameters.set(parameter, array_string.str()); } json_output.set("parameters", json_parameters.asString()); } std::vector run_infos; for (size_t query_index = 0; query_index < test_info.queries.size(); ++query_index) { if (!queries_to_run.empty() && std::find(queries_to_run.begin(), queries_to_run.end(), query_index) == queries_to_run.end()) continue; for (size_t number_of_launch = 0; number_of_launch < test_info.times_to_run; ++number_of_launch) { size_t stat_index = number_of_launch * test_info.queries.size() + query_index; TestStats & statistics = stats[stat_index]; if (!statistics.ready) continue; JSONString runJSON; runJSON.set("query", jsonString(test_info.queries[query_index], settings), false); runJSON.set("query_index", query_index); if (!statistics.exception.empty()) { if (isASCIIString(statistics.exception)) runJSON.set("exception", jsonString(statistics.exception, settings), false); else runJSON.set("exception", "Some exception occurred with non ASCII message. This may produce invalid JSON. Try reproduce locally."); } /// in seconds runJSON.set("min_time", statistics.min_time / double(1000)); if (statistics.sampler.size() != 0) { JSONString quantiles(4); /// here, 4 is the size of \t padding for (double percent = 10; percent <= 90; percent += 10) { std::string quantile_key = std::to_string(percent / 100.0); while (quantile_key.back() == '0') quantile_key.pop_back(); quantiles.set(quantile_key, statistics.sampler.quantileInterpolated(percent / 100.0)); } quantiles.set("0.95", statistics.sampler.quantileInterpolated(95 / 100.0)); quantiles.set("0.99", statistics.sampler.quantileInterpolated(99 / 100.0)); quantiles.set("0.999", statistics.sampler.quantileInterpolated(99.9 / 100.0)); quantiles.set("0.9999", statistics.sampler.quantileInterpolated(99.99 / 100.0)); runJSON.set("quantiles", quantiles.asString()); } runJSON.set("total_time", statistics.total_time); if (statistics.total_time != 0) { runJSON.set("queries_per_second", static_cast(statistics.queries) / statistics.total_time); runJSON.set("rows_per_second", static_cast(statistics.total_rows_read) / statistics.total_time); runJSON.set("bytes_per_second", static_cast(statistics.total_bytes_read) / statistics.total_time); } runJSON.set("memory_usage", statistics.memory_usage); run_infos.push_back(runJSON); } } json_output.set("runs", run_infos); return json_output.asString(); } std::string ReportBuilder::buildCompactReport( const PerformanceTestInfo & test_info, std::vector & stats, const std::vector & queries_to_run) { FormatSettings settings; std::ostringstream output; for (size_t query_index = 0; query_index < test_info.queries.size(); ++query_index) { if (!queries_to_run.empty() && std::find(queries_to_run.begin(), queries_to_run.end(), query_index) == queries_to_run.end()) continue; for (size_t number_of_launch = 0; number_of_launch < test_info.times_to_run; ++number_of_launch) { if (test_info.queries.size() > 1) output << "query " << jsonString(test_info.queries[query_index], settings) << ", "; output << "run " << std::to_string(number_of_launch + 1) << ": "; std::string main_metric = "min_time"; output << main_metric << " = "; size_t index = number_of_launch * test_info.queries.size() + query_index; output << stats[index].getStatisticByName(main_metric); output << "\n"; } } return output.str(); } }