Skip to content

Commit 4669172

Browse files
committed
[JPMS] Spec modularity patterns (passing lookup)
1 parent b87e2f7 commit 4669172

File tree

14 files changed

+280
-0
lines changed

14 files changed

+280
-0
lines changed
Lines changed: 20 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
<?xml version="1.0"?>
2+
<project
3+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"
4+
xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
5+
<modelVersion>4.0.0</modelVersion>
6+
<parent>
7+
<groupId>org.hibernate.demos.specmodules</groupId>
8+
<artifactId>spec-modules-parent</artifactId>
9+
<version>1.0-SNAPSHOT</version>
10+
</parent>
11+
<artifactId>acme-fieldreader-impl</artifactId>
12+
13+
<dependencies>
14+
<dependency>
15+
<groupId>${project.groupId}</groupId>
16+
<artifactId>fieldreader-spec</artifactId>
17+
<version>${project.version}</version>
18+
</dependency>
19+
</dependencies>
20+
</project>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
package acme.fieldreader.impl;
2+
3+
import fieldreader.spec.FieldValueReader;
4+
import fieldreader.spec.bootstrap.BootstrapDelegate;
5+
6+
public class AcmeBootstrapDelegate implements BootstrapDelegate {
7+
8+
@Override
9+
public FieldValueReader getFieldValueReader(LookupProvider lookupProvider) {
10+
return new FieldValueReaderImpl( lookupProvider );
11+
}
12+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
package acme.fieldreader.impl;
2+
3+
import fieldreader.spec.FieldValueReader;
4+
import fieldreader.spec.bootstrap.BootstrapDelegate.LookupProvider;
5+
6+
public class FieldValueReaderImpl implements FieldValueReader {
7+
8+
private LookupProvider lookupProvider;
9+
10+
public FieldValueReaderImpl(LookupProvider lookupProvider) {
11+
this.lookupProvider = lookupProvider;
12+
}
13+
14+
@Override
15+
public Object getFieldValue(Object o, String fieldName) {
16+
try {
17+
return lookupProvider.getPrivateLookup( o )
18+
.unreflectVarHandle( o.getClass().getDeclaredField( fieldName ) )
19+
.get( o );
20+
}
21+
catch (Throwable e) {
22+
throw new RuntimeException( e );
23+
}
24+
}
25+
}
Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
module acme.fieldreader.impl {
2+
requires fieldreader.spec;
3+
provides fieldreader.spec.bootstrap.BootstrapDelegate with acme.fieldreader.impl.AcmeBootstrapDelegate;
4+
}
Lines changed: 55 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
<?xml version="1.0"?>
2+
<project
3+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"
4+
xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
5+
<modelVersion>4.0.0</modelVersion>
6+
<parent>
7+
<groupId>org.hibernate.demos.specmodules</groupId>
8+
<artifactId>spec-modules-parent</artifactId>
9+
<version>1.0-SNAPSHOT</version>
10+
</parent>
11+
<artifactId>com-example-beans</artifactId>
12+
13+
<dependencies>
14+
<dependency>
15+
<groupId>${project.groupId}</groupId>
16+
<artifactId>fieldreader-spec</artifactId>
17+
<version>${project.version}</version>
18+
</dependency>
19+
<dependency>
20+
<groupId>${project.groupId}</groupId>
21+
<artifactId>acme-fieldreader-impl</artifactId>
22+
<version>${project.version}</version>
23+
<scope>runtime</scope>
24+
</dependency>
25+
</dependencies>
26+
27+
<build>
28+
<plugins>
29+
<plugin>
30+
<groupId>org.codehaus.mojo</groupId>
31+
<artifactId>exec-maven-plugin</artifactId>
32+
<version>1.6.0</version>
33+
<executions>
34+
<execution>
35+
<phase>integration-test</phase>
36+
<goals>
37+
<goal>exec</goal>
38+
</goals>
39+
<configuration>
40+
<executable>java</executable>
41+
<longModulepath>false</longModulepath>
42+
<arguments>
43+
<argument>-ea</argument>
44+
<argument>--module-path</argument>
45+
<modulepath/>
46+
<argument>--module</argument>
47+
<argument>com.example.beans/com.example.main.FieldReaderTest</argument>
48+
</arguments>
49+
</configuration>
50+
</execution>
51+
</executions>
52+
</plugin>
53+
</plugins>
54+
</build>
55+
</project>
Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
package com.example.beans;
2+
3+
public class MyEntity {
4+
5+
private String name;
6+
7+
public MyEntity(String name) {
8+
this.name = name;
9+
}
10+
11+
public String getName() {
12+
return name;
13+
}
14+
}
Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
package com.example.main;
2+
3+
import com.example.beans.MyEntity;
4+
5+
import fieldreader.spec.FieldReaderApi;
6+
import fieldreader.spec.FieldValueReader;
7+
8+
public class FieldReaderTest {
9+
10+
public static void main(String[] args) {
11+
FieldValueReader fieldValueReader = FieldReaderApi.getFieldValueReader();
12+
Object value = fieldValueReader.getFieldValue( new MyEntity( "bob" ), "name" );
13+
assert "bob".equals( value );
14+
}
15+
}
Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
module com.example.beans {
2+
requires fieldreader.spec;
3+
opens com.example.beans to fieldreader.spec;
4+
}
Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<?xml version="1.0"?>
2+
<project
3+
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"
4+
xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
5+
<modelVersion>4.0.0</modelVersion>
6+
<parent>
7+
<groupId>org.hibernate.demos.specmodules</groupId>
8+
<artifactId>spec-modules-parent</artifactId>
9+
<version>1.0-SNAPSHOT</version>
10+
</parent>
11+
<artifactId>fieldreader-spec</artifactId>
12+
<name>spec-api</name>
13+
</project>
Lines changed: 47 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
package fieldreader.spec;
2+
3+
import java.lang.invoke.MethodHandles;
4+
import java.lang.invoke.MethodHandles.Lookup;
5+
import java.util.ServiceLoader;
6+
7+
import fieldreader.spec.bootstrap.BootstrapDelegate;
8+
import fieldreader.spec.bootstrap.BootstrapDelegate.LookupProvider;
9+
10+
public class FieldReaderApi {
11+
12+
private static final LookupProvider LOOKUP_PROVIDER = new LookupProviderImpl();
13+
14+
public static FieldValueReader getFieldValueReader() {
15+
ServiceLoader<BootstrapDelegate> loader = ServiceLoader.load( BootstrapDelegate.class );
16+
17+
return loader.findFirst()
18+
.orElseThrow( () -> new IllegalStateException( "No provider of " + BootstrapDelegate.class.getName() + " available" ) )
19+
.getFieldValueReader( LOOKUP_PROVIDER );
20+
}
21+
22+
private static class LookupProviderImpl implements BootstrapDelegate.LookupProvider {
23+
24+
private final ClassValue<Lookup> classValue;
25+
26+
private LookupProviderImpl() {
27+
classValue = new ClassValue<Lookup>() {
28+
29+
@Override
30+
protected Lookup computeValue(Class<?> type) {
31+
FieldValueReader.class.getModule().addReads( type.getModule() );
32+
try {
33+
return MethodHandles.privateLookupIn( type, MethodHandles.lookup() );
34+
}
35+
catch (IllegalAccessException e) {
36+
throw new RuntimeException( e );
37+
}
38+
}
39+
};
40+
}
41+
42+
@Override
43+
public Lookup getPrivateLookup(Object o) {
44+
return classValue.get( o.getClass() );
45+
}
46+
}
47+
}

0 commit comments

Comments
 (0)