Added include_relations_also_as_members flag

This commit is contained in:
Bartek Kryza
2021-07-24 19:40:12 +02:00
parent adb1296aa5
commit ea8f0e83c1
10 changed files with 297 additions and 41 deletions

View File

@@ -15,6 +15,7 @@
* [t00014](./test_cases/t00014.md) - Alias template instantiation
* [t00015](./test_cases/t00015.md) - Namespace fun
* [t00016](./test_cases/t00016.md) - Unnamed enums and empty templates
* [t00017](./test_cases/t00017.md) - Test include relations also as members flag
## Sequence diagrams
* [t20001](./test_cases/t20001.md) - Basic sequence diagram
## Configuration diagrams

79
docs/test_cases/t00017.md Normal file
View File

@@ -0,0 +1,79 @@
# t00017 - Test include relations also as members flag
## Config
```yaml
compilation_database_dir: ..
output_directory: puml
diagrams:
t00017_class:
type: class
include_relations_also_as_members: false
glob:
- ../../tests/t00017/t00017.cc
using_namespace:
- clanguml::t00017
include:
namespaces:
- clanguml::t00017
```
## Source code
```cpp
namespace clanguml {
namespace t00017 {
class A {
};
class B {
};
class C {
};
class D {
};
class E {
};
class F {
};
class G {
};
class H {
};
class I {
};
class J {
};
class K {
};
class R {
private:
int some_int;
int *some_int_pointer;
int **some_int_pointer_pointer;
int &some_int_reference;
A a;
B *b;
C &c;
const D *d;
const E &e{};
F &&f;
G **g;
H ***h;
I *&i;
volatile J *j;
mutable K *k;
};
}
}
```
## Generated UML diagrams
![t00017_class](./t00017_class.png "Test include relations also as members flag")

Binary file not shown.

After

Width:  |  Height:  |  Size: 24 KiB

View File

@@ -165,6 +165,8 @@ struct class_diagram : public diagram {
virtual ~class_diagram() = default;
std::vector<std::string> classes;
bool include_relations_also_as_members{true};
bool has_class(std::string clazz)
{
for (const auto &c : classes) {
@@ -316,6 +318,11 @@ template <> struct convert<class_diagram> {
if (!decode_diagram(node, rhs))
return false;
if (node["include_relations_also_as_members"])
rhs.include_relations_also_as_members =
node["include_relations_also_as_members"]
.as<decltype(rhs.include_relations_also_as_members)>();
return true;
}
};

View File

@@ -195,6 +195,56 @@ public:
ostr << std::endl;
}
//
// Process relationships
//
std::set<std::string> rendered_relations;
std::stringstream all_relations_str;
for (const auto &r : c.relationships) {
if (!m_config.should_include_relationship(name(r.type)))
continue;
std::stringstream relstr;
std::string destination;
try {
if (r.destination.find("#") != std::string::npos ||
r.destination.find("@") != std::string::npos) {
destination = m_model.usr_to_name(uns, r.destination);
// If something went wrong and we have an empty destination
// generate the relationship but comment it out for
// debugging
if (destination.empty()) {
relstr << "' ";
destination = r.destination;
}
}
else {
destination = r.destination;
}
relstr << m_model.to_alias(
uns, ns_relative(uns, c.full_name(uns)))
<< " " << to_string(r.type) << " "
<< m_model.to_alias(uns, ns_relative(uns, destination));
if (!r.label.empty()) {
relstr << " : " << to_string(r.scope) << r.label;
rendered_relations.emplace(r.label);
}
relstr << std::endl;
all_relations_str << relstr.str();
}
catch (error::uml_alias_missing &e) {
LOG_ERROR("Skipping {} relation from {} to {} due "
"to: {}",
to_string(r.type), c.full_name(uns), destination, e.what());
}
}
//
// Process members
//
@@ -202,6 +252,10 @@ public:
if (!m_config.should_include(m.scope))
continue;
if (!m_config.include_relations_also_as_members &&
rendered_relations.find(m.name) != rendered_relations.end())
continue;
if (m.is_static)
ostr << "{static} ";
@@ -228,47 +282,8 @@ public:
}
}
for (const auto &r : c.relationships) {
if (!m_config.should_include_relationship(name(r.type)))
continue;
std::stringstream relstr;
std::string destination;
try {
if (r.destination.find("#") != std::string::npos ||
r.destination.find("@") != std::string::npos) {
destination = m_model.usr_to_name(uns, r.destination);
// If something went wrong and we have an empty destination
// generate the relationship but comment it out for
// debugging
if (destination.empty()) {
relstr << "' ";
destination = r.destination;
}
}
else {
destination = r.destination;
}
relstr << m_model.to_alias(
uns, ns_relative(uns, c.full_name(uns)))
<< " " << to_string(r.type) << " "
<< m_model.to_alias(uns, ns_relative(uns, destination));
if (!r.label.empty())
relstr << " : " << to_string(r.scope) << r.label;
relstr << std::endl;
ostr << relstr.str();
}
catch (error::uml_alias_missing &e) {
LOG_ERROR("Skipping {} relation from {} to {} due "
"to: {}",
to_string(r.type), c.full_name(uns), destination, e.what());
}
}
// Print relationships
ostr << all_relations_str.str();
}
void generate(const enum_ &e, std::ostream &ostr) const

13
tests/t00017/.clanguml Normal file
View File

@@ -0,0 +1,13 @@
compilation_database_dir: ..
output_directory: puml
diagrams:
t00017_class:
type: class
include_relations_also_as_members: false
glob:
- ../../tests/t00017/t00017.cc
using_namespace:
- clanguml::t00017
include:
namespaces:
- clanguml::t00017

55
tests/t00017/t00017.cc Normal file
View File

@@ -0,0 +1,55 @@
namespace clanguml {
namespace t00017 {
class A {
};
class B {
};
class C {
};
class D {
};
class E {
};
class F {
};
class G {
};
class H {
};
class I {
};
class J {
};
class K {
};
class R {
private:
int some_int;
int *some_int_pointer;
int **some_int_pointer_pointer;
int &some_int_reference;
A a;
B *b;
C &c;
const D *d;
const E &e{};
F &&f;
G **g;
H ***h;
I *&i;
volatile J *j;
mutable K *k;
};
}
}

82
tests/t00017/test_case.h Normal file
View File

@@ -0,0 +1,82 @@
/**
* tests/t00017/test_case.cc
*
* Copyright (c) 2021 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("t00017", "[test-case][class]")
{
auto [config, db] = load_config("t00017");
auto diagram = config.diagrams["t00017_class"];
REQUIRE(diagram->name == "t00017_class");
REQUIRE(diagram->include.namespaces.size() == 1);
REQUIRE_THAT(diagram->include.namespaces,
VectorContains(std::string{"clanguml::t00017"}));
REQUIRE(diagram->exclude.namespaces.size() == 0);
REQUIRE(diagram->should_include("clanguml::t00017::A"));
REQUIRE(diagram->should_include("clanguml::t00017::B"));
REQUIRE(diagram->should_include("clanguml::t00017::C"));
REQUIRE(diagram->should_include("clanguml::t00017::D"));
auto model = generate_class_diagram(db, diagram);
REQUIRE(model.name == "t00017_class");
auto puml = generate_class_puml(diagram, model);
AliasMatcher _A(puml);
REQUIRE_THAT(puml, StartsWith("@startuml"));
REQUIRE_THAT(puml, EndsWith("@enduml\n"));
REQUIRE_THAT(puml, IsClass(_A("A")));
REQUIRE_THAT(puml, IsClass(_A("B")));
REQUIRE_THAT(puml, IsClass(_A("C")));
REQUIRE_THAT(puml, IsClass(_A("D")));
REQUIRE_THAT(puml, IsClass(_A("E")));
REQUIRE_THAT(puml, IsClass(_A("F")));
REQUIRE_THAT(puml, IsClass(_A("G")));
REQUIRE_THAT(puml, IsClass(_A("H")));
REQUIRE_THAT(puml, IsClass(_A("I")));
REQUIRE_THAT(puml, IsClass(_A("J")));
REQUIRE_THAT(puml, IsClass(_A("K")));
REQUIRE_THAT(puml, IsClass(_A("R")));
REQUIRE_THAT(puml, (IsField<Private>("some_int", "int")));
REQUIRE_THAT(puml, (IsField<Private>("some_int_pointer", "int*")));
REQUIRE_THAT(puml, (IsField<Private>("some_int_pointer_pointer", "int**")));
// Relationship members should not be rendered as part of this testcase
REQUIRE_THAT(puml, !(IsField<Private>("a", _A("A"))));
REQUIRE_THAT(puml, !(IsField<Private>("b", _A("B"))));
REQUIRE_THAT(puml, IsAggregation(_A("R"), _A("A"), "-a"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("B"), "-b"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("C"), "-c"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("D"), "-d"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("E"), "-e"));
REQUIRE_THAT(puml, IsAggregation(_A("R"), _A("F"), "-f"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("G"), "-g"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("H"), "-h"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("I"), "-i"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("J"), "-j"));
REQUIRE_THAT(puml, IsAssociation(_A("R"), _A("K"), "-k"));
save_puml(
"./" + config.output_directory + "/" + diagram->name + ".puml", puml);
}

View File

@@ -120,6 +120,7 @@ using namespace clanguml::test::matchers;
#include "t00014/test_case.h"
#include "t00015/test_case.h"
#include "t00016/test_case.h"
#include "t00017/test_case.h"
//
// Sequence diagram tests

View File

@@ -45,6 +45,9 @@ test_cases:
- name: t00016
title: Unnamed enums and empty templates
description:
- name: t00017
title: Test include relations also as members flag
description:
Sequence diagrams:
- name: t20001
title: Basic sequence diagram