io.helidon.codegen.ModuleInfo Maven / Gradle / Ivy
/*
* Copyright (c) 2023, 2024 Oracle and/or its affiliates.
*
* 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 io.helidon.codegen;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Consumer;
import io.helidon.common.Errors;
import io.helidon.common.types.TypeName;
/**
* Module info.
*
* @see #builder()
*/
public interface ModuleInfo {
/**
* The default module name (i.e., "unnamed").
*/
String DEFAULT_MODULE_NAME = "unnamed";
/**
* The file name in sources ({@value}).
*/
String FILE_NAME = "module-info.java";
/**
* Create a new fluent API builder to customize configuration.
*
* @return a new builder
*/
static Builder builder() {
return new Builder();
}
/**
* Create a new fluent API builder from an existing instance.
*
* @param instance an existing instance used as a base for the builder
* @return a builder based on an instance
*/
static Builder builder(ModuleInfo instance) {
return ModuleInfo.builder().from(instance);
}
/**
* Name of the module.
*
* @return module name
*/
String name();
/**
* Whether this module is declared as open module.
*
* @return whether this module is open
*/
boolean isOpen();
/**
* Declared dependencies of the module.
*
* @return list of requires
*/
List requires();
/**
* Exports of the module.
*
* @return map of exported packages (exports x.y.z to a.b.cSomeModule).
*/
Map> exports();
/**
* Used service loader providers.
*
* @return list of used provider interfaces
*/
List uses();
/**
* Map of provider interfaces to provider implementations provided by this module.
*
* @return map of interface to implementations
*/
Map> provides();
/**
* Map of opened packages to modules (if any).
*
* @return map of package to modules
*/
Map> opens();
/**
* first export that does not export to a specific module (if present).
*
* @return package that is exported
*/
default Optional firstUnqualifiedExport() {
return exports().entrySet()
.stream()
.filter(it -> it.getValue().isEmpty())
.map(Map.Entry::getKey)
.findFirst();
}
/**
* Fluent API builder base for {@link io.helidon.codegen.ModuleInfo}.
*
* @param type of the builder extending this abstract builder
*/
abstract class BuilderBase> implements io.helidon.common.Builder {
private final List uses = new ArrayList<>();
private final List requires = new ArrayList<>();
private final Map> exports = new LinkedHashMap<>();
private final Map> provides = new LinkedHashMap<>();
private final Map> opens = new LinkedHashMap<>();
private boolean isOpen = false;
private String name;
/**
* Protected to support extensibility.
*/
protected BuilderBase() {
}
/**
* Update this builder from an existing prototype instance.
*
* @param prototype existing prototype to update this builder from
* @return updated builder instance
*/
public BUILDER from(ModuleInfo prototype) {
name(prototype.name());
isOpen(prototype.isOpen());
addRequires(prototype.requires());
addExports(prototype.exports());
addUses(prototype.uses());
addProvides(prototype.provides());
addOpens(prototype.opens());
return identity();
}
/**
* Update this builder from an existing prototype builder instance.
*
* @param builder existing builder prototype to update this builder from
* @return updated builder instance
*/
public BUILDER from(BuilderBase> builder) {
builder.name().ifPresent(this::name);
isOpen(builder.isOpen());
addRequires(builder.requires());
addExports(builder.exports());
addUses(builder.uses());
addProvides(builder.provides());
addOpens(builder.opens());
return identity();
}
/**
* Name of the module.
*
* @param name module name
* @return updated builder instance
* @see #name()
*/
public BUILDER name(String name) {
Objects.requireNonNull(name);
this.name = name;
return identity();
}
/**
* Whether this module is declared as open module.
*
* @param isOpen whether this module is open
* @return updated builder instance
* @see #isOpen()
*/
public BUILDER isOpen(boolean isOpen) {
this.isOpen = isOpen;
return identity();
}
/**
* Declared dependencies of the module.
*
* @param requires list of requires
* @return updated builder instance
* @see #requires()
*/
public BUILDER requires(List extends ModuleInfoRequires> requires) {
Objects.requireNonNull(requires);
this.requires.clear();
this.requires.addAll(requires);
return identity();
}
/**
* Declared dependencies of the module.
*
* @param requires list of requires
* @return updated builder instance
* @see #requires()
*/
public BUILDER addRequires(List extends ModuleInfoRequires> requires) {
Objects.requireNonNull(requires);
this.requires.addAll(requires);
return identity();
}
/**
* Declared dependencies of the module.
*
* @param require list of requires
* @return updated builder instance
* @see #requires()
*/
public BUILDER addRequire(ModuleInfoRequires require) {
Objects.requireNonNull(require);
this.requires.add(require);
return identity();
}
/**
* Exports of the module.
*
* @param exports list of exported packages
* @return updated builder instance
* @see #exports()
*/
public BUILDER exports(Map extends String, List> exports) {
Objects.requireNonNull(exports);
this.exports.clear();
this.exports.putAll(exports);
return identity();
}
/**
* Exports of the module.
*
* @param exports list of exported packages
* @return updated builder instance
* @see #exports()
*/
public BUILDER addExports(Map extends String, List> exports) {
Objects.requireNonNull(exports);
this.exports.putAll(exports);
return identity();
}
/**
* This method adds a new value to the map, or replaces it if the key already exists.
*
* @param packageName key to add or replace
* @param moduleNames new value for the key
* @return updated builder instance
* @see #opens()
*/
public BUILDER putExports(String packageName, List moduleNames) {
Objects.requireNonNull(packageName);
Objects.requireNonNull(moduleNames);
this.exports.put(packageName, List.copyOf(moduleNames));
return identity();
}
/**
* Exports of the module.
*
* @param export package to export
* @param to exported to a module
* @return updated builder instance
* @see #exports()
*/
public BUILDER addExport(String export, String to) {
Objects.requireNonNull(export);
Objects.requireNonNull(to);
this.opens.compute(export, (k, v) -> {
v = v == null ? new ArrayList<>() : new ArrayList<>(v);
v.add(to);
return v;
});
return identity();
}
/**
* Used service loader providers.
*
* @param uses list of used provider interfaces
* @return updated builder instance
* @see #uses()
*/
public BUILDER uses(List extends TypeName> uses) {
Objects.requireNonNull(uses);
this.uses.clear();
this.uses.addAll(uses);
return identity();
}
/**
* Used service loader providers.
*
* @param uses list of used provider interfaces
* @return updated builder instance
* @see #uses()
*/
public BUILDER addUses(List extends TypeName> uses) {
Objects.requireNonNull(uses);
this.uses.addAll(uses);
return identity();
}
/**
* Used service loader providers.
*
* @param use list of used provider interfaces
* @return updated builder instance
* @see #uses()
*/
public BUILDER addUse(TypeName use) {
Objects.requireNonNull(use);
this.uses.add(use);
return identity();
}
/**
* Used service loader providers.
*
* @param consumer list of used provider interfaces
* @return updated builder instance
* @see #uses()
*/
public BUILDER addUse(Consumer consumer) {
Objects.requireNonNull(consumer);
var builder = TypeName.builder();
consumer.accept(builder);
this.uses.add(builder.build());
return identity();
}
/**
* This method replaces all values with the new ones.
*
* @param provides map of interface to implementations
* @return updated builder instance
* @see #provides()
*/
public BUILDER provides(Map extends TypeName, List> provides) {
Objects.requireNonNull(provides);
this.provides.clear();
this.provides.putAll(provides);
return identity();
}
/**
* This method keeps existing values, then puts all new values into the map.
*
* @param provides map of interface to implementations
* @return updated builder instance
* @see #provides()
*/
public BUILDER addProvides(Map extends TypeName, List> provides) {
Objects.requireNonNull(provides);
this.provides.putAll(provides);
return identity();
}
/**
* This method adds a new value to the map value, or creates a new value.
*
* @param key key to add to
* @param provide additional value for the key
* @return updated builder instance
* @see #provides()
*/
public BUILDER addProvide(TypeName key, TypeName provide) {
Objects.requireNonNull(key);
Objects.requireNonNull(provide);
this.provides.compute(key, (k, v) -> {
v = v == null ? new ArrayList<>() : new ArrayList<>(v);
v.add(provide);
return v;
});
return identity();
}
/**
* This method adds a new value to the map value, or creates a new value.
*
* @param key key to add to
* @param provides additional values for the key
* @return updated builder instance
* @see #provides()
*/
public BUILDER addProvides(TypeName key, List provides) {
Objects.requireNonNull(key);
Objects.requireNonNull(provides);
this.provides.compute(key, (k, v) -> {
v = v == null ? new ArrayList<>() : new ArrayList<>(v);
v.addAll(provides);
return v;
});
return identity();
}
/**
* This method adds a new value to the map, or replaces it if the key already exists.
*
* @param key key to add or replace
* @param provide new value for the key
* @return updated builder instance
* @see #provides()
*/
public BUILDER putProvide(TypeName key, List provide) {
Objects.requireNonNull(key);
Objects.requireNonNull(provide);
this.provides.put(key, List.copyOf(provide));
return identity();
}
/**
* This method replaces all values with the new ones.
*
* @param opens map of package to modules
* @return updated builder instance
* @see #opens()
*/
public BUILDER opens(Map extends String, List> opens) {
Objects.requireNonNull(opens);
this.opens.clear();
this.opens.putAll(opens);
return identity();
}
/**
* This method keeps existing values, then puts all new values into the map.
*
* @param opens map of package to modules
* @return updated builder instance
* @see #opens()
*/
public BUILDER addOpens(Map extends String, List> opens) {
Objects.requireNonNull(opens);
this.opens.putAll(opens);
return identity();
}
/**
* This method adds a new value to the map value, or creates a new value.
*
* @param key key to add to
* @param open additional value for the key
* @return updated builder instance
* @see #opens()
*/
public BUILDER addOpen(String key, String open) {
Objects.requireNonNull(key);
Objects.requireNonNull(open);
this.opens.compute(key, (k, v) -> {
v = v == null ? new ArrayList<>() : new ArrayList<>(v);
v.add(open);
return v;
});
return identity();
}
/**
* This method adds a new value to the map value, or creates a new value.
*
* @param key key to add to
* @param opens additional values for the key
* @return updated builder instance
* @see #opens()
*/
public BUILDER addOpens(String key, List opens) {
Objects.requireNonNull(key);
Objects.requireNonNull(opens);
this.opens.compute(key, (k, v) -> {
v = v == null ? new ArrayList<>() : new ArrayList<>(v);
v.addAll(opens);
return v;
});
return identity();
}
/**
* This method adds a new value to the map, or replaces it if the key already exists.
*
* @param key key to add or replace
* @param open new value for the key
* @return updated builder instance
* @see #opens()
*/
public BUILDER putOpen(String key, List open) {
Objects.requireNonNull(key);
Objects.requireNonNull(open);
this.opens.put(key, List.copyOf(open));
return identity();
}
/**
* Name of the module.
*
* @return the name
*/
public Optional name() {
return Optional.ofNullable(name);
}
/**
* Whether this module is declared as open module.
*
* @return the is open
*/
public boolean isOpen() {
return isOpen;
}
/**
* Declared dependencies of the module.
*
* @return the requires
*/
public List requires() {
return requires;
}
/**
* Exports of the module.
*
* @return the exports
*/
public Map> exports() {
return exports;
}
/**
* Used service loader providers.
*
* @return the uses
*/
public List uses() {
return uses;
}
/**
* Map of provider interfaces to provider implementations provided by this module.
*
* @return the provides
*/
public Map> provides() {
return provides;
}
/**
* Map of opened packages to modules (if any).
*
* @return the opens
*/
public Map> opens() {
return opens;
}
@Override
public String toString() {
return "ModuleInfoBuilder{"
+ "name=" + name + ","
+ "isOpen=" + isOpen + ","
+ "requires=" + requires + ","
+ "exports=" + exports + ","
+ "uses=" + uses + ","
+ "provides=" + provides + ","
+ "opens=" + opens
+ "}";
}
/**
* Handles providers and decorators.
*/
protected void preBuildPrototype() {
}
/**
* Validates required properties.
*/
protected void validatePrototype() {
Errors.Collector collector = Errors.collector();
if (name == null) {
collector.fatal(getClass(), "Property \"name\" must not be null, but not set");
}
collector.collect().checkValid();
}
/**
* Generated implementation of the prototype, can be extended by descendant prototype implementations.
*/
protected static class ModuleInfoImpl implements ModuleInfo {
private final boolean isOpen;
private final List uses;
private final List requires;
private final Map> exports;
private final Map> provides;
private final Map> opens;
private final String name;
/**
* Create an instance providing a builder.
*
* @param builder extending builder base of this prototype
*/
protected ModuleInfoImpl(ModuleInfo.BuilderBase> builder) {
this.name = builder.name().get();
this.isOpen = builder.isOpen();
this.requires = List.copyOf(builder.requires());
this.exports = Map.copyOf(builder.exports());
this.uses = List.copyOf(builder.uses());
this.provides = Collections.unmodifiableMap(new LinkedHashMap<>(builder.provides()));
this.opens = Collections.unmodifiableMap(new LinkedHashMap<>(builder.opens()));
}
@Override
public String name() {
return name;
}
@Override
public boolean isOpen() {
return isOpen;
}
@Override
public List requires() {
return requires;
}
@Override
public Map> exports() {
return exports;
}
@Override
public List uses() {
return uses;
}
@Override
public Map> provides() {
return provides;
}
@Override
public Map> opens() {
return opens;
}
@Override
public String toString() {
return "ModuleInfo{"
+ "name=" + name + ","
+ "isOpen=" + isOpen + ","
+ "requires=" + requires + ","
+ "exports=" + exports + ","
+ "uses=" + uses + ","
+ "provides=" + provides + ","
+ "opens=" + opens
+ "}";
}
@Override
public boolean equals(Object o) {
if (o == this) {
return true;
}
if (!(o instanceof ModuleInfo other)) {
return false;
}
return Objects.equals(name, other.name())
&& isOpen == other.isOpen()
&& Objects.equals(requires, other.requires())
&& Objects.equals(exports, other.exports())
&& Objects.equals(uses, other.uses())
&& Objects.equals(provides, other.provides())
&& Objects.equals(opens, other.opens());
}
@Override
public int hashCode() {
return Objects.hash(name, isOpen, requires, exports, uses, provides, opens);
}
}
}
/**
* Fluent API builder for {@link io.helidon.codegen.ModuleInfo}.
*/
class Builder extends BuilderBase implements io.helidon.common.Builder {
private Builder() {
}
@Override
public ModuleInfo build() {
preBuildPrototype();
validatePrototype();
return new ModuleInfoImpl(this);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy