VOOZH about

URL: https://www.javacodegeeks.com/2015/05/akka-samples-with-scala-and-spring.html

⇱ Akka samples with scala and Spring - Java Code Geeks


I was looking around recently for Akka samples with Spring and found a starter project which appeared to fit the bill well. The project however utilizes Spring-Scala which is an excellent project, but is no longer maintained. So I wanted to update the sample to use core Spring java libraries instead. So here is an attempt on a fork of this starter project with core Spring instead of Spring-scala. The code is available here.

The project utilizes Akka extensions to hook in Spring based dependency injection into Akka.

Here is what the extension looks like:

package sample

import akka.actor.{ActorSystem, Props, Extension}
import org.springframework.context.ApplicationContext
/**
 * The Extension implementation.
 */
class SpringExtension extends Extension {
 var applicationContext: ApplicationContext = _

 /**
 * Used to initialize the Spring application context for the extension.
 * @param applicationContext
 */
 def initialize(applicationContext: ApplicationContext) = {
 this.applicationContext = applicationContext
 this
 }

 /**
 * Create a Props for the specified actorBeanName using the
 * SpringActorProducer class.
 *
 * @param actorBeanName The name of the actor bean to create Props for
 * @return a Props that will create the named actor bean using Spring
 */
 def props(actorBeanName: String): Props =
 Props(classOf[SpringActorProducer], applicationContext, actorBeanName)

}

object SpringExtension {
 def apply(system : ActorSystem )(implicit ctx: ApplicationContext) : SpringExtension = SpringExt(system).initialize(ctx)
}

So the extension wraps around a Spring application context. The extensions provides a props method which returns an Akka Props configuration object which uses the application context and the name which the actor is registered with Spring to return an instance of the Actor. The following is the SpringActorProducer:

package sample

import akka.actor.{Actor, IndirectActorProducer}
import org.springframework.context.ApplicationContext


class SpringActorProducer(ctx: ApplicationContext, actorBeanName: String) extends IndirectActorProducer {

 override def produce: Actor = ctx.getBean(actorBeanName, classOf[Actor])

 override def actorClass: Class[_ <: Actor] =
 ctx.getType(actorBeanName).asInstanceOf[Class[Actor]]

}

Given this base code, how does Spring find the actors, I have used scanning annotations to annotate the actors this way:

package sample.actor

import akka.actor.Actor
import sample.service.CountingService
import sample.SpringExtension._
import org.springframework.stereotype.Component
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.context.annotation.Scope
import akka.actor.ActorRef
import sample.SpringExtension
import org.springframework.context.ApplicationContext

@Component("countingCoordinatingActor")
@Scope("prototype")
class CountingCoordinating @Autowired() (implicit ctx: ApplicationContext) extends Actor {

 import sample.messages._

 var counter: Option[ActorRef] = None

 
 def receive = {
 case COUNT => countingActor() ! COUNT
 case g:GET => countingActor() ! g
 }
 
 private def countingActor(): ActorRef = {
 if (counter.isEmpty) {
 val countingActorProp = SpringExtension(context.system).props("countingActor")
 counter = Some(context.actorOf(countingActorProp, "counter"))
 } 
 
 counter.get
 }
 
}


@Component("countingActor")
@Scope("prototype")
class CountingActor @Autowired()(countingService: CountingService) extends Actor {

 import sample.messages._

 private var count = 0

 def receive = {
 case COUNT => count = countingService.increment(count)
 case GET(requester: ActorRef) => requester ! RESULT(count)
 }
 
}

The CountingService is a simple service that gets injected in by Spring. The following is the main Spring Application configuration where all the wiring takes place:

import akka.actor.ActorSystem
import org.springframework.context.ApplicationContext
import org.springframework.context.annotation.Configuration
import org.springframework.context.annotation.Bean
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.context.annotation.ComponentScan

@Configuration
@ComponentScan(Array("sample.service", "sample.actor"))
class AppConfiguration {

 @Autowired
 implicit var ctx: ApplicationContext = _;
 
 /**
 * Actor system singleton for this application.
 */
 @Bean
 def actorSystem() = {
 val system = ActorSystem("AkkaScalaSpring")
 // initialize the application context in the Akka Spring Extension
 SpringExt(system)
 system 
 }
}

To make use of this entire set-up in a sample program:

import akka.actor.{ActorRef, ActorSystem}
import sample.SpringExtension._
import scala.concurrent.duration._
import scala.concurrent._
import scala.util._
import sample.messages._
import org.springframework.context.annotation.AnnotationConfigApplicationContext
import akka.actor.Inbox


object Main extends App {
 // create a spring context
 implicit val ctx = new AnnotationConfigApplicationContext(classOf[AppConfiguration])

 import Config._

 // get hold of the actor system
 val system = ctx.getBean(classOf[ActorSystem])

 val inbox = Inbox.create(system)
 
 val prop = SpringExtension(system).props("countingCoordinatingActor")

 // use the Spring Extension to create props for a named actor bean
 val countingCoordinator = system.actorOf(prop, "counter")

 // tell it to count three times
 inbox.send(countingCoordinator, COUNT)
 inbox.send(countingCoordinator, COUNT)
 inbox.send(countingCoordinator, COUNT)
 
 inbox.send(countingCoordinator, GET(inbox.getRef()))
 
 val RESULT(count) = inbox.receive(5.seconds)

 println(s"Got $count")
 system.shutdown
 system.awaitTermination
}

References:

Reference: Akka samples with scala and Spring from our JCG partner Biju Kunjummen at the all and sundry blog.
Do you want to know how to develop your skillset to become a Java Rockstar?
Subscribe to our newsletter to start Rocking right now!
To get you started we give you our best selling eBooks for FREE!
1. JPA Mini Book
2. JVM Troubleshooting Guide
3. JUnit Tutorial for Unit Testing
4. Java Annotations Tutorial
5. Java Interview Questions
6. Spring Interview Questions
7. Android UI Design
and many more ....
I agree to the Terms and Privacy Policy

Thank you!

We will contact you soon.

πŸ‘ Photo of Biju Kunjummen
Biju Kunjummen
May 26th, 2015Last Updated: May 25th, 2015
3 114 3 minutes read
Subscribe

This site uses Akismet to reduce spam. Learn how your comment data is processed.

3 Comments
Oldest
Newest Most Voted
10 years ago

ِThanks for comprehensive tutorial. I think in CountingCoordinating class in better that make the countingActor function to lazy val and then remove the counter variable

0
Reply
10 years ago

What is the SpringExt that used in line 33 of SpringExtension?

0
Reply
10 years ago

I think you should change your SpringExtension object to below code :

@Component
class SpringExtension(applicationContext: ApplicationContext) extends Extension {

def props(actorBeanName: String): Props =
Props(classOf[SpringActorProducer], applicationContext, actorBeanName)

}

object SpringExtension {
def apply(system : ActorSystem )(implicit ctx: ApplicationContext) : SpringExtension = new SpringExtension(ctx)
}

0
Reply
Back to top button
Close
wpDiscuz