Beim Erstellen von Software definieren Entwicklungsteams in der Regel eine Reihe von Richtlinien und Konventionen für die Codierung, die als Best Practices gelten.
Dies sind Methoden, die normalerweise dokumentiert und dem gesamten Entwicklungsteam mitgeteilt werden, das sie übernommen hat. Während der Entwicklung können Entwickler jedoch gegen diese Richtlinien verstoßen, die bei Codeüberprüfungen oder durch Codequalitätsprüfungen ermittelt werden.
Ein wichtiger Aspekt ist daher, diese Anweisungen in der gesamten Projektarchitektur so weit wie möglich zu automatisieren, um die Überprüfungen zu optimieren.
Wir können diese Richtlinien als überprüfbare JUnit-Tests mit ArchUnit implementieren . Dadurch wird sichergestellt, dass der Build der Softwareversion im Falle eines Architekturverstoßes gestoppt wird.

ArchUnit ist eine kostenlose, einfache und erweiterbare Bibliothek zum Testen der Architektur Ihres Java-Codes zur Verwendung in jeder einfachen Java-Unit-Testumgebung. Das heißt, ArchUnit kann Abhängigkeiten zwischen Paketen und Klassen, Ebenen und Slices überprüfen, zirkuläre Abhängigkeiten überprüfen und vieles mehr. Dazu wird ein bestimmter Java-Bytecode analysiert und alle Klassen in die Java-Codestruktur importiert.
ArchUnit , :
ArchUnit JUnit 5, Maven Central:
pom.xml
XML
<dependency>
<groupId>com.tngtech.archunit</groupId>
<artifactId>archunit-junit5</artifactId>
<version>0.14.1</version>
<scope>test</scope>
</dependency>build.gradle
Groovy
dependencies {
testImplementation 'com.tngtech.archunit:archunit-junit5:0.14.1'
} }
Java
class ArchunitApplicationTests {
private JavaClasses importedClasses;
@BeforeEach
public void setup() {
importedClasses = new ClassFileImporter()
.withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS)
.importPackages("com.springboot.testing.archunit");
}
@Test
void servicesAndRepositoriesShouldNotDependOnWebLayer() {
noClasses()
.that().resideInAnyPackage("com.springboot.testing.archunit.service..")
.or().resideInAnyPackage("com.springboot.testing.archunit.repository..")
.should()
.dependOnClassesThat()
.resideInAnyPackage("com.springboot.testing.archunit.controller..")
.because("Services and repositories should not depend on web layer")
.check(importedClasses);
}
}-.
class ArchunitApplicationTests {
private JavaClasses importedClasses;
@BeforeEach
public void setup() {
importedClasses = new ClassFileImporter()
.withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS)
.importPackages("com.springboot.testing.archunit");
}
@Test
void serviceClassesShouldOnlyBeAccessedByController() {
classes()
.that().resideInAPackage("..service..")
.should().onlyBeAccessed().byAnyPackage("..service..", "..controller..")
.check(importedClasses);
}
}ArchUnit API-, DSL, , , . .
( AspectJ Pointcuts).
Java
class ArchunitApplicationTests {
private JavaClasses importedClasses;
@BeforeEach
public void setup() {
importedClasses = new ClassFileImporter()
importedClasses = new ClassFileImporter()
.withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS)
.importPackages("com.springboot.testing.archunit");
}
@Test
void serviceClassesShouldBeNamedXServiceOrXComponentOrXServiceImpl() {
classes()
.that().resideInAPackage("..service..")
.should().haveSimpleNameEndingWith("Service")
.orShould().haveSimpleNameEndingWith("ServiceImpl")
.orShould().haveSimpleNameEndingWith("Component")
.check(importedClasses);
}
@Test
void repositoryClassesShouldBeNamedXRepository() {
classes()
.that().resideInAPackage("..repository..")
.should().haveSimpleNameEndingWith("Repository")
.check(importedClasses);
}
@Test
void controllerClassesShouldBeNamedXController() {
classes()
.that().resideInAPackage("..controller..")
.should().haveSimpleNameEndingWith("Controller")
.check(importedClasses);
}
}
— . , Service, Component . .
Java
class ArchunitApplicationTests {
private JavaClasses importedClasses;
@BeforeEach
public void setup() {
importedClasses = new ClassFileImporter()
.withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS)
.importPackages("com.springboot.testing.archunit");
}
@Test
void fieldInjectionNotUseAutowiredAnnotation() {
noFields()
.should().beAnnotatedWith(Autowired.class)
.check(importedClasses);
}
@Test
void repositoryClassesShouldHaveSpringRepositoryAnnotation() {
classes()
.that().resideInAPackage("..repository..")
.should().beAnnotatedWith(Repository.class)
.check(importedClasses);
}
@Test
void serviceClassesShouldHaveSpringServiceAnnotation() {
classes()
.that().resideInAPackage("..service..")
.should().beAnnotatedWith(Service.class)
.check(importedClasses);
}
}API ArchUnit Lang Java. , , .
class ArchunitApplicationTests {
private JavaClasses importedClasses;
@BeforeEach
public void setup() {
importedClasses = new ClassFileImporter()
.withImportOption(ImportOption.Predefined.DO_NOT_INCLUDE_TESTS)
.importPackages("com.springboot.testing.archunit");
}
@Test
void layeredArchitectureShouldBeRespected() {
layeredArchitecture()
.layer("Controller").definedBy("..controller..")
.layer("Service").definedBy("..service..")
.layer("Repository").definedBy("..repository..")
.whereLayer("Controller").mayNotBeAccessedByAnyLayer()
.whereLayer("Service").mayOnlyBeAccessedByLayers("Controller")
.whereLayer("Repository").mayOnlyBeAccessedByLayers("Service")
.check(importedClasses);
}
}Spring Boot , .
ArchUnit bietet eine Reihe von Funktionen, mit denen Sie überprüfen können, ob Ihre Ebenenarchitektur eingehalten wird. Diese Tests stellen automatisch sicher, dass Zugriff und Verwendung innerhalb der von Ihnen festgelegten Grenzen gehalten werden. Daher können Sie Ihre eigenen Regeln schreiben. In diesem Artikel haben wir verschiedene Regeln beschrieben. Die offizielle Dokumentation von ArchUnit bietet viele weitere Möglichkeiten.
Den vollständigen Quellcode der Beispiele finden Sie in meinem GitHub-Repository .