Java¶
ALFA tools generates set of Java POJO classes for the model defined.
The generated code has a single dependency;
<dependency>
<artifactId>alfa-runtime-java</artifactId>
<groupId>com.schemarisealfa.runtime</groupId>
<version>3.4.0</version>
</dependency>
Refer to the ALFA Maven Plugin for setting up Maven to generate ALFA into Java.
Exporter Concepts¶
- Code is generated into matching packages and as a
.java
file per user-defined type. - Immutablity is strongly enforced in generated code.
- Similar to API patterns used in POJO/serialisation frameworks, ALFA supports builder pattern and method chaining.
E.g.
MyType.builder().setProp1(p1).setProp2(p2).build();
- Core Java interfaces are used to represent ALFA collection ( e.g. java.util.Map/Set/List/Optional, java.util.stream.Stream ).
- To minimize dependencies, generated code only depend on the runtime library, and that in turn only depends on Jackson JSON parser.
- If assert are defined in the model, code is generated in the
Builder
such that beforebuild()
completes, all assert conditions and expressions are applied on the object.
Exporter Example¶
Consider the following demo.alfa
definition. Lets look at what Java code is generated from this.
namespace com.acme
key BankAccountKey {
IBAN : string
}
entity BankAccount key BankAccountKey {
Name : string
Transactions : list< Transaction >
RegisteredDate : date
Type : enum< Silver, Gold, Platinum >
}
record Transaction {
Amount : double
Description : string
Date : date
CorrespondingIBAN : string
}
service BankService( AuthToken : uuid ) {
getAccount( IBAN : string ) : try< BankAccount >
getTransactions( k : BankAccountKey ) : stream< Transaction >
}
This can be generated to Java using the ALFA CLI - alfa -c -e java -o gen/java demo.alfa
.
The generated Java code is shown in the screenshot below.
- A
.java
file is generated per user-defined type. - The in-line enum in BankAccount Type field is generated as a top level enum.
- Use platform native types -
java.*
classes over adding unnecessary abstractions. - Utility inner classes are generated -
Builder
,Mutator
,TypeDescriptor
andConcrete
, and are enclosed in auto code-folded blocks. - Inner classes are generated with auto code-folding for IntelliJ.
Type Mappings¶
ALFA types are mapped to Java as outlined in the table below.
AlfaType | JavaType | Comments |
---|---|---|
entity | Java class | Implements alfa.rt.Entity with method to get entity key |
enum | Java enum | Implements alfa.rt.Enum |
key | Java class | Implements alfa.rt.Key |
record | Java class | Implements alfa.rt.Record |
service | Java interface | Implements alfa.rt.Service, and contains all method declarations |
trait | Java interface | Implements alfa.rt.Trait |
union | Java interface | Implements alfa.rt.Union, and a class per union case containing a single value |
compressed< T > | alfa.rt.Compressed< T > | Where T can be any ALFA type |
future< T > | java.util.concurrent.Future< T > | Where T can be any ALFA type |
either< L, R > | alfa.rt.Either< L, R > | Where L and R can be any ALFA type |
encrypted< T > | alfa.rt.Encrypted< T > | Where T can be any ALFA type |
try< T > | alfa.rt.Try< T > | Where T can be any ALFA type |
stream< T > | java.util.stream.Stream< T > | Where T can be any ALFA type |
T ? | java.lang.Optional< T > | Where T can be any ALFA type |
list< T > | java.util.List< T > | Where T can be any ALFA type |
map< K, V > | java.util.Map< K, V > | Where K and V can be any ALFA type |
set< T > | java.util.Set< T > | Where T can be any ALFA type |
boolean | boolean | |
byte | byte | |
binary | byte[ ] | |
char | char | |
int | int | |
short | short | |
long | long | |
double | double | |
float | float | |
string | String | |
date | java.time.LocalDate | |
time | java.time.LocalTime | |
datetime | java.time.LocalDatetime | |
duration | java.time.Duration | |
uuid | java.util.UUID | |
uri | java.net.URI | |
Runtime Library¶
The ALFA runtime library, alfa-rt-java-core
, contains the core interfaces and utilities required to use the generated code along
with a JSON encoder/decoder.
Example creating an ALFA Java POJO:
import com.schemarise.alfa.runtime.*;
public void testBuildingPojo() throws IOException {
BankAccount.Builder accBuilder = BankAccount.builder();
accBuilder.setName("Joe Bloggs").setRegisteredDate( LocalDate.of(1990, 10, 12));
accBuilder.setType( BankAccount_Type_.Gold);
BankAccountKey.Builder kb = BankAccountKey.builder();
accBuilder.key( kb.setIBAN("IBAN342").build() );
for (int i = 0; i < 5; i++) {
Transaction.Builder tb = Transaction.builder();
tb.setAmount( Math.random() * 1000 ).
setCorrespondingIBAN( "ABCDE" + i ).
setDate( LocalDate.of( 2020, 2, i+1)).
setDescription("Purchase from Store" + i );
accBuilder.addTransactions( tb.build() );
}
// Create JSON representation
String json = Alfa.jsonCodec().toJsonString(ba);
System.out.println(json);
// Decode JSON
BankAccount decoded = Alfa.jsonCodec().fromJsonString(json);
Assert.assertEquals(ba, decoded);
}
Refer to Runtime APIs & Guides Java section for documentation on the ALFA Java APIs and a How-To guide.
Runtime Utilities Library¶
Further to the ALFA Java Runtime library, an optional ALFA Runtime Utilities library is available.
Randomiser¶
This is a useful utility that generates a random value generators which is extremely useful for testing applications with sample data. The Java ALFA Object randomiser is able to randomise any value also while respecting any ALFA field constraints.
Given an ALFA definition such as:
record Sample.Result {
// 3x3 matrix of numbers between 10 to 100
Matrix3x3 : list<
list<
int(10,100)
>(3,3)
>(3,3)
}
The randomiser generates the following object, serialized to JSON.
Naturally, the generator respects the model and constraints to accurately generate a valid object. The randomiser is fully model-aware, as a result can generate random values for any ALFA definition.
{
"$type":"Sample.Result",
"Matrix": [
[96, 10, 92],
[26, 97, 20],
[14, 44, 83]
]
}
If no constraints are defined, the randomiser will create lists with a default length of 5 elements. In the example above, when the JSON is being read, the ALFA Runtime will validate the values against the constraints that have been defined.
Refer to Runtime APIs & Guides Java section How-To guide for examples of using the Randomiser.