Skip to content

Latest commit

 

History

History
328 lines (256 loc) · 11.7 KB

README.md

File metadata and controls

328 lines (256 loc) · 11.7 KB

BananaRama

alt text

Overview

BananaRama provides a simple and extensible CRUD framework to access the persistency layer in a modular and scalable way. It aims to provide simple APIs to query entities without the client really knowing where those entities are stored.

This is an example of how the client code looks when entities are managed using BananaRama.

public class Crud {
    
    public void test() {

	BananaRama bananarama = new BananaRama();

        //Create entry on persistency layer
        Entry entry = new Entry("testKey","testVal");
        bananarama.create(Entry.class).from(Stream.of(entry));
	
        //Read entry
        Entry entry = bananarama.read(Entry.class).all().findAny().get();        

        //Update for previous entry
        Entry entryUpdated = new Entry("testKey","testValUpdated");       
        bananarama.update(Entry.class).from(Stream.of(entryUpdated));
        
        //Delete entry
        bananarama.delete(Entry.class).from(Stream.of(entryUpdated));
        
    }
}

Built-In persistency layer support

BananaRama natively supports the following frameworks/persistency layers

  • [JDBC] (#jdbc) - fast POJOs to SQL and viceversa.
  • JPA - Manage JPA entities with BananaRama.
  • CQEngine - Integrated support for CQEngine indexed collection on top of other persistency layers or standalone.
  • Magic - Easy to use implementation of the DTO pattern.

Simplest Example

Integration of entities in BananaRama is done in two steps. First we need to create a class which implements the Adapter interface. This will be the class that will actually manage the entities, either by accessing the persistency layer or calling other adapters.

public class NoOpAdapter implements Adapter<Object>{
    //A really simple adapter which does nothing :)
    @Override
    public <T> CreateOperation<T> create(Class<T> clazz) {
      return new NoOpCreateOperation();
    }
    
    @Override
    public <T> ReadOperation<T> read(Class<T> clazz) {
        return new NoOpReadOperation<>();
    }
    
    @Override
    public <T> UpdateOperation<T> update(Class<T> clazz) {
      return new NoOpUpdateOperation<>();
    }
    
    @Override
    public <T> DeleteOperation<T> delete(Class<T> clazz) {
        return new NoOpDeleteOperation<>();
    }
    
}

Then we create the class that reflects the data on the persistency layer

@Banana(adapter = NoOpAdapter.class)
public class Entry {
    private String id;
    
    public String getId(){
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }
    
    @Override
    public boolean equals(Object obj) {
        if(obj instanceof Entry)
            return  id.equals(((Entry)obj).id);
       return false;
    }

    @Override
    public int hashCode() {
       return id.hashCode();
    }
    
    
}

The @Banana annotation takes care of redirecting BananaRama to the class in charge of CRUD operations for the Entry type. In this example all CRUD operations will then be delegated to the NoOpAdapter. The basic idea is that the client code contains no references to the adapter, therefore the persistency layer can be swapped but the client code won't change.

Adapter dependencies

It is possible to chain adapters in case some entities depend on other entities. A list of adapter types can be provided using the @BananaRamaAdapter annotation: BananaRama will load the required adapters beforehand and will pass it to the annotated adapter's constructor so that they can be used to do CRUD operations on child entities. See the following example

@BananaRamaAdapter(requires = {SubAdapter.class,BananaRama.class})
public class MyAdapter extends NoOpAdapter{
    private BananaRama parent;
    private SubAdapter sub;

    public MyAdapter(SubAdapter sub, BananaRama parent){//Order of args
        this.parent = parent;                           //is the same in the annotation
        this.sub = sub;
    }
}

Note that since the BananaRama class itself implements an adapter (i.e the root adapter) it can also be passed as a requirement in order to cascade CRUD operations on child entities without knowing the specific adapter for those entities. All adapters are stored in the instance of BananaRama that was used to instantiate them.

Explicit usage of adapters

Although you are discouraged to do so, BananaRama also allows to use specific adapters for a specific entity with the using method

 //will return Optional<Entry> retrieved by an instance of MyAdapter.class
 bananarama.using(MyAdapter.class).read(Entry.class).all().findFirst();

Thread-safety

As a general rule, take into account the following table for thread-safety of BananaRama's components

Component Thread-Safe
Adapter
Any CRUD operation

Built-In Adapters

JDBC

The Jdbc adapter is designed to do fast and simple CRUD operations using JDBC in conjunction with any driver you want to use. So far it has been tested successfully on H2 and PostgreSQL drivers.

First of all implement your adapter which extends the SQLAdapter class

public class MySqlAdapter extends SqlAdapter {

    public MySqlAdapter() {
        super(DataSourceFactory.getDataSource());
    }

}

Then you can automatically map database tables to classes by annotating them

@Banana(adapter = MySqlAdapter.class)
@Table(name = "pojo")
@Convert(type = LocalDateTime.class,with = LocalDateTimeConverter.class)
public class Pojo {

    @Id 
    private int id;

    @Column(name = "name")
    private String laBel;
    
    private Double xyz;
    
    @ConvertWith(MapConverter.class)
    private Map<String,String> attrs;
    
    public int getId() {
        return id;
    }

    public void setId(int idno) {
        this.id = idno;
    }

    public String getLaBel() {
        return laBel;
    }

    public void setLaBel(String laBel) {
        this.laBel = laBel;
    }

    public Double getXyz() {
        return xyz;
    }

    public void setXyz(Double xyz) {
        this.xyz = xyz;
    }

    public Map<String, String> getAttrs() {
        return attrs;
    }

    public void setAttrs(Map<String, String> attrs) {
        this.attrs = attrs;
    }
}

All annotation in the previous example are part of BananaRama. Here is a quick overview of them:

  • @Id identifies key fields. It may be used on multiple fields, but use it carefully: if fields a and b of type A and B define the object's identity, then the List passed to the fromKeys method should contain alternating values of type A and B.
  • @Column and @Table work very similarly to the the equivalent JPA annotations, with some extra features explained in the javadoc.
  • @Convert and @ConvertWith provide the adapter with custom SQL types translators. The first is for defining global translators for the annotated class, the latter is for setting a translator for a specific field and overrides any global translator.

CqEngine

BananaRama also supports a simple caching mechanism which relies on the awesome CQEngine. Entities can be loaded on an IndexedCollection, as an intermediate layer, by using the BufferedOnIndexedCollection annotation as follows.

@BufferedOnIndexedCollection
@Banana(adapter = FooAdapter.class)
public class CacheEntry extends Entry{
    private final String key;
    private String val;
    
    public String getValue(){
        return val;
    }

    public String getKey(){
        return key;
    }

    public String getVal() {
        return val;
    }
    
    public void setVal(String val) {
        this.val = val;
    }

    @Indexed
    public static final Attribute<CacheEntry,String> KEY = new SimpleAttribute<CacheEntry, String>("key"){
        @Override
        public String getValue(CacheEntry o, QueryOptions qo) {
            return o.getKey();
        }
    };
    
    @Indexed(NavigableIndexProvider.class)
    public static final Attribute<CacheEntry,String> VAL = new SimpleAttribute<CacheEntry, String>("val"){
        @Override
        public String getValue(CacheEntry o, QueryOptions qo) {
            return o.getVal();
        }
    };
}

The key annotations in the previous example are

  • @BufferedOnIndexedCollection, which is used to notify BananaRama that all entities of the annotated type have to be loaded on a IndexedCollection upon the first reading operation. All modifications (deletion, update, creation) will be performed on both the collection and the underlying layer using the provided adapter. By default all entities are stored in a ConcurrentIndexedCollection but a custom colllection provider may be provided. BananaRama has the following providers for IndexedCollection implementaions:

    • ConcurrentIndexedCollection
    • ObjectLockingIndexedCollection
  • Indexed, which tells BananaRama that an index should be created on the annotated CQEngine attribute (see CQEngine documentation for details regarding attributes and indexes). By default a HashIndex will be created, but a custom IndexProvider may be supplied in order to use any other available index. BananaRama has default providers for

    • HashIndex
    • NavigableIndex
    • RadixTreeIndex
    • DiskIndex

Magic (DTO to Object and viceversa)

Programmers dealing with serialized entities might be interested in using the DTO pattern. BananaRama provides an automatic translation mechanism in order to go from DTO to Object and vice-versa. The Magic adapter is what implements such translation mechanism. As a simple example, let's assume we have the following DTO, which is managed by the FooAdapter adapter.

@Banana(adapter = FooAdapter.class)
public class SimpleDto {
    private int id;
    private String date;
    
    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getDate() {
        return date;
    }

    public void setDate(String date) {
        this.date = date;
    }
    
}

and the complete object, with a LocalDate field which we want to obtain by parsing the date field of the DTO

@MapWith(SimpleMapper.class)
@Banana(adapter = MagicAdapter.class)
public class SimpleObj {
    private int id;
    private LocalDate date;
    
    public int getId() {
        return id;
    }
    
    public void setId(int id) {
        this.id = id;
    }
    
    public LocalDate getDate() {
        return date;
    }
    
    public void setDate(LocalDate date) {
        this.date = date;
    }

}

The @MapWith annotation tells the MagicAdapter that this object doesn't actually exists in the persistency layer, but it is an abstraction of its DTO counterpart. As a value accepts a class which implements the ObjToDto<O,D> interface. This class is then instantiated and used by the MagicAdapter to convert objects in DTOs and viceversa, and also to retrieve the DTO class in order to access the persistency layer. As such, this annotation is a requirement when using the MagicAdapter. A library which provides a very powerful set of tools when dealing with the DTO pattern is MapStruct.

JPA

Implement a custom adapter in order to access a specific persistency unit.

public class MyJpaAdapter extends AbstractJpaAdapter{
    
    public MyJpaAdapter() {
        super("my_awesome_unit");//must match a persistency unit name in persistence.xml
    }

}

Then simply annotate with @Banana( adapter = MyJpaAdapter.class) any class annotated with @Entity and other annotations which are part of JPA and use it with BananaRama.