0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
generators.cc
Go to the documentation of this file.
1/**
2 * @file src/common/generators/generators.cc
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18
19#include "generators.h"
20
21#include "progress_indicator.h"
22
25 inja::json &context, const std::string &prefix)
26{
27 if (!context.contains("element"))
28 return;
29
30 if (!context["element"].contains("source"))
31 return;
32
33 auto &source = context["element"]["source"];
34
35 if (source.at("path").empty())
36 return;
37
38 auto path = std::filesystem::path{source.at("path").get<std::string>()};
39 auto prefix_path = std::filesystem::path(prefix);
40 if (path.is_absolute() && util::is_relative_to(path, prefix_path)) {
41 source["path"] = relative(path, prefix_path);
42 return;
43 }
44}
45
47 const std::vector<std::string> &diagram_names,
50 std::map<std::string, std::vector<std::string>> &translation_units_map)
51{
52 for (const auto &[name, diagram] : config.diagrams) {
53 // If there are any specific diagram names provided on the command line,
54 // and this diagram is not in that list - skip it
55 if (!diagram_names.empty() && !util::contains(diagram_names, name))
56 continue;
57
58 translation_units_map[name] =
59 diagram->glob_translation_units(compilation_database.getAllFiles(),
61
62 LOG_DBG("Found {} translation units for diagram '{}'",
63 translation_units_map.at(name).size(), name);
64 }
65}
66
68 std::shared_ptr<config::diagram> diagram_config)
69{
70 std::string cmd;
71 switch (generator_type) {
73 cmd = diagram_config->puml().cmd;
74 break;
76 cmd = diagram_config->mermaid().cmd;
77 break;
78 default:
79 return;
80 };
81
82 if (cmd.empty())
83 throw std::runtime_error(
84 fmt::format("No render command template provided for {} diagrams",
85 to_string(diagram_config->type())));
86
87 util::replace_all(cmd, "{}", diagram_config->name);
88
89 LOG_INFO("Rendering diagram {} using {}", diagram_config->name,
90 to_string(generator_type));
91
93}
94
95namespace detail {
96
97template <typename DiagramConfig, typename GeneratorTag, typename DiagramModel>
98void generate_diagram_select_generator(const std::string &od,
99 const std::string &name, std::shared_ptr<clanguml::config::diagram> diagram,
100 const DiagramModel &model)
101{
102 using diagram_generator =
104
105 if constexpr (!std::is_same_v<diagram_generator, not_supported>) {
106
107 std::stringstream buffer;
108 buffer << diagram_generator(
109 dynamic_cast<DiagramConfig &>(*diagram), *model);
110
111 // Only open the file after the diagram has been generated successfully
112 // in order not to overwrite previous diagram in case of failure
113 auto path = std::filesystem::path{od} /
114 fmt::format("{}.{}", name, GeneratorTag::extension);
115 std::ofstream ofs;
116 ofs.open(path, std::ofstream::out | std::ofstream::trunc);
117 ofs << buffer.str();
118
119 ofs.close();
120
121 LOG_INFO("Written {} diagram to {}", name, path.string());
122 }
123 else {
124 LOG_INFO("Serialization to {} not supported for {}",
125 GeneratorTag::extension, name);
126 }
127}
128
129template <typename DiagramConfig>
130void generate_diagram_impl(const std::string &name,
131 std::shared_ptr<clanguml::config::diagram> diagram,
133 const std::vector<std::string> &translation_units,
134 const cli::runtime_config &runtime_config, std::function<void()> &&progress)
135{
136 using diagram_config = DiagramConfig;
137 using diagram_model = typename diagram_model_t<DiagramConfig>::type;
138 using diagram_visitor = typename diagram_visitor_t<DiagramConfig>::type;
139
140 auto model = clanguml::common::generators::generate<diagram_model,
141 diagram_config, diagram_visitor>(db, diagram->name,
142 dynamic_cast<diagram_config &>(*diagram), translation_units,
143 runtime_config.verbose, std::move(progress));
144
145 if constexpr (std::is_same_v<DiagramConfig, config::sequence_diagram>) {
146 if (runtime_config.print_from) {
147 auto from_values = model->list_from_values();
148
149 if (logging::logger_type() == logging::logger_type_t::text) {
150 for (const auto &from : from_values) {
151 std::cout << from << '\n';
152 }
153 }
154 else {
155 inja::json j = inja::json::array();
156 for (const auto &from : from_values) {
157 j.emplace_back(logging::escape_json(from));
158 }
159 std::cout << j.dump();
160 }
161
162 return;
163 }
164 if (runtime_config.print_to) {
165 auto to_values = model->list_to_values();
166 if (logging::logger_type() == logging::logger_type_t::text) {
167 for (const auto &to : to_values) {
168 std::cout << to << '\n';
169 }
170 }
171 else {
172 inja::json j = inja::json::array();
173 for (const auto &to : to_values) {
174 j.emplace_back(logging::escape_json(to));
175 }
176 std::cout << j.dump();
177 }
178 return;
179 }
180 }
181
182 for (const auto generator_type : runtime_config.generators) {
183 if (generator_type == generator_type_t::plantuml) {
186 runtime_config.output_directory, name, diagram, model);
187 }
188 else if (generator_type == generator_type_t::json) {
191 runtime_config.output_directory, name, diagram, model);
192 }
193 else if (generator_type == generator_type_t::mermaid) {
196 runtime_config.output_directory, name, diagram, model);
197 }
198 else if (generator_type == generator_type_t::graphml) {
201 runtime_config.output_directory, name, diagram, model);
202 }
203
204 // Convert plantuml or mermaid to an image using command provided
205 // in the command line arguments
206 if (runtime_config.render_diagrams) {
207 render_diagram(generator_type, diagram);
208 }
209 }
210}
211} // namespace detail
212
213void generate_diagram(const std::string &name,
214 std::shared_ptr<clanguml::config::diagram> diagram,
216 const std::vector<std::string> &translation_units,
217 const cli::runtime_config &runtime_config, std::function<void()> &&progress)
218{
221
226
227 if (diagram->type() == diagram_t::kClass) {
228 detail::generate_diagram_impl<class_diagram>(name, diagram, db,
229 translation_units, runtime_config, std::move(progress));
230 }
231 else if (diagram->type() == diagram_t::kSequence) {
232 detail::generate_diagram_impl<sequence_diagram>(name, diagram, db,
233 translation_units, runtime_config, std::move(progress));
234 }
235 else if (diagram->type() == diagram_t::kPackage) {
236 detail::generate_diagram_impl<package_diagram>(name, diagram, db,
237 translation_units, runtime_config, std::move(progress));
238 }
239 else if (diagram->type() == diagram_t::kInclude) {
240 detail::generate_diagram_impl<include_diagram>(name, diagram, db,
241 translation_units, runtime_config, std::move(progress));
242 }
243}
244
245int generate_diagrams(const std::vector<std::string> &diagram_names,
247 const cli::runtime_config &runtime_config,
248 const std::map<std::string, std::vector<std::string>>
249 &translation_units_map)
250{
251 util::thread_pool_executor generator_executor{runtime_config.thread_count};
252 std::vector<std::future<void>> futs;
253
254 std::unique_ptr<progress_indicator_base> indicator;
255
256 if (runtime_config.progress) {
257 if (clanguml::logging::logger_type() == logging::logger_type_t::text) {
258 std::cout
259 << termcolor::white
260 << "Processing translation units and generating diagrams:\n";
261 indicator = std::make_unique<progress_indicator>();
262 }
263 else {
264 indicator = std::make_unique<json_logger_progress_indicator>();
265 }
266 }
267
268 std::vector<std::exception_ptr> errors;
269
270 for (const auto &[name, diagram] : config.diagrams) {
271 // If there are any specific diagram names provided on the command
272 // line, and this diagram is not in that list - skip it
273 if (!diagram_names.empty() && !util::contains(diagram_names, name))
274 continue;
275
276 // If none of the generators supports the diagram type - skip it
277 bool at_least_one_generator_supports_diagram_type{false};
278 for (const auto generator_type : runtime_config.generators) {
279 if (generator_type == generator_type_t::plantuml) {
280 if (generator_supports_diagram_type<plantuml_generator_tag>(
281 diagram->type()))
282 at_least_one_generator_supports_diagram_type = true;
283 }
284 else if (generator_type == generator_type_t::json) {
285 if (generator_supports_diagram_type<json_generator_tag>(
286 diagram->type()))
287 at_least_one_generator_supports_diagram_type = true;
288 }
289 else if (generator_type == generator_type_t::mermaid) {
290 if (generator_supports_diagram_type<mermaid_generator_tag>(
291 diagram->type()))
292 at_least_one_generator_supports_diagram_type = true;
293 }
294 else if (generator_type == generator_type_t::graphml) {
295 if (generator_supports_diagram_type<graphml_generator_tag>(
296 diagram->type()))
297 at_least_one_generator_supports_diagram_type = true;
298 }
299 }
300
301 if (!at_least_one_generator_supports_diagram_type) {
302 LOG_INFO("Diagram '{}' not supported by any of selected "
303 "generators - skipping...",
304 name);
305 continue;
306 }
307
308 const auto &valid_translation_units = translation_units_map.at(name);
309
310 LOG_DBG("Found {} possible translation units for diagram '{}'",
311 valid_translation_units.size(), name);
312
313 const auto matching_commands_count =
314 db->count_matching_commands(valid_translation_units);
315
316 if (matching_commands_count == 0) {
317 const auto error_msg = fmt::format(
318 "Diagram '{}' generation failed: no translation units "
319 "found. Please make sure that your 'glob' patterns match "
320 "at least 1 file in 'compile_commands.json'.",
321 name);
322
323 if (indicator) {
324 indicator->add_progress_bar(
325 name, 0, diagram_type_to_color(diagram->type()));
326 indicator->fail(name);
327 try {
328 throw std::runtime_error(error_msg);
329 }
330 catch (std::runtime_error &e) {
331 errors.emplace_back(std::current_exception());
332 }
333 }
334 else {
335 LOG_ERROR(error_msg);
336 }
337
338 continue;
339 }
340
341 LOG_DBG("Found {} matching translation unit commands for diagram {}",
342 matching_commands_count, name);
343
344 auto generator = [&name = name, &diagram = diagram, &indicator,
345 db = std::ref(*db), matching_commands_count,
346 translation_units = valid_translation_units,
347 runtime_config]() mutable -> void {
348 try {
349 if (indicator) {
350 indicator->add_progress_bar(name, matching_commands_count,
351 diagram_type_to_color(diagram->type()));
352
353 generate_diagram(name, diagram, db, translation_units,
354 runtime_config, [&indicator, &name]() {
355 if (indicator)
356 indicator->increment(name);
357 });
358
359 if (indicator)
360 indicator->complete(name);
361 }
362 else {
363 generate_diagram(name, diagram, db, translation_units,
364 runtime_config, {});
365 }
366 }
368 if (indicator)
369 indicator->fail(name);
370 throw std::move(e);
371 }
372 catch (std::exception &e) {
373 if (indicator)
374 indicator->fail(name);
375
376 LOG_ERROR(
377 "Failed to generate diagram '{}': {}", name, e.what());
378
379 throw std::runtime_error(fmt::format(
380 "Failed to generate diagram '{}': {}", name, e.what()));
381 }
382 };
383
384 futs.emplace_back(generator_executor.add(std::move(generator)));
385 }
386
387 for (auto &fut : futs) {
388 try {
389 fut.get();
390 }
391 catch (std::exception &e) {
392 errors.emplace_back(std::current_exception());
393 }
394 }
395
396 if (runtime_config.progress &&
397 clanguml::logging::logger_type() == logging::logger_type_t::text) {
398 indicator->stop();
399 if (errors.empty()) {
400 std::cout << termcolor::white << "Done\n";
401 }
402 else {
403 std::cout << termcolor::white << "\n";
404 }
405 std::cout << termcolor::reset;
406 }
407
408 if (errors.empty())
409 return 0;
410
411 for (auto &e : errors) {
412 try {
413 std::rethrow_exception(e);
414 }
417 logging::logger_type_t::text) {
418
419 fmt::println("ERROR: Failed to generate {} diagram '{}' due to "
420 "following issues:",
421 e.diagram_type(), e.diagram_name());
422 for (const auto &d : e.diagnostics) {
423 fmt::println(" - {}", d);
424 }
425 fmt::println("");
426 }
427 else {
428 inja::json j;
429 j["diagram_name"] = e.diagram_name();
430 j["clang_errors"] = inja::json::array();
431 for (const auto &d : e.diagnostics) {
432 j["clang_errors"].emplace_back(d);
433 }
434
435 spdlog::get("clanguml-logger")
436 ->log(spdlog::level::err,
437 fmt::runtime(
438 R"("file": "{}", "line": {}, "message": {})"),
439 FILENAME_, __LINE__, j.dump());
440 }
441 }
442 catch (const std::exception &e) {
444 logging::logger_type_t::text) {
445 fmt::println("ERROR: {}", e.what());
446 }
447 else {
448 LOG_ERROR("{}", e.what());
449 }
450 }
451 }
452
453 return 1;
454}
455
456indicators::Color diagram_type_to_color(model::diagram_t diagram_type)
457{
458 switch (diagram_type) {
460 return indicators::Color::yellow;
462 return indicators::Color::blue;
464 return indicators::Color::cyan;
466 return indicators::Color::magenta;
467 default:
468 return indicators::Color::unspecified;
469 }
470}
471
472} // namespace clanguml::common::generators