Skip to content

techarts0/whale

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

81 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Whale: A JSR330 Based Java DI Framework

Generic badge Generic badge Generic badge Generic badge Generic badge Generic badge Generic badge Generic badge Generic badge

1. Summary

Whale is a lightweight dependence Injection(DI) container that fully implements JSR330, and supports javax.inject and jakarta.inject API both. If you are a Java developer and familiar with spring framework or google guice, we highly recommend you giving whale a try.

2. Annotations

JSR330's appeal lies in its simplicity, consisting of just 4 annotations and one interface. To enhance flexibility, whale adds 2 annotations: Valued and Bind, as follows:

# Annotation Usage
1 Inject Indicates a field, constructor or method that will be injected with a managed object.
2 Named Gives the managed object a qualifier name, or tells the injector which will be injected in.
3 Valued Injects a value(primitive types like int, string, boolean) or a key from configuration file. (Non JSR330)
4 Singleton There is only one instance of the managed object in DI container.
5 Bind Relates an interface or abstraction to a specific implementation. (Non JSR330)
6 Qualifier Meta annotation.
7 Provider An interface, not an annotation, used for resolving circular dependencies or lazy loading.
8 Ready The method is an initializer of the object. It will be call ONCE after object creating.(Non JSR330)
9 Advice Interceptor. The class will be weaved some appects.(Non JSR330)
10 Advise Interceptor. The method of an interface will be enhanced.(Non JSR330)

Whale supports three dependence types:

  • REF: A managed object in container. XML property is ref;
  • KEY: A configuration from properties file. XML property is key;
  • VAL: A value of primitive type(e. g. int, String, boolean, float), XML property is val;

3. Basic Usage

Whale offers four approaches for managing the dependencies between Java objects. To illustrate these methods, let's examine some sample test code. We'll assume the test code is located in the directory on the classpath: "/tmp/project/demo/bin".

  • The Class Person dependents on the class Mobile, a given number value and some configurations:
package whale.demo;

@Named
@Singleton
public class Person{
    @Inject 
    @Valued(val="3")
    private int id;
    
    @Inject
    @Valued(key="user.name") 
    private String name;
    
    private int age;
    
    @Inject
    private Mobile mobile;
    
    public Person(){}

    @Inject
    public void setAge(@Valued(key="user.age")int age){
        this.age = age;
    }
    
    //Getters and Setters
}
  • The class Mobile dependents on two keys from configuration and is injected via the constructor:
package whale.demo;

@Singleton
public class Mobile{
    private String areaCode;
    private String number;

    @Inject
    public Mobile(@Valued(key="mobile.area")String areaCode, @Valued(key="mobile.number")String number){
        this.areaCode = areaCode;
        this.number = number;
    }
    //Getters & Setters
}
  • To ensure successful testing, we need to prepare a configuration beforehand (/tmp/project/demo/config.properties):
user.age=18
user.name=John Denver
mobile.area=+86
mobile.number=13603166666
  • or some static test data:
public class TestData{
	public static final Map<String, String>
	CONFIGS = Map.of("user.age", "18",
					"user.name", "John Denver", 
             		 "mobile.area", "+86", 
             		 "mobile.number", "13603166666");
}

A. Scan classpath to resolve the dependencies:

The JUNIT test case as following:

public class WhaleTest{
    @Test
    public void testScanClasspath(){
        var context = Context.make(CONFIGS);
        
        //Or read configuration from a properties file
        //var context = Context.make("/tmp/project/demo/config.properties");
        
        var loader = context.getLoader();
        loader.scan("/tmp/project/demo/bin");
        
        //If you have more than one classpath:
        //loader.scan("Another classpath");
        //loader.scan("classpath-1", "class-path2");
        
        context.start();
        
        //The chain-stype calling is supported:
        //context.getLoader().scan("/tmp/project/demo/bin").start();
        
        var person = context.get(Person.class);
        var mobile = context.get(Mobile.class);
        TestCase.assertEquals("John Denver", person.getName());
        TestCase.assertEquals(18, person.getAge());
        TestCase.assertEquals("+86", mobile.getAreaCode());
        TestCase.assertEquals("13603166666", person.getMobile().getNumber());
    }
}

B. Register managed objects manually:

    @Test
    public void testRegisterManually(){
        var context = Context.make(CONFIGS);
        var binder = context.getBinder();
        binder.register(Person.class);
        binder.register(Mobile.class);
        context.start();
        
        //Chain-style calling:
        //binder.register(Person.class, Mobile.class).start();
        
        var person = context.get(Person.class);
        var mobile = context.get(Mobile.class);
        TestCase.assertEquals(18, person.getAge());
        TestCase.assertEquals("John Denver", person.getName());
        TestCase.assertEquals("+86", mobile.getAreaCode());
        TestCase.assertEquals("13603166666", person.getMobile().getNumber());
    }

C. Load classes and dependencies from a given JAR file:

We assume to packed these 2 classes into a JAR file "/tmp/project/demo/lib/demo.jar"

    @Test
    public void testLoadFromJAR(){
    	var context = Context.make(CONFIGS);
    	var loader = context.getLoader();
    	loader.load("/tmp/project/demo/lib/demo.jar");
    	context.start();
       
   	var person = context.get(Person.class);
    	var mobile = context.get(Mobile.class);
    	TestCase.assertEquals(18, person.getAge());
     	TestCase.assertEquals("John Denver", person.getName());
    	TestCase.assertEquals("+86", mobile.getAreaCode());
    	TestCase.assertEquals("13603166666", person.getMobile().getNumber());
    }

D. Parse the XML Definition (beans.xml)

If you are a Spring Framework developer, you will be very familiar with XML configuration. Whale also supports you defining the manged objects in the XML file located in "/tmp/project/demo/beans.xml":

<beans>
	<bean id="person" singleton="true" type="whale.demo.Person">
    	<props>
			<prop name="id" val="45" />
			<prop name="name" key="user.name" />
    		<prop name="mobile" ref="mobile" />
    	</props>
        <methods>
        	<method name="setAge">
            	<arg key="user.age" type="int" />
            </method>
        </methods>
    </bean>
    <bean id="mobile" singleton="true" type="whale.demo.Mobile">
        <args>
	    	<arg key="mobile.area" type="String" />
	    	<arg key="mobile.number" type="String" />
	    </args>
	</bean>
</beans> 

Please note that XML definition just supports field injection(using the props tag), constructor injection(using the args tag) and method injection(using the methods tag). For the constructor and method injection, you must explicitily declare the parameter types. Othewise, whale may not be able to correctly identify overloaded methods in certain situations. For example:

//In constructor:
public Student(String studentNumber);
public Student(int age);
//How do we correctly explain the value "21" from configuration? 

//In general methods:
public void setScore(int age);
public void setScore(float age);
//We can convert the value "85" to 85(int) or 85.0(float). Which method will be invoked? 

More advanced features are forbidden because it makes the XML schema very ugly.

    @Test
    public void testParseXMLDefinition(){
    	var context = Context.make(CONFIGS);
     	var loader = context.getLoader();
      	loader.parse("/tmp/project/demo/beans.xml");
      	context.start();
       	
       	//Chain-stype calling
       	//context.createFactory().parse("/tmp/project/demo/beans.xml").start();
       	
       	var person = context.get(Person.class);
        var mobile = context.get(Mobile.class);
        
        TestCase.assertEquals(18, person.getAge());
        TestCase.assertEquals("John Denver", person.getName());
        TestCase.assertEquals("+86", mobile.getAreaCode());
        TestCase.assertEquals("13603166666", person.getMobile().getNumber());
    }

You can actually pass multiple XML definitions to the method parse. For example:

    loader.parse("/tmp/project/demo/beans-1.xml", "/tmp/project/demo/beans-2.xml");

4. Provider

The Provider interface is similar to the ObjectFactory in Spring Framework. One of its primary benefits is resolving the circular dependent. For example:

// In class Person:
@Inject
private Mobile mobie;

//In class Mobile
@Inject
private Person owner;

Whale cannot assemble the above 2 objects and throws an exception "Circular dependent is detected". We can refactor it using Provider interface as following:

// In class Person:
@Inject
private Provider<Mobile> mobile;

//In class Mobile
@Inject
private Provider<Person> owner;

//In test case:
var name = mobile.getOwner().get().getName();
var code = person.getMobile().get().getAreaCode();

Now, it works correctly. Please note that you should avoid calling the method Provider.get() directly within constructor or other method injection, Doing so can lead to unexpected behavior:

private Mobile mobile;

@Inject
public Person(Provider<Mobile> mobile){
	this.mobile = mobile.get(); //Here
}

A correct approach is as below:

private Provider<Mobile> mobile;

@Inject
public Person(Provider<Mobile> mobile){
	this.mobile = mobile;
}

More information about Provider please refer to the document on github.

5. Advanced Features

The section describes some advanced features in whale. It helps developers more flexibilities.

A. Bind Annotation

Please consider the following example code:

package whale.demo.service;

public interface DemoService{
	public Object doSomething(Object args);
}

package whale.demo.service;
@Singleton
public class DemoServiceImpl implements DemoService{
	public Object doSomething(Object args){
		Object result = handle_your_business();
		return result;
	}
}

Since the interface DemoService can not be instantiated directly, we must register the implementation class DemoServiceImpl as a managed object into DI container.

//The first approach:
public class Demo{
	@Inject
	@Named("whale.demo.service.DemoServiceImpl")
	private DemoService service;
}

//The second approach:
public class Demo{
	@Inject
	private DemoServiceImpl service;
}

The first approach ontlined above is overly verbose, and the second approach deviates from the principles of Interface-Oriented programming. Bind annotation offers a more concise and elegant way for developer, as demonstrated below:

package whale.demo.service;

@Bind(target=DemoServiceImpl.class)
public interface DemoService{
	public Object doSomething(Object args);
}

Or, 

@Singleton
@Bind(value=Demoservice.class, target=DemoServiceImpl.class)
public class DemoServiceImpl implements DemoService{
	public Object doSomething(Object args){
		Object result = handle_your_business();
		return result;
	}
}

public class Demo{
	@Inject
	private DemoService service;
}

Certainly, you can call the bind method manually in code:

binder.bind(DemoService.class, DemoServiceImpl.class);

To summarize, the Bind annotation provides a straightforward way for mapping an abstraction (interface or abstract class) to its concrete implementation, simplifying the dependence configuration.

B. Append Managed Object

Whale offers the flexibility to append managed objects into DI container even after the container has been initialized.

 @Test
    public void testAppendBeans(){
        var context = Context.make(CONFIGS);
        var binder = context.getBinder();
        binder.register(Person.class);
        binder.register(Mobile.class);
        context.start(); //Container Initialized
        
      	binder.append(DemoServiceImpl.class);
        
        var person = context.get(Person.class);
        var mobile = context.get(Mobile.class);
        TestCase.assertEquals(18, person.getAge());
        TestCase.assertEquals("John Denver", person.getName());
        TestCase.assertEquals("+86", mobile.getAreaCode());
        TestCase.assertEquals("13603166666", person.getMobile().getNumber());
    }

If the append() method is invoked before the start() method, whale will disregard the call.

C. Customized Qualifier Annotation

As mentioned earlier, the Qualifier is a meta-annotation, so it cannot be used directly. Developers can create custom annotations that extend from it. Let's illustrate this with an example:

//User defines two qualifier annotations
//The first
@Qualifier
@Documented
@Retention(RUNTIME)
public @interface Cat {
}

//The second
@Qualifier
@Documented
@Retention(RUNTIME)
public @interface Dog {
}

//Declare an interface
public interface Animal{
	public String howl();
}

//There are 2 implements of above interface: Cat and Dog
@Cat
public class Cat implements Animal{
	public String howl(){
		return "Miao Miao";
	}
}

@Dog
public class Dog implements Animal{
	public String howl(){
		return "Wang Wang";
	}
}

// The class Family dependents on the implementation Cat and Dog
public class Family{
	private Person father;
	private Person mother;
    private Person child;
	
    @Inject
    @Cat
    private Animal cat;
    
    @Inject
    @Dog
    private Animal dog;
}

Whale will seamlessly inject the correct implementation into the cat and dog properties, eliminating the need for verbose configuration. While the Bind annotation associates an implementation with an interface, it's not well-suited for scenarios with multiple implementations. Custom Qualifiers provide a more effective way to specify and inject different implementations, offering a more elegant solution.

D. Initializer and Finalizer

If you mark a method with the annotation @Ready, the method will be invoked after finishing assemble. The initializer is executed later than the constructor because it MUST wait all injections are finished (especially methods injection).

@Ready
public void init(){
    //do something here    
}

Whale does not provide the finalizer annotation. You should implement the AutoCloseable interface. When the DI container is shutdown, the close method will be called automatically.

E. Import external singleton object(Non-JSR330) as a managed bean into container:

public void testIncludeObject(){
    var context = Context.make();
    var binder = context.getBinder();
    binder.include(new Object());
    binder.include(new Object(), "myObject");
    context.start();
    TestCase.assertEequals(true, context.get("myObject") != null);
    TestCase.assertEquaqls(true, context.get(Object.class) != null);
    TestCase.assertEquals(false, context.get("myObject") == context.get(Object.class));
}

6. Web Application

The WebListener class enables the intergation of whale into a web application. Please add a listener declaration(using the listener tag) in web.xml file:

<listener>
	<listener-class>cn.techarts.whale.web.WebListener</listener-class>
</listener>

Now, you can retrieve the managed objects from the servlet context, as demonstrated below:

public DemoServlet extends HttpServlet{
	public void init(ServletConfig arg) {
		var context = Context.from(arg.getServletContext());
		DemoService service = context.get(DemoSrevice.class);	
		service.doSomething(context.get(Person.class));
	}
}

7. Interceptor

The interceptor in whale is based on JDK dynamic proxy. There are 2 annotations(Advice, Advise) and an interface(Advisor), it's limited but very easy to use. The following example describes the usage.

public class LogAdvice implements Advisor {
    @Override
    // args: The parameters of the intercepted method
    // result: The return value of the intercepted method
    // threw: The potential exception threw by the method
    public Object advise(Object[] args, Object result, Throwable threw) {
    	System.out.println("Intercepted.");
        return null;
    }
}

public class ResultAdvice implements Advisor{
    @Override
    public Object advise(Object[] args, Object result, Throwable threw) {
        var tmp = (Integer)result;
        return tmp + 100;
    }	
}
@Bind(target=SomeInterfaceImpl.class)
public interface SomeInterface {
    
    // You can intercept a method at 4 places: 
    // 1. before the first statement(before: Initialize), 
    // 2. after the return statement(after: Change the result), 
    // 3. an exception threw(threw: Handle the exception) and
    // 4. in the finally block(last: Cleanup the resources).
    
    @Advise(before=LogAdvice.class, after=ResultAdvice.class)
    public int getValue();
}

@Singleton
@Advice(SomeInterface.class)
public class SomeInterfaceImpl implements SomeInterface {
    private int val;
    @Inject
    public SomeInterfaceImpl(@Valued(val="33")int value) {
        this.val = value;
    }
	
    @Override
    public int getValue() {
    	return this.val;
    }
}

8. Todo List

We plan to add the following features:

  • Support namespace or module
  • Refactor code to improve performance.
  • Fix bugs as soon as they are found.