Hibernate Distirbuted cache


Hazelcast, Infinispan, etc.


Key Terms



Perfomance - number of operations per unit of time (requests/second)

Perfomance is not Scalability

Scalability can be Vertical and Horizontal

Happy App


more happy App?


Solution?


Cache concept


An area of local memory that holds a copy of 
frequently accessed data that is otherwise 
expensive to get or compute

Cache types:
  • Application cache
  • Second level (L2) cache
  • Hybrid cache

Application cache


l2 cache


Hybrid cache


key cache terms



  • Cache size - defines how many elements a cache can hold;
  • Cache eviction - algorithm defines what to do when the number of elements in cache exceeds the size (LRU, LFU, FIFO) & eviction percentage;
  • Time-to-live - defines time after that a cache key should be remove from the cache (expired);

REPLICATED CACHE VS distributed cache 



Peer-to-peer cluster



Clients and server cluster


Hybrid cluster


frameworks overview



*IMDG - In-Memory Data Grid


Scalable data grid platform
  • open-source - LGPL
  • based on some JBoss Cache code

JBoss Cache
  • clustered caching library

Infinispan has a Map-like API(JSR-107 JCACHE)
  • can be used as Key-Value No-SQL solution
  • has Hibernate4 cache region&provider implementation

POm.xml


         <!--Infinispan-->
        <dependency>
            <groupId>org.infinispan</groupId>
            <artifactId>infinispan-core</artifactId>
            <version>6.0.2.Final</version>
        </dependency>
        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-infinispan</artifactId>
            <version>${hibernate.version}</version>
        </dependency>
        <dependency>
            <groupId>org.jboss.jbossts</groupId>
            <artifactId>jbossjta</artifactId>
            <version>4.16.4.Final</version>
        </dependency>

Infinispan.xml

<infinispan xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="urn:infinispan:config:6.0 http://www.infinispan.org/schemas/infinispan-config-6.0.xsd"
    xmlns="urn:infinispan:config:6.0">
    <global>
        <transport nodeName="caching" clusterName="caching_cluster" transportClass="org.infinispan.remoting.transport.jgroups.JGroupsTransport">
            <properties> <property name="configurationFile" value="tcp.xml"/>
            </properties>
        </transport>
    </global>
    <default>
        <clustering mode="dist"><async/>
        </clustering>
        <eviction maxEntries="1000" strategy="LRU"/>
    </default>
    <namedCache name="productBrand.region">
        <clustering mode="dist"/>
        <eviction strategy="LRU" maxEntries="5000" threadPolicy="DEFAULT"/>
        <expiration maxIdle="10"/>
    </namedCache>
</infinispan>

datasourceContext-infinispan.xml


cached entity

@Entity
@Table(name = "PRODUCT_CATEGORY")
@Cache(region = "productCategory.region", usage = CacheConcurrencyStrategy.TRANSACTIONAL)
public class ProductCategory {

    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY)
    @Column(name = "CATEGORY_ID")
    private Long id;

    @Column(name = "NAME")
    private String name;

    @Cache(region = "productCategory.brands.region", usage = CacheConcurrencyStrategy.TRANSACTIONAL)
    @OneToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY)
    @JoinTable(name = "BRAND_CATEGORY", joinColumns = @JoinColumn(name = "CATEGORY_ID"), inverseJoinColumns = @JoinColumn(name = "BRAND_ID"))
    private List<ProductBrand> brands;
...
}

application cache


Cache<Object, Object> cache = new DefaultCacheManager().getCache();// Add a entry
cache.put("key", "value");
// Remove the entry from the cache
Object v = cache.remove("key");
// And replace it if missing
cache.putIfAbsent("key", "newValue");
cache.clear();
//By default entries are immortal but we can override this on a per-key basis and provide lifespans.
cache.put("key", "value", 5, SECONDS);


Apache license

1 solid Jar file ~3MB + HibernateProvider

Transactional & Secure

IMDG

3-way configuration: XML, Java, Spring

pom.xml


  <dependency>
       <groupId>com.hazelcast</groupId>
       <artifactId>hazelcast</artifactId>
       <version>3.2.1</version>
  </dependency>
  <dependency>
       <groupId>com.hazelcast</groupId>
       <artifactId>hazelcast-hibernate4</artifactId>
       <version>3.2.1</version>
   </dependency>
   <dependency>
       <groupId>com.hazelcast</groupId>
       <artifactId>hazelcast-spring</artifactId>
       <version>3.2.1</version>
   </dependency>

Hazelcast.xml

<hazelcast>
    <group name="dev" password="dev"/>
    <network>
        <port auto-increment="true">5701</port>
        <outbound-ports><ports>0</ports></outbound-ports>
        <join>
            <multicast enabled="false">
                <multicast-group>231.12.21.135</multicast-group>
                <multicast-port>45564</multicast-port>
                <multicast-timeout-seconds>2</multicast-timeout-seconds>
            </multicast>
            <tcp-ip enabled="true"><interface>127.0.0.1</interface></tcp-ip>
        </join>
        <interfaces enabled="true"><interface>127.0.0.*</interface></interfaces>
        <ssl enabled="false"/>
        <socket-interceptor enabled="false"/>
        <symmetric-encryption enabled="false"/>
    </network>
    <partition-group enabled="false"/>
    <executor-service name="default">
        <pool-size>16</pool-size>
        <queue-capacity>0</queue-capacity>
    </executor-service>

HAZELCAST.XML


   <map name="default"/>

    <map name="productCategory.brands.region">
        <in-memory-format>OBJECT</in-memory-format>
        <backup-count>2</backup-count>
        <async-backup-count>0</async-backup-count>
        <time-to-live-seconds>90</time-to-live-seconds>
        <max-idle-seconds>0</max-idle-seconds>
        <eviction-policy>LRU</eviction-policy>
        <max-size policy="PER_NODE">5000</max-size>
        <eviction-percentage>30</eviction-percentage>
        <merge-policy>com.hazelcast.map.merge.PassThroughMergePolicy</merge-policy>
        <read-backup-data>true</read-backup-data>
    </map>
</hazelcast>

DATASOURCECONTEXT-hazelcast.XML



  <!-- SessionFactory bean with Hazelcast Hibernate properties -->
<prop key="hibernate.cache.region.factory_class">com.hazelcast.hibernate.HazelcastCacheRegionFactory</prop>
<prop key="hibernate.cache.hazelcast.configuration_file_path">hazelcast.xml</prop>
<prop key="hibernate.cache.hazelcast.use_lite_member">true</prop>
<prop key="hibernate.cache.hazelcast.shutdown_on_session_factory_close">false</prop>

Cached Entity


 
@Entity
@Table(name = "PRODUCT_CATEGORY")
@Cache(region = "productCategory.region", usage = CacheConcurrencyStrategy.READ_WRITE)
public class ProductCategory {

	...
    @Cache(region = "productCategory.brands.region", usage = CacheConcurrencyStrategy.READ_WRITE)
    @OneToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY)
    @JoinTable(name = "BRAND_CATEGORY", joinColumns = @JoinColumn(name = "CATEGORY_ID"), inverseJoinColumns = @JoinColumn(name = "BRAND_ID"))
    private List<ProductBrand> brands;
	...
}

Application cache




Config cfg = new Config();
HazelcastInstance instance = Hazelcast.newHazelcastInstance(cfg);
Map<Integer, String> mapCustomers = instance.getMap("customers");
mapCustomers.put(1, "Joe");
mapCustomers.put(2, "Ali");
mapCustomers.put(3, "Avi");

conclusion


Hazelcast wins!


REferences

  • http://docs.oracle.com/cd/E18686_01/coh.37/e18677/cache_intro.htm
  • http://coherence.oracle.com/display/COH31UG/Read-Through,+Write- Through,+Refresh-Ahead+and+Write-Behind+Caching
  • http://blog.tekmindsolutions.com/oracle-coherence-diffrence-between- replicated-cache-vs-partitioneddistributed-cache/
  • http://www.slideshare.net/MaxAlexejev/from-distributed-caches-to- inmemory-data-grids
  • http://www.slideshare.net/jaxlondon2012/clustering-your-application- with-hazelcast
  • http://www.gridgain.com/blog/fyi/cache-data-grid-database/
  • http://gridgaintech.wordpress.com/2013/10/19/distributed-caching-is- dead-long-live/
  • http://www.hazelcast.com/resources/the-book-of-hazelcast/
  • https://labs.consol.de/java-caches/part-3-3-peer-to-peer-with-hazelcast/

hibernate-dist-cache

By Yegor Bondar

hibernate-dist-cache

  • 1,609