#pragma once #include #include namespace DB { /** Print pipeline in "dot" format for GraphViz. * You can render it with: * dot -T png < pipeline.dot > pipeline.png */ template void printPipeline(const Processors & processors, const Statuses & statuses, WriteBuffer & out) { out << "digraph\n{\n"; out << " rankdir=\"LR\";\n"; out << " { node [shape = rect]\n"; std::unordered_map pointer_to_id; auto get_proc_id = [&](const IProcessor & proc) -> std::size_t { auto [it, inserted] = pointer_to_id.try_emplace(&proc, pointer_to_id.size()); return it->second; }; auto statuses_iter = statuses.begin(); /// Nodes // TODO quoting and escaping for (const auto & processor : processors) { const auto & description = processor->getDescription(); out << " n" << get_proc_id(*processor) << "[label=\"" << processor->getName() << (description.empty() ? "" : ":") << description; if (statuses_iter != statuses.end()) { out << " (" << IProcessor::statusToName(*statuses_iter) << ")"; ++statuses_iter; } out << "\"];\n"; } out << " }\n"; /// Edges for (const auto & processor : processors) { for (const auto & port : processor->getOutputs()) { if (!port.isConnected()) continue; const IProcessor & curr = *processor; const IProcessor & next = port.getInputPort().getProcessor(); out << " n" << get_proc_id(curr) << " -> n" << get_proc_id(next) << ";\n"; } } out << "}\n"; } template void printPipeline(const Processors & processors, WriteBuffer & out) { printPipeline(processors, std::vector(), out); } /// Prints pipeline in compact representation. /// Group processors by it's name, QueryPlanStep and QueryPlanStepGroup. /// If QueryPlanStep wasn't set for processor, representation may be not correct. /// If with_header is set, prints block header for each edge. void printPipelineCompact(const Processors & processors, WriteBuffer & out, bool with_header); }