All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.tngtech.archunit.core.domain.JavaClasses Maven / Gradle / Ivy

Go to download

A Java architecture test library, to specify and assert architecture rules in plain Java - Module 'archunit'

There is a newer version: 1.3.0
Show newest version
/*
 * Copyright 2014-2024 TNG Technology Consulting GmbH
 *
 * 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.
 */
package com.tngtech.archunit.core.domain;

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;

import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Iterables;
import com.tngtech.archunit.PublicAPI;
import com.tngtech.archunit.base.DescribedIterable;
import com.tngtech.archunit.base.DescribedPredicate;
import com.tngtech.archunit.base.ForwardingCollection;
import com.tngtech.archunit.core.domain.properties.CanOverrideDescription;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.tngtech.archunit.PublicAPI.Usage.ACCESS;
import static java.util.stream.Collectors.toMap;

@PublicAPI(usage = ACCESS)
public final class JavaClasses extends ForwardingCollection implements DescribedIterable, CanOverrideDescription {
    private final ImmutableMap classes;
    private final JavaPackage defaultPackage;
    private final String description;

    private JavaClasses(JavaPackage defaultPackage, Map classes) {
        this(defaultPackage, classes, "classes");
    }

    private JavaClasses(JavaPackage defaultPackage, Map classes, String description) {
        this.classes = ImmutableMap.copyOf(classes);
        this.defaultPackage = checkNotNull(defaultPackage);
        this.description = checkNotNull(description);
    }

    /**
     * @param predicate a {@link DescribedPredicate} to determine which classes match
     * @return {@link JavaClasses} matching the given predicate; the description will be adjusted according to the predicate's description
     */
    @PublicAPI(usage = ACCESS)
    public JavaClasses that(DescribedPredicate predicate) {
        Map matchingElements = classes.entrySet().stream()
                .filter(e -> predicate.test(e.getValue()))
                .collect(toMap(Map.Entry::getKey, Map.Entry::getValue));
        String newDescription = String.format("%s that %s", description, predicate.getDescription());
        return new JavaClasses(defaultPackage, matchingElements, newDescription);
    }

    @Override
    public JavaClasses as(String description) {
        return new JavaClasses(defaultPackage, classes, description);
    }

    @Override
    public String getDescription() {
        return description;
    }

    @Override
    public String toString() {
        return getClass().getSimpleName() + "{classes=" + classes + '}';
    }

    /**
     * @param reflectedType a Java {@link Class} object
     * @return true, if an equivalent {@link JavaClass} is contained, false otherwise
     * @see #get(Class)
     * @see #contain(String)
     */
    @PublicAPI(usage = ACCESS)
    public boolean contain(Class reflectedType) {
        return contain(reflectedType.getName());
    }

    /**
     * @param reflectedType a Java {@link Class} object
     * @return a {@link JavaClass} equivalent to the given type; throws an exception if there is no equivalent class
     * @see #contain(Class)
     * @see #get(String)
     */
    @PublicAPI(usage = ACCESS)
    public JavaClass get(Class reflectedType) {
        return get(reflectedType.getName());
    }

    /**
     * @param typeName a fully qualified name of a Java class
     * @return true, if a {@link JavaClass} with the given name is contained, false otherwise
     * @see #get(String)
     * @see #contain(Class)
     */
    @PublicAPI(usage = ACCESS)
    public boolean contain(String typeName) {
        return classes.containsKey(typeName);
    }

    /**
     * @param typeName a fully qualified name of a Java class
     * @return a {@link JavaClass} with the given name; throws an exception if there is no class with the given name
     * @see #contain(Class)
     * @see #get(Class)
     */
    @PublicAPI(usage = ACCESS)
    public JavaClass get(String typeName) {
        checkArgument(contain(typeName), "%s do not contain %s of type %s",
                getClass().getSimpleName(), JavaClass.class.getSimpleName(), typeName);

        return classes.get(typeName);
    }

    /**
     * @param packageName name of a package, may consist of several parts, e.g. {@code com.myapp.some.subpackage}
     * @return true, if some package with this name is contained, false otherwise
     */
    @PublicAPI(usage = ACCESS)
    public boolean containPackage(String packageName) {
        return defaultPackage.containsPackage(packageName);
    }

    /**
     * @param packageName name of a package, may consist of several parts, e.g. {@code com.myapp.some.subpackage}
     * @return the package with the given name; throws an exception if the package does not exist
     * @see #containPackage(String)
     */
    @PublicAPI(usage = ACCESS)
    public JavaPackage getPackage(String packageName) {
        return defaultPackage.getPackage(packageName);
    }

    /**
     * @return the default package, i.e. the root of all packages. The default package name ist the empty string.
     */
    @PublicAPI(usage = ACCESS)
    public JavaPackage getDefaultPackage() {
        return defaultPackage;
    }

    @Override
    public int hashCode() {
        return Objects.hash(classes.keySet(), description);
    }

    @Override
    public boolean equals(Object obj) {
        if (this == obj) {
            return true;
        }
        if (obj == null || getClass() != obj.getClass()) {
            return false;
        }
        JavaClasses other = (JavaClasses) obj;
        return Objects.equals(this.classes.keySet(), other.classes.keySet())
                && Objects.equals(this.description, other.description);
    }

    @Override
    protected Collection delegate() {
        return classes.values();
    }

    static JavaClasses of(Iterable classes) {
        Map mapping = new HashMap<>();
        for (JavaClass clazz : classes) {
            mapping.put(clazz.getName(), clazz);
        }
        JavaPackage defaultPackage = !Iterables.isEmpty(classes)
                ? getRoot(classes.iterator().next().getPackage())
                : JavaPackage.from(classes);
        return new JavaClasses(defaultPackage, mapping);
    }

    private static JavaPackage getRoot(JavaPackage javaPackage) {
        JavaPackage result = javaPackage;
        while (result.getParent().isPresent()) {
            result = result.getParent().get();
        }
        return result;
    }

    static JavaClasses of(
            Map selectedClasses, Collection allClasses, ImportContext importContext) {

        ReverseDependencies.Creation reverseDependenciesCreation = new ReverseDependencies.Creation();
        JavaPackage defaultPackage = JavaPackage.from(allClasses);
        for (JavaClass clazz : allClasses) {
            setPackage(clazz, defaultPackage);
            JavaClassDependencies classDependencies = clazz.completeFrom(importContext);
            reverseDependenciesCreation.registerDependenciesOf(clazz, classDependencies);
        }
        reverseDependenciesCreation.finish(allClasses);
        return new JavaClasses(defaultPackage, selectedClasses);
    }

    private static void setPackage(JavaClass clazz, JavaPackage defaultPackage) {
        JavaPackage javaPackage = clazz.getPackageName().isEmpty()
                ? defaultPackage
                : defaultPackage.getPackage(clazz.getPackageName());
        clazz.setPackage(javaPackage);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy