Fixed generation of lambda names in class diagrams (#78)
This commit is contained in:
@@ -828,9 +828,13 @@ void translation_unit_visitor::process_method(
|
|||||||
if (mf.isDefaulted() && !mf.isExplicitlyDefaulted())
|
if (mf.isDefaulted() && !mf.isExplicitlyDefaulted())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
auto method_return_type =
|
||||||
|
common::to_string(mf.getReturnType(), mf.getASTContext());
|
||||||
|
|
||||||
|
ensure_lambda_type_is_relative(method_return_type);
|
||||||
|
|
||||||
class_method method{common::access_specifier_to_access_t(mf.getAccess()),
|
class_method method{common::access_specifier_to_access_t(mf.getAccess()),
|
||||||
util::trim(mf.getNameAsString()),
|
util::trim(mf.getNameAsString()), std::move(method_return_type)};
|
||||||
common::to_string(mf.getReturnType(), mf.getASTContext())};
|
|
||||||
|
|
||||||
method.is_pure_virtual(mf.isPure());
|
method.is_pure_virtual(mf.isPure());
|
||||||
method.is_virtual(mf.isVirtual());
|
method.is_virtual(mf.isVirtual());
|
||||||
@@ -923,13 +927,6 @@ bool translation_unit_visitor::find_relationships(const clang::QualType &type,
|
|||||||
const auto *type_instantiation_decl =
|
const auto *type_instantiation_decl =
|
||||||
type->getAs<clang::TemplateSpecializationType>();
|
type->getAs<clang::TemplateSpecializationType>();
|
||||||
|
|
||||||
// if (type_instantiation_decl != nullptr) {
|
|
||||||
// if (type_instantiation_decl->isTypeAlias())
|
|
||||||
// type_instantiation_decl =
|
|
||||||
// type_instantiation_decl->getAliasedType()
|
|
||||||
// ->getAs<clang::TemplateSpecializationType>();
|
|
||||||
// }
|
|
||||||
|
|
||||||
if (type_instantiation_decl != nullptr) {
|
if (type_instantiation_decl != nullptr) {
|
||||||
for (const auto &template_argument : *type_instantiation_decl) {
|
for (const auto &template_argument : *type_instantiation_decl) {
|
||||||
const auto template_argument_kind = template_argument.getKind();
|
const auto template_argument_kind = template_argument.getKind();
|
||||||
@@ -996,7 +993,12 @@ void translation_unit_visitor::process_function_parameter(
|
|||||||
if (parameter.skip())
|
if (parameter.skip())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
parameter.set_type(common::to_string(p.getType(), p.getASTContext()));
|
auto parameter_type = common::to_string(p.getType(), p.getASTContext());
|
||||||
|
|
||||||
|
// Is there no better way to determine that 'type' is a lambda?
|
||||||
|
ensure_lambda_type_is_relative(parameter_type);
|
||||||
|
|
||||||
|
parameter.set_type(parameter_type);
|
||||||
|
|
||||||
if (p.hasDefaultArg()) {
|
if (p.hasDefaultArg()) {
|
||||||
const auto *default_arg = p.getDefaultArg();
|
const auto *default_arg = p.getDefaultArg();
|
||||||
@@ -1048,6 +1050,30 @@ void translation_unit_visitor::process_function_parameter(
|
|||||||
method.add_parameter(std::move(parameter));
|
method.add_parameter(std::move(parameter));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void translation_unit_visitor::ensure_lambda_type_is_relative(
|
||||||
|
std::string ¶meter_type) const
|
||||||
|
{
|
||||||
|
std::string lambda_prefix{"(lambda at /"};
|
||||||
|
|
||||||
|
while (parameter_type.find(lambda_prefix) != std::string::npos) {
|
||||||
|
auto lambda_begin = parameter_type.find(lambda_prefix);
|
||||||
|
|
||||||
|
auto absolute_lambda_path_end = parameter_type.find(":", lambda_begin);
|
||||||
|
auto absolute_lambda_path =
|
||||||
|
parameter_type.substr(lambda_begin + lambda_prefix.size() - 1,
|
||||||
|
absolute_lambda_path_end -
|
||||||
|
(lambda_begin + lambda_prefix.size() - 1));
|
||||||
|
|
||||||
|
auto relative_lambda_path = std::filesystem::relative(
|
||||||
|
absolute_lambda_path, config().relative_to())
|
||||||
|
.string();
|
||||||
|
|
||||||
|
parameter_type = fmt::format("{}(lambda at {}{}",
|
||||||
|
parameter_type.substr(0, lambda_begin), relative_lambda_path,
|
||||||
|
parameter_type.substr(absolute_lambda_path_end));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void translation_unit_visitor::
|
void translation_unit_visitor::
|
||||||
process_function_parameter_find_relationships_in_template(class_ &c,
|
process_function_parameter_find_relationships_in_template(class_ &c,
|
||||||
const std::set<std::string> & /*template_parameter_names*/,
|
const std::set<std::string> & /*template_parameter_names*/,
|
||||||
|
|||||||
@@ -254,5 +254,6 @@ private:
|
|||||||
std::tuple<std::string /* field name */, common::model::relationship_t,
|
std::tuple<std::string /* field name */, common::model::relationship_t,
|
||||||
common::model::access_t>>
|
common::model::access_t>>
|
||||||
anonymous_struct_relationships_;
|
anonymous_struct_relationships_;
|
||||||
|
void ensure_lambda_type_is_relative(std::string ¶meter_type) const;
|
||||||
};
|
};
|
||||||
} // namespace clanguml::class_diagram::visitor
|
} // namespace clanguml::class_diagram::visitor
|
||||||
|
|||||||
@@ -162,7 +162,8 @@ template <>
|
|||||||
bool starts_with(
|
bool starts_with(
|
||||||
const std::filesystem::path &path, const std::filesystem::path &prefix);
|
const std::filesystem::path &path, const std::filesystem::path &prefix);
|
||||||
|
|
||||||
template <> bool starts_with(const std::string &s, const std::string &prefix);
|
template <>
|
||||||
|
bool starts_with(const std::string &s, const std::string &prefix);
|
||||||
|
|
||||||
template <typename T> bool ends_with(const T &value, const T &suffix);
|
template <typename T> bool ends_with(const T &value, const T &suffix);
|
||||||
|
|
||||||
|
|||||||
12
tests/t00051/.clang-uml
Normal file
12
tests/t00051/.clang-uml
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
compilation_database_dir: ..
|
||||||
|
output_directory: puml
|
||||||
|
diagrams:
|
||||||
|
t00051_class:
|
||||||
|
type: class
|
||||||
|
glob:
|
||||||
|
- ../../tests/t00051/t00051.cc
|
||||||
|
include:
|
||||||
|
namespaces:
|
||||||
|
- clanguml::t00051
|
||||||
|
using_namespace:
|
||||||
|
- clanguml::t00051
|
||||||
63
tests/t00051/t00051.cc
Normal file
63
tests/t00051/t00051.cc
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
#include <thread>
|
||||||
|
|
||||||
|
namespace clanguml {
|
||||||
|
namespace t00051 {
|
||||||
|
|
||||||
|
template <typename F, typename FF = F> struct B : private std::thread {
|
||||||
|
B(F &&f, FF &&ff)
|
||||||
|
: f_{std::move(f)}
|
||||||
|
, ff_{std::move(ff)}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void f() { f_(); }
|
||||||
|
void ff() { ff_(); }
|
||||||
|
|
||||||
|
F f_;
|
||||||
|
FF ff_;
|
||||||
|
};
|
||||||
|
|
||||||
|
class A {
|
||||||
|
public:
|
||||||
|
private:
|
||||||
|
class custom_thread1 : private std::thread {
|
||||||
|
public:
|
||||||
|
template <class Function, class... Args>
|
||||||
|
explicit custom_thread1(Function &&f, Args &&...args)
|
||||||
|
: std::thread::thread(
|
||||||
|
std::forward<Function>(f), std::forward<Args>(args)...)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
static custom_thread1 start_thread1();
|
||||||
|
|
||||||
|
class custom_thread2 : private std::thread {
|
||||||
|
using std::thread::thread;
|
||||||
|
};
|
||||||
|
|
||||||
|
static custom_thread2 start_thread2();
|
||||||
|
|
||||||
|
auto start_thread3()
|
||||||
|
{
|
||||||
|
return B{[]() {}, []() {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
auto get_function()
|
||||||
|
{
|
||||||
|
return []() {};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
A::custom_thread1 A::start_thread1()
|
||||||
|
{
|
||||||
|
return custom_thread1{[]() {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
A::custom_thread2 A::start_thread2()
|
||||||
|
{
|
||||||
|
return custom_thread2{[]() {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
57
tests/t00051/test_case.h
Normal file
57
tests/t00051/test_case.h
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
/**
|
||||||
|
* tests/t00051/test_case.h
|
||||||
|
*
|
||||||
|
* Copyright (c) 2021-2023 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("t00051", "[test-case][class]")
|
||||||
|
{
|
||||||
|
auto [config, db] = load_config("t00051");
|
||||||
|
|
||||||
|
auto diagram = config.diagrams["t00051_class"];
|
||||||
|
|
||||||
|
REQUIRE(diagram->name == "t00051_class");
|
||||||
|
|
||||||
|
auto model = generate_class_diagram(*db, diagram);
|
||||||
|
|
||||||
|
REQUIRE(model->name() == "t00051_class");
|
||||||
|
|
||||||
|
auto puml = generate_class_puml(diagram, *model);
|
||||||
|
AliasMatcher _A(puml);
|
||||||
|
|
||||||
|
REQUIRE_THAT(puml, StartsWith("@startuml"));
|
||||||
|
REQUIRE_THAT(puml, EndsWith("@enduml\n"));
|
||||||
|
|
||||||
|
// Check if all classes exist
|
||||||
|
REQUIRE_THAT(puml, IsClass(_A("A")));
|
||||||
|
REQUIRE_THAT(puml, IsInnerClass(_A("A"), _A("A::custom_thread1")));
|
||||||
|
REQUIRE_THAT(puml, IsInnerClass(_A("A"), _A("A::custom_thread2")));
|
||||||
|
|
||||||
|
REQUIRE_THAT(puml,
|
||||||
|
(IsMethod<Public>(
|
||||||
|
"custom_thread1", "void", "Function && f, Args &&... args")));
|
||||||
|
REQUIRE_THAT(puml,
|
||||||
|
(IsMethod<Public>("thread", "void",
|
||||||
|
"(lambda at ../../tests/t00051/t00051.cc:59:27) && ")));
|
||||||
|
REQUIRE_THAT(puml,
|
||||||
|
(IsMethod<Private>("start_thread3",
|
||||||
|
"B<(lambda at ../../tests/t00051/t00051.cc:43:18),(lambda at "
|
||||||
|
"../../tests/t00051/t00051.cc:43:27)>")));
|
||||||
|
REQUIRE_THAT(puml,
|
||||||
|
(IsMethod<Private>(
|
||||||
|
"get_function", "(lambda at ../../tests/t00051/t00051.cc:48:16)")));
|
||||||
|
|
||||||
|
save_puml(config.output_directory() + "/" + diagram->name + ".puml", puml);
|
||||||
|
}
|
||||||
@@ -244,6 +244,7 @@ using namespace clanguml::test::matchers;
|
|||||||
#include "t00048/test_case.h"
|
#include "t00048/test_case.h"
|
||||||
#include "t00049/test_case.h"
|
#include "t00049/test_case.h"
|
||||||
#include "t00050/test_case.h"
|
#include "t00050/test_case.h"
|
||||||
|
#include "t00051/test_case.h"
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Sequence diagram tests
|
/// Sequence diagram tests
|
||||||
|
|||||||
@@ -147,6 +147,9 @@ test_cases:
|
|||||||
- name: t00050
|
- name: t00050
|
||||||
title: Test case for generating notes from comments using jinja templates
|
title: Test case for generating notes from comments using jinja templates
|
||||||
description:
|
description:
|
||||||
|
- name: t00051
|
||||||
|
title: Test case for relative paths in lambda names
|
||||||
|
description:
|
||||||
Sequence diagrams:
|
Sequence diagrams:
|
||||||
- name: t20001
|
- name: t20001
|
||||||
title: Basic sequence diagram test case
|
title: Basic sequence diagram test case
|
||||||
|
|||||||
@@ -36,6 +36,5 @@ TEST_CASE("{{ name }}", "[test-case][{{ type }}]")
|
|||||||
|
|
||||||
{{ examples }}
|
{{ examples }}
|
||||||
|
|
||||||
save_puml(
|
save_puml(config.output_directory() + "/" + diagram->name + ".puml", puml);
|
||||||
"./" + config.output_directory() + "/" + diagram->name + ".puml", puml);
|
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user