-
Notifications
You must be signed in to change notification settings - Fork 2
Extensions
AutoRecord Extensions provide a way to customize the record generation process in AutoRecord. They allow you to extend and modify the behavior of AutoRecord to better suit your specific use cases and requirements. Extensions can be used to add additional features, customize code generation, and provide extra functionality beyond what is available with the default AutoRecord behavior.
To create a custom extension, you need to implement at least one of the following interfaces that extends the AutoRecordExtension interface:
These interfaces provide a set of methods that you can override to customize various aspects of the record generation process.
Please see Javadoc comments of specific interfaces to see what methods needs to be implemented while developing an extension
To add an extension to the record generation process, @AutoRecord.Extension annotation needs to be used. The annotation allows to specify the class of the extension and parameters that will be passed to it.
Here is an example of what it might look like:
@AutoRecord.Extension(extensionClass = LoggingExtension.class, parameters = "info")
To add extensions to the generation of a single record, you have to use the @AutoRecord.Extension
annotations next to @AutoRecod
. You can add many extensions.
Here is an example of what it might look like:
@AutoRecord
@AutoRecord.Extension(extensionClass = IsPersonAdultVerifierExtension.class, parameters = "java.lang.IllegalStateException")
@AutoRecord.Extension(extensionClass = LoggingExtension.class)
To add extensions to the records generation with custom annotation, you have to specify the @AutoRecord.Extension
annotations in @AutoRecord.Template#extensions()
method.
Here is an example of what such custom annotation might look like:
package pl.com.labaj.autorecord;
import io.soabase.recordbuilder.core.RecordBuilder;
import pl.com.labaj.autorecord.extension.compact.IsPersonAdultVerifierExtension;
import pl.com.labaj.autorecord.extension.compact.LoggingExtension;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import static java.lang.annotation.ElementType.TYPE;
import static java.lang.annotation.RetentionPolicy.SOURCE;
@Retention(SOURCE)
@Target(TYPE)
@AutoRecord.Template(
recordOptions = @AutoRecord.Options(withBuilder = true),
builderOptions = @RecordBuilder.Options(useUnmodifiableCollections = true),
extensions = {
@AutoRecord.Extension(extensionClass = LoggingExtension.class, parameters = "info"),
@AutoRecord.Extension(extensionClass = IsPersonAdultVerifierExtension.class)
}
)
public @interface AutoRecordWithExtensions {}
Here's example extension:
package pl.com.labaj.autorecord.extension.compact;
import com.squareup.javapoet.CodeBlock;
import pl.com.labaj.autorecord.context.Context;
import pl.com.labaj.autorecord.context.RecordComponent;
import pl.com.labaj.autorecord.context.StaticImports;
import pl.com.labaj.autorecord.extension.CompactConstructorExtension;
import java.util.HashMap;
import java.util.logging.Level;
import java.util.logging.Logger;
import static java.util.logging.Level.FINE;
public class LoggingExtension implements CompactConstructorExtension {
private Level level;
@Override
public void setParameters(String[] parameters) {
if (parameters.length < 1) {
level = FINE;
} else {
level = Level.parse(parameters[0].toUpperCase());
}
}
@Override
public boolean shouldGenerate(boolean isGeneratedByProcessor, Context context) {
return true;
}
@Override
public CodeBlock prefixCompactConstructorContent(Context context, StaticImports staticImports) {
var codeBuilder = CodeBlock.builder();
codeBuilder.addStatement("var map = new $T<$T, $T>()", HashMap.class, String.class, Object.class);
context.components().stream()
.map(RecordComponent::name)
.forEach(name -> codeBuilder.addStatement("map.put($S, $L)", name, name));
codeBuilder.addStatement("var logger = getLogger($L.class.getName())", context.recordName())
.addStatement("logger.log($L, $S, $L)", level, "Parameters passed to record: {0}", "map");
staticImports.add(Logger.class, "getLogger")
.add(Level.class, level.getName());
return codeBuilder.build();
}
}
Here's an example interface:
import pl.com.labaj.autorecord.AutoRecord;
import pl.com.labaj.autorecord.extension.compact.LoggingExtension;
import javax.annotation.Nullable;
@AutoRecord
@AutoRecord.Extension(extensionClass = LoggingExtension.class, parameters = "info")
interface Person {
String name();
@Nullable String surname();
int age();
}
Here's the corresponding generated record that demonstrates builder generation:
import pl.com.labaj.autorecord.GeneratedWithAutoRecord;
import javax.annotation.Nullable;
import javax.annotation.processing.Generated;
import java.util.HashMap;
import static java.util.Objects.requireNonNull;
import static java.util.logging.Level.INFO;
import static java.util.logging.Logger.getLogger;
@Generated("pl.com.labaj.autorecord.AutoRecord")
@GeneratedWithAutoRecord
record PersonRecord(String name, @Nullable String surname, int age) implements Person {
PersonRecord {
// pl.com.labaj.autorecord.extension.compact.LoggingExtension
var map = new HashMap<String, Object>();
map.put("name", name);
map.put("surname", surname);
map.put("age", age);
var logger = getLogger(PersonRecord.class.getName());
logger.log(INFO, "Parameters passed to record: {0}", map);
// pl.com.labaj.autorecord.processor.AutoRecordProcessor
requireNonNull(name, "name must not be null");
}
}