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

io.github.amayaframework.di.AbstractProviderBuilder Maven / Gradle / Ivy

package io.github.amayaframework.di;

import com.github.romanqed.jfunc.Function0;
import com.github.romanqed.jfunc.Function1;
import com.github.romanqed.jfunc.LazyFunction0;

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

/**
 * A class that provides a skeletal implementation of the {@link ServiceProviderBuilder},
 * containing implementations of all methods, structures for subsequent analysis,
 * and a reset mechanism to the initial state.
 */
public abstract class AbstractProviderBuilder implements ServiceProviderBuilder {

    /**
     * The map contains "strong" services, that is, they definitely do not have dependencies.
     */
    protected Map> strong;

    /**
     * A map containing "weak" services, that is, having dependencies that need to be resolved.
     */
    protected Map any;

    /**
     * The {@link Repository} instance used, may be null
     */
    protected Repository repository;

    /**
     * Constructs {@link AbstractProviderBuilder}, reset to the initial state.
     * Does not contain any services, the repository used is set to null.
     */
    protected AbstractProviderBuilder() {
        this.reset();
    }

    /**
     * Resets the contents of this {@link AbstractProviderBuilder} instance.
     */
    protected void reset() {
        this.strong = new HashMap<>();
        this.any = new HashMap<>();
        this.repository = null;
    }

    /**
     * Determines whether an artifact implementation exists.
     *
     * @param artifact the specified artifact
     * @return true, if exists, false otherwise
     */
    protected boolean resolve(Artifact artifact) {
        return strong.containsKey(artifact) || any.containsKey(artifact);
    }

    @Override
    public ServiceProviderBuilder setRepository(Repository repository) {
        this.repository = repository;
        return this;
    }

    @Override
    public  ServiceProviderBuilder addService(Artifact artifact,
                                                 Class implementation,
                                                 Function1, Function0> wrapper) {
        // Non-null checks
        Objects.requireNonNull(artifact);
        Objects.requireNonNull(implementation);
        Objects.requireNonNull(wrapper);
        // Check if the implementation is a child class of an artifact type
        var parent = artifact.getType();
        if (!parent.isAssignableFrom(implementation)) {
            throw new IllegalArgumentException("The implementation is not a child class of the artifact type");
        }
        strong.remove(artifact);
        any.put(artifact, Entry.of(implementation, wrapper));
        return this;
    }

    @Override
    public ServiceProviderBuilder addSingleton(Artifact artifact, Class implementation) {
        return addService(artifact, implementation, LazyFunction0::new);
    }

    @Override
    public ServiceProviderBuilder addTransient(Artifact artifact, Class implementation) {
        return addService(artifact, implementation, Function1.identity());
    }

    @Override
    public  ServiceProviderBuilder addService(Class type,
                                                 Class implementation,
                                                 Function1, Function0> wrapper) {
        return addService(new Artifact(type), implementation, wrapper);
    }

    @Override
    public  ServiceProviderBuilder addSingleton(Class type, Class implementation) {
        return addService(type, implementation, LazyFunction0::new);
    }

    @Override
    public  ServiceProviderBuilder addTransient(Class type, Class implementation) {
        return addService(type, implementation, Function1.identity());
    }

    @Override
    public  ServiceProviderBuilder addService(Class type, Function1, Function0> wrapper) {
        return addService(new Artifact(type), type, wrapper);
    }

    @Override
    public ServiceProviderBuilder addSingleton(Class type) {
        return addService(type, LazyFunction0::new);
    }

    @Override
    public ServiceProviderBuilder addTransient(Class type) {
        return addService(type, Function1.identity());
    }

    @Override
    @SuppressWarnings("unchecked")
    public ServiceProviderBuilder addService(Artifact artifact, Function0 supplier) {
        Objects.requireNonNull(artifact);
        Objects.requireNonNull(supplier);
        any.remove(artifact);
        strong.put(artifact, (Function0) supplier);
        return this;
    }

    @Override
    public  ServiceProviderBuilder addService(Class type, Function0 supplier) {
        return addService(new Artifact(type), supplier);
    }

    @Override
    public ServiceProviderBuilder removeService(Artifact artifact) {
        Objects.requireNonNull(artifact);
        strong.remove(artifact);
        any.remove(artifact);
        return this;
    }

    @Override
    public ServiceProviderBuilder removeService(Class type) {
        return removeService(new Artifact(type));
    }

    /**
     * Builds a ready-to-use {@link ServiceProvider} implementation.
     * Override this method instead of {@link ServiceProviderBuilder#build()},
     * so as not to worry about resetting the builder to its original state.
     * Can handle checked exceptions.
     *
     * @return {@link ServiceProvider} instance
     * @throws Throwable if any problems occurs
     */
    protected abstract ServiceProvider checkedBuild() throws Throwable;

    @Override
    public ServiceProvider build() {
        try {
            var ret = checkedBuild();
            reset();
            return ret;
        } catch (Error | RuntimeException e) {
            reset();
            throw e;
        } catch (Throwable e) {
            reset();
            throw new RuntimeException(e);
        }
    }

    protected static final class Entry {
        Class implementation;
        Function1, Function0> wrapper;

        @SuppressWarnings("unchecked")
        Entry(Class implementation, Function1 wrapper) {
            this.implementation = implementation;
            this.wrapper = (Function1, Function0>) wrapper;
        }

        static  Entry of(Class implementation, Function1, Function0> wrapper) {
            return new Entry(implementation, wrapper);
        }
    }
}