0

我有两个包,一个用于接口,一个用于实现。我想将类的访问限制在它们实现的接口上。说:

package com.interfaces
interface AI {
}

package com.interfaces
interface BI {
}

package com.impls
class A implements AI{
}

package com.impls
class B implements BI{
}

我已经有一个规则来确保 com.impls 中的每个类都实现来自 com.interfaces 的接口:

ArchRuleDefinition.classes()
        .that().resideInAPackage("com.impls..")
.should().implement(JavaClass.Predicates.resideInAPackage("com.interfaces.."))

现在我需要确保一个类只访问它实现的接口。如果有帮助,我也可以依赖上面示例中的命名约定。我正在寻找类似的东西:

ArchRuleDefinition.classes()
        .that().resideInAPackage("com.impls..")
.should().implement(JavaClass.Predicates.resideInAPackage("com.interfaces.."))
.andShould().onlyHaveDependentClassesThat().haveSimpleNameContaining("THE NAME OF THE IMPLEMENTING CLASS")

我们将在任何地方使用接口,并注入实现(Spring Boot)。但是不允许实现访问它没有实现的接口。所以允许 AI 访问 A 但不允许访问 B。允许 BI 访问 B 但不允许访问 A!另一方面,接口不限于实现。它们将用于其他包中。

有没有办法重用规则第一部分的名称或在 ArchUnit 中以另一种方式限制访问?

4

1 回答 1

1

如果您想防止除A自身以外的任何类依赖A等,您可以使用

ArchRuleDefinition.classes()
  .that().resideInAPackage("com.impls..")
  .should().implement(JavaClass.Predicates.resideInAPackage("com.interfaces.."))
  .andShould(new ArchCondition<JavaClass>("not have other classes depending on them") {
    @Override
    public void check(JavaClass javaClass, ConditionEvents events) {
      List<JavaClass> otherClassesThatDependOnJavaClass = javaClass.getDirectDependenciesToSelf().stream()
              .map(Dependency::getOriginClass)
              .filter(origin -> !origin.equals(javaClass))  // TODO: inner classes?
              .collect(toList());
      boolean satisfied = otherClassesThatDependOnJavaClass.isEmpty();
      String message = satisfied
              ? "no other classes depend on " + javaClass.getName()
              : otherClassesThatDependOnJavaClass.stream().map(JavaClass::getName)
                .collect(joining(", ", "other classes depend on " + javaClass.getName() + ": ", ""));
      events.add(new SimpleConditionEvent(javaClass, satisfied, message));
    }
  });

如果这对您来说是一个用例,您可能希望在其中调整过滤器(请参阅 参考资料TODO)以考虑内部类。

.filter(origin -> !origin.getSimpleName().contains(javaClass.getSimpleName()))您要求的很容易实现,但可能过于宽容。

于 2021-07-14T19:38:19.913 回答