Java SE and the Java ecosystem offer a multitude of Metaprogramming Apis. They serve the same purpose just in different contexts. Annotation processing to analyse classes in the during compilation process, or Reflection to do the same at runtime. Each having unique concepts and quirks. This makes Metaprogramming harder to understand and creates a maintenance overhead. It’s very hard to switch from one Api to another. Once something is written with java.lang.reflect it’s hard to change it to an Annotation Processor. Some JDK Apis are not up to the normal standard.
This project is an Abstraction for Metaprogramming
It has two Types of Apis
This Api works with any kind of Metaprogramming, but the featureset is limited to common operations
Example
The Api is request based. As a caller you can request for example a field of a class. Accessing fields may or may not be supported.
@Test
void request()
{
//adapter for the reflection api
C_Class systemClass = R_Adapter.generalize(System.class);
//request the field "out" for the class java.lang.System
Response<C_Field> out = Provider.request(systemClass,
Operations.DECLARED_GET_FIELD,
"out");
switch (out)
{
//the implementation may not support this operation
//e.g. it's impossible to access fields with reflection
case Response.Unsupported<C_Field> unsupported -> Assertions.fail();
//the implementation may support this operation, but there is no
//result for this instance
//e.g. the class java.lang.System does not have a field called "out"
case Response.Empty<C_Field> empty -> Assertions.fail();
//accessing fields via reflection is possible and java.lang.System
//does have a field called "out" therefore a result is expected
case Response.Result<C_Field> result -> assertNotNull(result.value());
}
}
Or use a convenience method if Optional.empty()
or throwing an
Exception is a fitting default behavior.
@Test
void requestOrEmpty()
{
//adapter for the reflection api
C_Class systemClass = R_Adapter.generalize(System.class);
//request the field "out" for the class java.lang.System.
//If its unsupported an Empty Optional is returned
Optional<C_Field> out = Provider.requestOrEmpty(systemClass,
Operations.DECLARED_GET_FIELD,
"out");
assertTrue(out.isPresent());
}
@Test
void requestOrThrow()
{
//adapter for the reflection api
C_Class systemClass = R_Adapter.generalize(System.class);
//request the field "out" for the class java.lang.System.
//If its unsupported an Exception is thrown
C_Field out = Provider.requestOrThrow(systemClass,
Operations.DECLARED_GET_FIELD,
"out");
assertNotNull(out);
}
There is one Api for each kind of Metaprogramming. One for Annotation processing and another very similar one for reflection. Each being able to support the complete featureset.
Example
This Annotation Processor generates Builder
package io.determann.shadow.builder;
import io.determann.shadow.api.annotation_processing.AP_Context;
import io.determann.shadow.api.annotation_processing.AP_Processor;
import io.determann.shadow.api.lang_model.shadow.LM_Nameable;
import io.determann.shadow.api.lang_model.shadow.LM_QualifiedNameable;
import io.determann.shadow.api.lang_model.shadow.structure.LM_Property;
import io.determann.shadow.api.lang_model.shadow.type.LM_Class;
import io.determann.shadow.api.lang_model.shadow.type.LM_Type;
import java.util.List;
import java.util.stream.Collectors;
import static org.apache.commons.lang3.StringUtils.capitalize;
import static org.apache.commons.lang3.StringUtils.uncapitalize;
/**
* Builds a companion Builder class for each annotated class
*/
public class ShadowBuilderProcessor extends AP_Processor
{
@Override
public void process(final AP_Context context)
{
//iterate over every class annotated with the BuilderPattern annotation
for (LM_Class aClass : context
.getClassesAnnotatedWith("io.determann.shadow.builder.BuilderPattern"))
{
String toBuildQualifiedName = aClass.getQualifiedName();
//qualifiedName of the companion builder class
String builderQualifiedName = toBuildQualifiedName + "ShadowBuilder";
//simpleName of the companion builder class
String builderSimpleName = aClass.getName() + "ShadowBuilder";
String builderVariableName = uncapitalize(builderSimpleName);
//create a record holding the code needed to render a property in the builder
List<BuilderElement> builderElements =
aClass.getProperties()
.stream()
.filter(LM_Property::isMutable)
.map(property -> renderProperty(builderSimpleName,
builderVariableName,
property))
.toList();
//writes the builder
context.writeAndCompileSourceFile(builderQualifiedName,
renderBuilder(aClass,
toBuildQualifiedName,
builderSimpleName,
builderVariableName,
builderElements));
}
}
/**
* renders a companion builder class
*/
private String renderBuilder(final LM_Class aClass,
final String toBuildQualifiedName,
final String builderSimpleName,
final String builderVariableName,
final List<BuilderElement> builderElements)
{
String fields = builderElements.stream()
.map(BuilderElement::field)
.collect(Collectors.joining("\n\n"));
String mutators = builderElements.stream()
.map(BuilderElement::mutator)
.collect(Collectors.joining("\n\n"));
String setterInvocations = builderElements.stream()
.map(BuilderElement::toBuildSetter)
.collect(Collectors.joining("\n\n"));
return """
package %1$s;
public class %2$s{
%3$s
%4$s
public %5$s build() {
%5$s %6$s = new %5$s();
%7$s
return %6$s;
}
}
""".formatted(aClass.getPackage().getQualifiedName(),
builderSimpleName,
fields,
mutators,
toBuildQualifiedName,
builderVariableName,
setterInvocations);
}
/**
* Creates a {@link BuilderElement} for each property of the annotated pojo
*/
private BuilderElement renderProperty(final String builderSimpleName,
final String builderVariableName,
final LM_Property property)
{
String propertyName = property.getName();
String type = renderType(property.getType());
String field = "private " + type + " " + propertyName + ";";
String mutator = """
public %1$s with%2$s(%3$s %4$s) {
this.%4$s = %4$s;
return this;
}
""".formatted(builderSimpleName,
capitalize(propertyName),
type,
propertyName);
String toBuildSetter = builderVariableName + "." +
property.getSetterOrThrow().getName() +
"(" + propertyName + ");";
return new BuilderElement(field, mutator, toBuildSetter);
}
/**
* Used to render the code needed to render a property in the builder
*
* @param field ones rendered will hold the values being used to build the pojo
* @param mutator ones rendered will set the value of the {@link #field}
* @param toBuildSetter ones rendered will modify the build pojo
*/
private record BuilderElement(String field,
String mutator,
String toBuildSetter) {}
private static String renderType(LM_Type type)
{
if (type instanceof LM_QualifiedNameable qualifiedNameable)
{
return qualifiedNameable.getQualifiedName();
}
if (type instanceof LM_Nameable nameable)
{
return nameable.getName();
}
return type.toString();
}
}
- Annotation Processing
- java.lang.model
- reflection (experimental)
- Improve Annotation Processing Support
- Add a Java DSL Api for Rendering
- Add support for specific Annotations
- Improve Support for other Metaprogramming Apis
- Move more of the implementation to the
shadow-implementation-support
module for better consistency and easier support of multiple implementations - Support the complete Lexical Structure
- Create a maven archetype to reduce the effort of supporting new Metaprogramming Implementations
- Move more of the implementation to the