Added support for 'together' option in class diagrams with rendered namespaces
This commit is contained in:
@@ -26,6 +26,7 @@ namespace clanguml::class_diagram::generators::plantuml {
|
|||||||
|
|
||||||
generator::generator(diagram_config &config, diagram_model &model)
|
generator::generator(diagram_config &config, diagram_model &model)
|
||||||
: common_generator<diagram_config, diagram_model>{config, model}
|
: common_generator<diagram_config, diagram_model>{config, model}
|
||||||
|
, together_group_stack_{!config.generate_packages()}
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -288,6 +289,25 @@ void generator::generate_member_notes(std::ostream &ostr,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void generator::generate_relationships(std::ostream &ostr) const
|
||||||
|
{
|
||||||
|
for (const auto &p : m_model) {
|
||||||
|
if (auto *pkg = dynamic_cast<package *>(p.get()); pkg) {
|
||||||
|
generate_relationships(*pkg, ostr);
|
||||||
|
}
|
||||||
|
else if (auto *cls = dynamic_cast<class_ *>(p.get()); cls) {
|
||||||
|
if (m_model.should_include(*cls)) {
|
||||||
|
generate_relationships(*cls, ostr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (auto *enm = dynamic_cast<enum_ *>(p.get()); enm) {
|
||||||
|
if (m_model.should_include(*enm)) {
|
||||||
|
generate_relationships(*enm, ostr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void generator::generate_relationships(
|
void generator::generate_relationships(
|
||||||
const class_ &c, std::ostream &ostr) const
|
const class_ &c, std::ostream &ostr) const
|
||||||
{
|
{
|
||||||
@@ -474,40 +494,65 @@ void generator::generate(const package &p, std::ostream &ostr) const
|
|||||||
if (dynamic_cast<package *>(subpackage.get()) != nullptr) {
|
if (dynamic_cast<package *>(subpackage.get()) != nullptr) {
|
||||||
// TODO: add option - generate_empty_packages
|
// TODO: add option - generate_empty_packages
|
||||||
const auto &sp = dynamic_cast<package &>(*subpackage);
|
const auto &sp = dynamic_cast<package &>(*subpackage);
|
||||||
if (!sp.is_empty())
|
if (!sp.is_empty()) {
|
||||||
|
together_group_stack_.enter();
|
||||||
|
|
||||||
generate(sp, ostr);
|
generate(sp, ostr);
|
||||||
|
|
||||||
|
together_group_stack_.leave();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (dynamic_cast<class_ *>(subpackage.get()) != nullptr) {
|
else if (auto *cls = dynamic_cast<class_ *>(subpackage.get()); cls) {
|
||||||
if (m_model.should_include(*subpackage)) {
|
if (m_model.should_include(*subpackage)) {
|
||||||
auto together_group =
|
auto together_group =
|
||||||
m_config.get_together_group(subpackage->full_name(false));
|
m_config.get_together_group(cls->full_name(false));
|
||||||
if (together_group) {
|
if (together_group) {
|
||||||
current_level_together_groups_[together_group.value()]
|
together_group_stack_.group_together(
|
||||||
.push_back(subpackage.get());
|
together_group.value(), cls);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
generate_alias(dynamic_cast<class_ &>(*subpackage), ostr);
|
generate_alias(*cls, ostr);
|
||||||
generate(dynamic_cast<class_ &>(*subpackage), ostr);
|
generate(*cls, ostr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (dynamic_cast<enum_ *>(subpackage.get()) != nullptr) {
|
else if (auto *enm = dynamic_cast<enum_ *>(subpackage.get()); enm) {
|
||||||
if (m_model.should_include(*subpackage)) {
|
if (m_model.should_include(*subpackage)) {
|
||||||
auto together_group =
|
auto together_group =
|
||||||
m_config.get_together_group(subpackage->full_name(false));
|
m_config.get_together_group(subpackage->full_name(false));
|
||||||
if (together_group) {
|
if (together_group) {
|
||||||
current_level_together_groups_[together_group.value()]
|
together_group_stack_.group_together(
|
||||||
.push_back(subpackage.get());
|
together_group.value(), enm);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
generate_alias(dynamic_cast<enum_ &>(*subpackage), ostr);
|
generate_alias(*enm, ostr);
|
||||||
generate(dynamic_cast<enum_ &>(*subpackage), ostr);
|
generate(*enm, ostr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_config.generate_packages()) {
|
if (m_config.generate_packages()) {
|
||||||
|
// Now generate any diagram elements which are in together
|
||||||
|
// groups
|
||||||
|
for (const auto &[group_name, group_elements] :
|
||||||
|
together_group_stack_.get_current_groups()) {
|
||||||
|
ostr << "together {\n";
|
||||||
|
|
||||||
|
for (auto *e : group_elements) {
|
||||||
|
if (auto *cls = dynamic_cast<class_ *>(e); cls) {
|
||||||
|
generate_alias(*cls, ostr);
|
||||||
|
generate(*cls, ostr);
|
||||||
|
}
|
||||||
|
if (auto *enm = dynamic_cast<enum_ *>(e); enm) {
|
||||||
|
generate_alias(*enm, ostr);
|
||||||
|
generate(*enm, ostr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ostr << "}\n";
|
||||||
|
}
|
||||||
|
|
||||||
// Don't generate packages from namespaces filtered out by
|
// Don't generate packages from namespaces filtered out by
|
||||||
// using_namespace
|
// using_namespace
|
||||||
if (!uns.starts_with({p.full_name(false)})) {
|
if (!uns.starts_with({p.full_name(false)})) {
|
||||||
@@ -551,6 +596,21 @@ void generator::generate(std::ostream &ostr) const
|
|||||||
|
|
||||||
generate_plantuml_directives(ostr, m_config.puml().before);
|
generate_plantuml_directives(ostr, m_config.puml().before);
|
||||||
|
|
||||||
|
generate_top_level_elements(ostr);
|
||||||
|
|
||||||
|
generate_groups(ostr);
|
||||||
|
|
||||||
|
generate_relationships(ostr);
|
||||||
|
|
||||||
|
generate_config_layout_hints(ostr);
|
||||||
|
|
||||||
|
generate_plantuml_directives(ostr, m_config.puml().after);
|
||||||
|
|
||||||
|
ostr << "@enduml" << '\n';
|
||||||
|
}
|
||||||
|
|
||||||
|
void generator::generate_top_level_elements(std::ostream &ostr) const
|
||||||
|
{
|
||||||
for (const auto &p : m_model) {
|
for (const auto &p : m_model) {
|
||||||
if (auto *pkg = dynamic_cast<package *>(p.get()); pkg) {
|
if (auto *pkg = dynamic_cast<package *>(p.get()); pkg) {
|
||||||
if (!pkg->is_empty())
|
if (!pkg->is_empty())
|
||||||
@@ -561,8 +621,8 @@ void generator::generate(std::ostream &ostr) const
|
|||||||
auto together_group =
|
auto together_group =
|
||||||
m_config.get_together_group(cls->full_name(false));
|
m_config.get_together_group(cls->full_name(false));
|
||||||
if (together_group) {
|
if (together_group) {
|
||||||
current_level_together_groups_[together_group.value()]
|
together_group_stack_.group_together(
|
||||||
.push_back(cls);
|
together_group.value(), cls);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
generate_alias(*cls, ostr);
|
generate_alias(*cls, ostr);
|
||||||
@@ -575,8 +635,8 @@ void generator::generate(std::ostream &ostr) const
|
|||||||
auto together_group =
|
auto together_group =
|
||||||
m_config.get_together_group(enm->full_name(false));
|
m_config.get_together_group(enm->full_name(false));
|
||||||
if (together_group) {
|
if (together_group) {
|
||||||
current_level_together_groups_[together_group.value()]
|
together_group_stack_.group_together(
|
||||||
.push_back(enm);
|
together_group.value(), enm);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
generate_alias(*enm, ostr);
|
generate_alias(*enm, ostr);
|
||||||
@@ -585,13 +645,14 @@ void generator::generate(std::ostream &ostr) const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Now generate any diagram elements which are in together groups
|
void generator::generate_groups(std::ostream &ostr) const
|
||||||
|
{
|
||||||
for (const auto &[group_name, group_elements] :
|
for (const auto &[group_name, group_elements] :
|
||||||
current_level_together_groups_) {
|
together_group_stack_.get_current_groups()) {
|
||||||
|
|
||||||
ostr << "' together group for " << group_name << "\n";
|
|
||||||
ostr << "together {\n";
|
ostr << "together {\n";
|
||||||
|
|
||||||
for (auto *e : group_elements) {
|
for (auto *e : group_elements) {
|
||||||
if (auto *cls = dynamic_cast<class_ *>(e); cls) {
|
if (auto *cls = dynamic_cast<class_ *>(e); cls) {
|
||||||
generate_alias(*cls, ostr);
|
generate_alias(*cls, ostr);
|
||||||
@@ -602,30 +663,9 @@ void generator::generate(std::ostream &ostr) const
|
|||||||
generate(*enm, ostr);
|
generate(*enm, ostr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ostr << "}\n";
|
ostr << "}\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const auto &p : m_model) {
|
|
||||||
if (auto *pkg = dynamic_cast<package *>(p.get()); pkg) {
|
|
||||||
generate_relationships(*pkg, ostr);
|
|
||||||
}
|
|
||||||
else if (auto *cls = dynamic_cast<class_ *>(p.get()); cls) {
|
|
||||||
if (m_model.should_include(*cls)) {
|
|
||||||
generate_relationships(*cls, ostr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (auto *enm = dynamic_cast<enum_ *>(p.get()); enm) {
|
|
||||||
if (m_model.should_include(*enm)) {
|
|
||||||
generate_relationships(*enm, ostr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
generate_config_layout_hints(ostr);
|
|
||||||
|
|
||||||
generate_plantuml_directives(ostr, m_config.puml().after);
|
|
||||||
|
|
||||||
ostr << "@enduml" << '\n';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace clanguml::class_diagram::generators::plantuml
|
} // namespace clanguml::class_diagram::generators::plantuml
|
||||||
|
|||||||
@@ -21,6 +21,7 @@
|
|||||||
#include "class_diagram/model/diagram.h"
|
#include "class_diagram/model/diagram.h"
|
||||||
#include "class_diagram/model/enum.h"
|
#include "class_diagram/model/enum.h"
|
||||||
#include "class_diagram/visitor/translation_unit_visitor.h"
|
#include "class_diagram/visitor/translation_unit_visitor.h"
|
||||||
|
#include "common/generators/nested_element_stack.h"
|
||||||
#include "common/generators/plantuml/generator.h"
|
#include "common/generators/plantuml/generator.h"
|
||||||
#include "common/model/relationship.h"
|
#include "common/model/relationship.h"
|
||||||
#include "config/config.h"
|
#include "config/config.h"
|
||||||
@@ -66,6 +67,10 @@ public:
|
|||||||
|
|
||||||
void generate(const class_ &c, std::ostream &ostr) const;
|
void generate(const class_ &c, std::ostream &ostr) const;
|
||||||
|
|
||||||
|
void generate_top_level_elements(std::ostream &ostr) const;
|
||||||
|
|
||||||
|
void generate_relationships(std::ostream &ostr) const;
|
||||||
|
|
||||||
void generate_relationships(const class_ &c, std::ostream &ostr) const;
|
void generate_relationships(const class_ &c, std::ostream &ostr) const;
|
||||||
|
|
||||||
void generate(const enum_ &e, std::ostream &ostr) const;
|
void generate(const enum_ &e, std::ostream &ostr) const;
|
||||||
@@ -79,13 +84,15 @@ public:
|
|||||||
void generate_member_notes(std::ostream &ostream,
|
void generate_member_notes(std::ostream &ostream,
|
||||||
const class_element &member, const std::string &basicString) const;
|
const class_element &member, const std::string &basicString) const;
|
||||||
|
|
||||||
|
void generate_groups(std::ostream &ostr) const;
|
||||||
|
|
||||||
void generate(std::ostream &ostr) const override;
|
void generate(std::ostream &ostr) const override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::string render_name(std::string name) const;
|
std::string render_name(std::string name) const;
|
||||||
|
|
||||||
mutable std::map<std::string, std::vector<common::model::element *>>
|
mutable common::generators::nested_element_stack<common::model::element>
|
||||||
current_level_together_groups_;
|
together_group_stack_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace plantuml
|
} // namespace plantuml
|
||||||
|
|||||||
74
src/common/generators/nested_element_stack.h
Normal file
74
src/common/generators/nested_element_stack.h
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
/**
|
||||||
|
* src/common/generators/nested_element_stack.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.
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <map>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace clanguml::common::generators {
|
||||||
|
|
||||||
|
template <typename T> class nested_element_stack {
|
||||||
|
public:
|
||||||
|
nested_element_stack(bool is_flat)
|
||||||
|
: is_flat_{is_flat}
|
||||||
|
, current_level_{0}
|
||||||
|
{
|
||||||
|
current_level_groups_.push_back({});
|
||||||
|
}
|
||||||
|
|
||||||
|
void enter()
|
||||||
|
{
|
||||||
|
if (!is_flat_)
|
||||||
|
current_level_++;
|
||||||
|
|
||||||
|
current_level_groups_.push_back({});
|
||||||
|
}
|
||||||
|
|
||||||
|
void leave()
|
||||||
|
{
|
||||||
|
if (!is_flat_)
|
||||||
|
current_level_--;
|
||||||
|
|
||||||
|
current_level_groups_.pop_back();
|
||||||
|
}
|
||||||
|
|
||||||
|
void group_together(const std::string &group_name, T *e)
|
||||||
|
{
|
||||||
|
current_level_groups_[current_level_][group_name].push_back(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::map<std::string, std::vector<T *>> &get_current_groups()
|
||||||
|
{
|
||||||
|
return current_level_groups_.at(current_level_);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<T *> &get_group(const std::string &group_name)
|
||||||
|
{
|
||||||
|
return get_current_groups().at(group_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool is_flat_;
|
||||||
|
|
||||||
|
uint32_t current_level_;
|
||||||
|
|
||||||
|
std::vector<std::map<std::string, std::vector<T *>>> current_level_groups_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace clanguml::common::generators
|
||||||
24
tests/t00054/.clang-uml
Normal file
24
tests/t00054/.clang-uml
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
compilation_database_dir: ..
|
||||||
|
output_directory: puml
|
||||||
|
diagrams:
|
||||||
|
t00054_class:
|
||||||
|
type: class
|
||||||
|
glob:
|
||||||
|
- ../../tests/t00054/t00054.cc
|
||||||
|
include:
|
||||||
|
namespaces:
|
||||||
|
- clanguml::t00054
|
||||||
|
using_namespace:
|
||||||
|
- clanguml::t00054
|
||||||
|
generate_packages: true
|
||||||
|
layout:
|
||||||
|
a:
|
||||||
|
- together: [f]
|
||||||
|
"detail::c":
|
||||||
|
- together: [detail::e]
|
||||||
|
A:
|
||||||
|
- together: [B,G]
|
||||||
|
"detail2::detail3::D":
|
||||||
|
- together: [detail2::detail3::E]
|
||||||
|
"detail4::h":
|
||||||
|
- together: [detail4::i,detail4::j]
|
||||||
48
tests/t00054/t00054.cc
Normal file
48
tests/t00054/t00054.cc
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
namespace clanguml {
|
||||||
|
namespace t00054 {
|
||||||
|
struct a {
|
||||||
|
};
|
||||||
|
struct b {
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
struct c {
|
||||||
|
};
|
||||||
|
struct d {
|
||||||
|
};
|
||||||
|
struct e {
|
||||||
|
};
|
||||||
|
} // namespace detail
|
||||||
|
struct f {
|
||||||
|
};
|
||||||
|
struct g {
|
||||||
|
};
|
||||||
|
|
||||||
|
struct A {
|
||||||
|
};
|
||||||
|
struct B {
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace detail2 {
|
||||||
|
struct C {
|
||||||
|
};
|
||||||
|
namespace detail3 {
|
||||||
|
struct D {
|
||||||
|
};
|
||||||
|
struct E {
|
||||||
|
};
|
||||||
|
} // namespace detail3
|
||||||
|
struct F {
|
||||||
|
};
|
||||||
|
} // namespace detail2
|
||||||
|
struct G {
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace detail4 {
|
||||||
|
enum class h { hhh };
|
||||||
|
enum class i { iii };
|
||||||
|
enum class j { jjj };
|
||||||
|
} // namespace detail4
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
59
tests/t00054/test_case.h
Normal file
59
tests/t00054/test_case.h
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
/**
|
||||||
|
* tests/t00054/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("t00054", "[test-case][class]")
|
||||||
|
{
|
||||||
|
auto [config, db] = load_config("t00054");
|
||||||
|
|
||||||
|
auto diagram = config.diagrams["t00054_class"];
|
||||||
|
|
||||||
|
REQUIRE(diagram->name == "t00054_class");
|
||||||
|
|
||||||
|
auto model = generate_class_diagram(*db, diagram);
|
||||||
|
|
||||||
|
REQUIRE(model->name() == "t00054_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, 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("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, IsEnum(_A("i")));
|
||||||
|
REQUIRE_THAT(puml, IsEnum(_A("h")));
|
||||||
|
REQUIRE_THAT(puml, IsEnum(_A("j")));
|
||||||
|
|
||||||
|
save_puml(config.output_directory() + "/" + diagram->name + ".puml", puml);
|
||||||
|
}
|
||||||
@@ -247,6 +247,7 @@ using namespace clanguml::test::matchers;
|
|||||||
#include "t00051/test_case.h"
|
#include "t00051/test_case.h"
|
||||||
#include "t00052/test_case.h"
|
#include "t00052/test_case.h"
|
||||||
#include "t00053/test_case.h"
|
#include "t00053/test_case.h"
|
||||||
|
#include "t00054/test_case.h"
|
||||||
|
|
||||||
///
|
///
|
||||||
/// Sequence diagram tests
|
/// Sequence diagram tests
|
||||||
|
|||||||
@@ -154,7 +154,10 @@ test_cases:
|
|||||||
title: Test case for template methods rendering
|
title: Test case for template methods rendering
|
||||||
description:
|
description:
|
||||||
- name: t00053
|
- name: t00053
|
||||||
title: Test case for `together` layout hint
|
title: Test case for `together` layout hint in class diagram
|
||||||
|
description:
|
||||||
|
- name: t00054
|
||||||
|
title: Test case for `together` layout hint in class diagram with rendered namespaces
|
||||||
description:
|
description:
|
||||||
Sequence diagrams:
|
Sequence diagrams:
|
||||||
- name: t20001
|
- name: t20001
|
||||||
|
|||||||
Reference in New Issue
Block a user