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 extends T> 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 extends T> implementation,
Function1, Function0> wrapper) {
return addService(new Artifact(type), implementation, wrapper);
}
@Override
public ServiceProviderBuilder addSingleton(Class type, Class extends T> implementation) {
return addService(type, implementation, LazyFunction0::new);
}
@Override
public ServiceProviderBuilder addTransient(Class type, Class extends T> 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
© 2015 - 2025 Weber Informatics LLC | Privacy Policy