Enabled type_aliases config option for sequence diagrams (#224)
This commit is contained in:
@@ -1,5 +1,6 @@
|
|||||||
# CHANGELOG
|
# CHANGELOG
|
||||||
|
|
||||||
|
* Enabled type_aliases config option for sequence diagrams (#224)
|
||||||
* Refactored and unified JSON generators output (#223)
|
* Refactored and unified JSON generators output (#223)
|
||||||
* Added support for C++20 module based packages in class diagrams (#101)
|
* Added support for C++20 module based packages in class diagrams (#101)
|
||||||
* Added support for class diagram filtering based on C++20 modules (#195)
|
* Added support for class diagram filtering based on C++20 modules (#195)
|
||||||
|
|||||||
@@ -217,6 +217,7 @@ void inheritable_diagram_options::inherit(
|
|||||||
parent.generate_condition_statements);
|
parent.generate_condition_statements);
|
||||||
debug_mode.override(parent.debug_mode);
|
debug_mode.override(parent.debug_mode);
|
||||||
generate_metadata.override(parent.generate_metadata);
|
generate_metadata.override(parent.generate_metadata);
|
||||||
|
type_aliases.override(parent.type_aliases);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string inheritable_diagram_options::simplify_template_type(
|
std::string inheritable_diagram_options::simplify_template_type(
|
||||||
|
|||||||
@@ -538,7 +538,8 @@ struct inheritable_diagram_options {
|
|||||||
option<std::filesystem::path> base_directory{"__parent_path"};
|
option<std::filesystem::path> base_directory{"__parent_path"};
|
||||||
option<bool> generate_system_headers{"generate_system_headers", false};
|
option<bool> generate_system_headers{"generate_system_headers", false};
|
||||||
option<relationship_hints_t> relationship_hints{"relationship_hints"};
|
option<relationship_hints_t> relationship_hints{"relationship_hints"};
|
||||||
option<type_aliases_t> type_aliases{"type_aliases"};
|
option<type_aliases_t> type_aliases{
|
||||||
|
"type_aliases", option_inherit_mode::kAppend};
|
||||||
option<comment_parser_t> comment_parser{
|
option<comment_parser_t> comment_parser{
|
||||||
"comment_parser", comment_parser_t::plain};
|
"comment_parser", comment_parser_t::plain};
|
||||||
option<bool> combine_free_functions_into_file_participants{
|
option<bool> combine_free_functions_into_file_participants{
|
||||||
|
|||||||
@@ -25,6 +25,18 @@ namespace config {
|
|||||||
|
|
||||||
template <typename T> void append_value(T &l, const T &r) { l = r; }
|
template <typename T> void append_value(T &l, const T &r) { l = r; }
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
void append_value(std::vector<T> &l, const std::vector<T> &r)
|
||||||
|
{
|
||||||
|
l.insert(std::end(l), r.begin(), r.end());
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename K, typename V>
|
||||||
|
void append_value(std::map<K, V> &l, const std::map<K, V> &r)
|
||||||
|
{
|
||||||
|
l.insert(r.begin(), r.end());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Possible option inheritance methods from top level to diagram level.
|
* Possible option inheritance methods from top level to diagram level.
|
||||||
*/
|
*/
|
||||||
@@ -102,9 +114,7 @@ template <typename T> struct option {
|
|||||||
has_value = true;
|
has_value = true;
|
||||||
}
|
}
|
||||||
else if (!is_declared && o.is_declared) {
|
else if (!is_declared && o.is_declared) {
|
||||||
value = o.value;
|
set(o.value);
|
||||||
is_declared = true;
|
|
||||||
has_value = true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -205,6 +205,7 @@ types:
|
|||||||
after: !optional [string]
|
after: !optional [string]
|
||||||
cmd: !optional string
|
cmd: !optional string
|
||||||
relative_to: !optional string
|
relative_to: !optional string
|
||||||
|
type_aliases: !optional map_t<string;string>
|
||||||
using_namespace: !optional [string, [string]]
|
using_namespace: !optional [string, [string]]
|
||||||
generate_metadata: !optional bool
|
generate_metadata: !optional bool
|
||||||
title: !optional string
|
title: !optional string
|
||||||
@@ -342,6 +343,7 @@ root:
|
|||||||
package_type: !optional package_type_t
|
package_type: !optional package_type_t
|
||||||
generate_template_argument_dependencies: !optional bool
|
generate_template_argument_dependencies: !optional bool
|
||||||
skip_redundant_dependencies: !optional bool
|
skip_redundant_dependencies: !optional bool
|
||||||
|
type_aliases: !optional map_t<string;string>
|
||||||
)";
|
)";
|
||||||
|
|
||||||
} // namespace clanguml::config
|
} // namespace clanguml::config
|
||||||
@@ -663,6 +663,7 @@ template <> struct convert<sequence_diagram> {
|
|||||||
get_option(node, rhs.generate_method_arguments);
|
get_option(node, rhs.generate_method_arguments);
|
||||||
get_option(node, rhs.generate_message_comments);
|
get_option(node, rhs.generate_message_comments);
|
||||||
get_option(node, rhs.message_comment_width);
|
get_option(node, rhs.message_comment_width);
|
||||||
|
get_option(node, rhs.type_aliases);
|
||||||
|
|
||||||
get_option(node, rhs.get_relative_to());
|
get_option(node, rhs.get_relative_to());
|
||||||
|
|
||||||
@@ -836,6 +837,7 @@ template <> struct convert<config> {
|
|||||||
get_option(node, rhs.generate_condition_statements);
|
get_option(node, rhs.generate_condition_statements);
|
||||||
get_option(node, rhs.generate_message_comments);
|
get_option(node, rhs.generate_message_comments);
|
||||||
get_option(node, rhs.message_comment_width);
|
get_option(node, rhs.message_comment_width);
|
||||||
|
get_option(node, rhs.type_aliases);
|
||||||
|
|
||||||
rhs.base_directory.set(node["__parent_path"].as<std::string>());
|
rhs.base_directory.set(node["__parent_path"].as<std::string>());
|
||||||
get_option(node, rhs.get_relative_to());
|
get_option(node, rhs.get_relative_to());
|
||||||
|
|||||||
@@ -114,8 +114,8 @@ bool translation_unit_visitor::VisitCXXRecordDecl(
|
|||||||
forward_declarations_.erase(class_id);
|
forward_declarations_.erase(class_id);
|
||||||
|
|
||||||
if (diagram().should_include(class_model)) {
|
if (diagram().should_include(class_model)) {
|
||||||
LOG_DBG("Adding class {} with id {}", class_model.full_name(false),
|
LOG_DBG("Adding class participant {} with id {}",
|
||||||
class_model.id());
|
class_model.full_name(false), class_model.id());
|
||||||
|
|
||||||
assert(class_model.id() == class_id);
|
assert(class_model.id() == class_id);
|
||||||
|
|
||||||
@@ -166,7 +166,8 @@ bool translation_unit_visitor::VisitClassTemplateDecl(
|
|||||||
forward_declarations_.erase(id);
|
forward_declarations_.erase(id);
|
||||||
|
|
||||||
if (diagram().should_include(*class_model_ptr)) {
|
if (diagram().should_include(*class_model_ptr)) {
|
||||||
LOG_DBG("Adding class template {} with id {}", class_full_name, id);
|
LOG_DBG("Adding class template participant {} with id {}",
|
||||||
|
class_full_name, id);
|
||||||
|
|
||||||
context().set_caller_id(id);
|
context().set_caller_id(id);
|
||||||
context().update(declaration);
|
context().update(declaration);
|
||||||
@@ -212,7 +213,8 @@ bool translation_unit_visitor::VisitClassTemplateSpecializationDecl(
|
|||||||
forward_declarations_.erase(id);
|
forward_declarations_.erase(id);
|
||||||
|
|
||||||
if (diagram().should_include(*template_specialization_ptr)) {
|
if (diagram().should_include(*template_specialization_ptr)) {
|
||||||
LOG_DBG("Adding class template specialization {} with id {}",
|
LOG_DBG(
|
||||||
|
"Adding class template specialization participant {} with id {}",
|
||||||
class_full_name, id);
|
class_full_name, id);
|
||||||
|
|
||||||
context().set_caller_id(id);
|
context().set_caller_id(id);
|
||||||
@@ -2047,12 +2049,12 @@ void translation_unit_visitor::process_template_specialization_argument(
|
|||||||
argument.set_type(type_name);
|
argument.set_type(type_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG_TRACE("Adding template instantiation argument {}",
|
|
||||||
argument.to_string(config().using_namespace(), false));
|
|
||||||
|
|
||||||
simplify_system_template(
|
simplify_system_template(
|
||||||
argument, argument.to_string(config().using_namespace(), false));
|
argument, argument.to_string(config().using_namespace(), false));
|
||||||
|
|
||||||
|
LOG_TRACE("Adding template instantiation argument {}",
|
||||||
|
argument.to_string(config().using_namespace(), false));
|
||||||
|
|
||||||
template_instantiation.add_template(std::move(argument));
|
template_instantiation.add_template(std::move(argument));
|
||||||
}
|
}
|
||||||
else if (argument_kind == clang::TemplateArgument::Integral) {
|
else if (argument_kind == clang::TemplateArgument::Integral) {
|
||||||
|
|||||||
@@ -312,7 +312,7 @@ private:
|
|||||||
bool should_include(const clang::ClassTemplateDecl *decl) const;
|
bool should_include(const clang::ClassTemplateDecl *decl) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @todo Refactor this group of methods to @ref template_builder
|
* @todo #227 Refactor this group of methods to @ref template_builder
|
||||||
*
|
*
|
||||||
* @{
|
* @{
|
||||||
*/
|
*/
|
||||||
|
|||||||
14
tests/t20039/.clang-uml
Normal file
14
tests/t20039/.clang-uml
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
diagrams:
|
||||||
|
t20039_sequence:
|
||||||
|
type: sequence
|
||||||
|
glob:
|
||||||
|
- t20039.cc
|
||||||
|
include:
|
||||||
|
namespaces:
|
||||||
|
- clanguml::t20039
|
||||||
|
using_namespace: clanguml::t20039
|
||||||
|
type_aliases:
|
||||||
|
"std::vector<int>": int_vec_t
|
||||||
|
"std::map<int,int>": int_map_t
|
||||||
|
from:
|
||||||
|
- function: "clanguml::t20039::tmain()"
|
||||||
34
tests/t20039/t20039.cc
Normal file
34
tests/t20039/t20039.cc
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
#include <map>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace clanguml {
|
||||||
|
namespace t20039 {
|
||||||
|
|
||||||
|
template <typename T> struct A {
|
||||||
|
std::vector<std::vector<T>> a(T p) { return {}; }
|
||||||
|
};
|
||||||
|
|
||||||
|
struct R {
|
||||||
|
A<int> a_int;
|
||||||
|
A<std::vector<int>> a_intvec;
|
||||||
|
A<std::map<int, int>> a_intmap;
|
||||||
|
|
||||||
|
void run()
|
||||||
|
{
|
||||||
|
a_int.a({});
|
||||||
|
a_intvec.a({});
|
||||||
|
a_intmap.a({});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
int tmain()
|
||||||
|
{
|
||||||
|
R r;
|
||||||
|
|
||||||
|
r.run();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
63
tests/t20039/test_case.h
Normal file
63
tests/t20039/test_case.h
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
/**
|
||||||
|
* tests/t20039/test_case.h
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021-2024 Bartek Kryza <bkryza@gmail.com>
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
TEST_CASE("t20039", "[test-case][sequence]")
|
||||||
|
{
|
||||||
|
auto [config, db] = load_config("t20039");
|
||||||
|
|
||||||
|
auto diagram = config.diagrams["t20039_sequence"];
|
||||||
|
|
||||||
|
REQUIRE(diagram->name == "t20039_sequence");
|
||||||
|
|
||||||
|
auto model = generate_sequence_diagram(*db, diagram);
|
||||||
|
|
||||||
|
REQUIRE(model->name() == "t20039_sequence");
|
||||||
|
|
||||||
|
{
|
||||||
|
auto src = generate_sequence_puml(diagram, *model);
|
||||||
|
AliasMatcher _A(src);
|
||||||
|
|
||||||
|
REQUIRE_THAT(src, StartsWith("@startuml"));
|
||||||
|
REQUIRE_THAT(src, EndsWith("@enduml\n"));
|
||||||
|
|
||||||
|
// Check if all calls exist
|
||||||
|
REQUIRE_THAT(src, HasCall(_A("tmain()"), _A("R"), "run()"));
|
||||||
|
REQUIRE_THAT(src, HasCall(_A("R"), _A("A<int>"), "a(int)"));
|
||||||
|
REQUIRE_THAT(src, HasCall(_A("R"), _A("A<int_vec_t>"), "a(int_vec_t)"));
|
||||||
|
REQUIRE_THAT(src, HasCall(_A("R"), _A("A<int_map_t>"), "a(int_map_t)"));
|
||||||
|
|
||||||
|
save_puml(config.output_directory(), diagram->name + ".puml", src);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
auto j = generate_sequence_json(diagram, *model);
|
||||||
|
|
||||||
|
using namespace json;
|
||||||
|
|
||||||
|
save_json(config.output_directory(), diagram->name + ".json", j);
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
auto src = generate_sequence_mermaid(diagram, *model);
|
||||||
|
|
||||||
|
mermaid::AliasMatcher _A(src);
|
||||||
|
using mermaid::IsClass;
|
||||||
|
|
||||||
|
save_mermaid(config.output_directory(), diagram->name + ".mmd", src);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -458,6 +458,7 @@ using namespace clanguml::test::matchers;
|
|||||||
#include "t20036/test_case.h"
|
#include "t20036/test_case.h"
|
||||||
#include "t20037/test_case.h"
|
#include "t20037/test_case.h"
|
||||||
#include "t20038/test_case.h"
|
#include "t20038/test_case.h"
|
||||||
|
#include "t20039/test_case.h"
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Package diagram tests
|
/// Package diagram tests
|
||||||
|
|||||||
@@ -328,6 +328,9 @@ test_cases:
|
|||||||
- name: t20038
|
- name: t20038
|
||||||
title: Sequence diagram comment decorator test case
|
title: Sequence diagram comment decorator test case
|
||||||
description:
|
description:
|
||||||
|
- name: t20039
|
||||||
|
title: Test case for type aliases config option in sequence diagrams
|
||||||
|
description:
|
||||||
Package diagrams:
|
Package diagrams:
|
||||||
- name: t30001
|
- name: t30001
|
||||||
title: Basic package diagram test case
|
title: Basic package diagram test case
|
||||||
|
|||||||
@@ -410,6 +410,37 @@ TEST_CASE("Test config full clang uml dump", "[unit-test]")
|
|||||||
CHECK(cfg.diagrams.size() == 32);
|
CHECK(cfg.diagrams.size() == 32);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE("Test config type aliases", "[unit-test]")
|
||||||
|
{
|
||||||
|
auto cfg = clanguml::config::load("./test_config_data/type_aliases.yml");
|
||||||
|
|
||||||
|
CHECK(cfg.diagrams.size() == 2);
|
||||||
|
auto &def = *cfg.diagrams["class_diagram"];
|
||||||
|
CHECK(
|
||||||
|
def.simplify_template_type(
|
||||||
|
"ns1::ns2::container<ns2::key_t,ns2::value_t>") == "custom_map_t");
|
||||||
|
CHECK(def.simplify_template_type(
|
||||||
|
"ns1::ns2::container<ns1::ns2::key_t,std::string>") ==
|
||||||
|
"string_map_t");
|
||||||
|
CHECK(
|
||||||
|
def.simplify_template_type("std::basic_string<char>") == "std::string");
|
||||||
|
CHECK(def.simplify_template_type("std::basic_string<char32_t>") ==
|
||||||
|
"unicode_t");
|
||||||
|
|
||||||
|
def = *cfg.diagrams["sequence_diagram"];
|
||||||
|
CHECK(
|
||||||
|
def.simplify_template_type(
|
||||||
|
"ns1::ns2::container<ns2::key_t,ns2::value_t>") == "custom_map_t");
|
||||||
|
CHECK(def.simplify_template_type(
|
||||||
|
"ns1::ns2::Object::iterator<std::weak_ptr<Object> "
|
||||||
|
"*,std::vector<std::weak_ptr<Object>,std::allocator<std::weak_"
|
||||||
|
"ptr<Object>>>>") == "ObjectPtrIt");
|
||||||
|
CHECK(
|
||||||
|
def.simplify_template_type("std::basic_string<char>") == "std::string");
|
||||||
|
CHECK(def.simplify_template_type("std::vector<std::basic_string<char>>") ==
|
||||||
|
"std::vector<std::string>");
|
||||||
|
}
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Main test function
|
/// Main test function
|
||||||
///
|
///
|
||||||
|
|||||||
17
tests/test_config_data/type_aliases.yml
Normal file
17
tests/test_config_data/type_aliases.yml
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
type_aliases:
|
||||||
|
"ns1::ns2::container<ns2::key_t,ns2::value_t>": custom_map_t
|
||||||
|
diagrams:
|
||||||
|
class_diagram:
|
||||||
|
type: class
|
||||||
|
glob:
|
||||||
|
- test.cc
|
||||||
|
type_aliases:
|
||||||
|
"ns1::ns2::container<ns1::ns2::key_t,std::string>": string_map_t
|
||||||
|
"std::basic_string<char32_t>": unicode_t
|
||||||
|
sequence_diagram:
|
||||||
|
type: class
|
||||||
|
relative_to: .
|
||||||
|
glob:
|
||||||
|
- test.cc
|
||||||
|
type_aliases:
|
||||||
|
"ns1::ns2::Object::iterator<std::weak_ptr<Object> *,std::vector<std::weak_ptr<Object>,std::allocator<std::weak_ptr<Object>>>>": "ObjectPtrIt"
|
||||||
Reference in New Issue
Block a user