Skip to content

Instantly share code, notes, and snippets.

@patriknw
Created November 21, 2012 14:20
Show Gist options
  • Save patriknw/4125062 to your computer and use it in GitHub Desktop.
Save patriknw/4125062 to your computer and use it in GitHub Desktop.
PeekMailbox example
package example
import java.util.concurrent.ConcurrentHashMap
import java.util.concurrent.ConcurrentLinkedQueue
import com.typesafe.config.Config
import com.typesafe.config.ConfigFactory
import akka.actor.Actor
import akka.actor.ActorContext
import akka.actor.ActorRef
import akka.actor.ActorSystem
import akka.actor.ExtendedActorSystem
import akka.actor.Extension
import akka.actor.ExtensionId
import akka.actor.ExtensionIdProvider
import akka.actor.Props
import akka.actor.PoisonPill
import akka.dispatch.Envelope
import akka.dispatch.MailboxType
import akka.dispatch.MessageQueue
import akka.dispatch.QueueBasedMessageQueue
import akka.dispatch.UnboundedMessageQueueSemantics
object PeekMailboxExtension extends ExtensionId[PeekMailboxExtension] with ExtensionIdProvider {
def lookup = this
def createExtension(s: ExtendedActorSystem) = new PeekMailboxExtension(s)
def ack()(implicit context: ActorContext): Unit = {
PeekMailboxExtension(context.system).ack()
}
}
class PeekMailboxExtension(val system: ExtendedActorSystem) extends Extension {
private val mailboxes = new ConcurrentHashMap[ActorRef, PeekMailbox]
def register(actorRef: ActorRef, mailbox: PeekMailbox): Unit =
mailboxes.put(actorRef, mailbox)
def unregister(actorRef: ActorRef): Unit = mailboxes.remove(actorRef)
def ack()(implicit context: ActorContext): Unit = {
mailboxes.get(context.self) match {
case null ⇒ throw new IllegalArgumentException("Mailbox not registered for: " + context.self)
case mailbox ⇒ mailbox.ack()
}
}
}
/**
* configure the mailbox via dispatcher configuration:
* {{{
* peek-dispatcher {
* mailbox-type = "example.PeekMailboxType"
* }
* }}}
*/
class PeekMailboxType(settings: ActorSystem.Settings, config: Config) extends MailboxType {
override def create(owner: Option[ActorRef], system: Option[ActorSystem]) = (owner, system) match {
case (Some(o), Some(s)) ⇒
val mailbox = new PeekMailbox(o, s)
PeekMailboxExtension(s).register(o, mailbox)
mailbox
case _ ⇒ throw new Exception("no mailbox owner or system given")
}
}
class PeekMailbox(owner: ActorRef, system: ActorSystem) extends QueueBasedMessageQueue with UnboundedMessageQueueSemantics {
final val queue = new ConcurrentLinkedQueue[Envelope]()
override def dequeue(): Envelope = queue.peek()
def ack(): Unit = queue.poll()
override def cleanUp(owner: ActorRef, deadLetters: MessageQueue): Unit = {
super.cleanUp(owner, deadLetters)
PeekMailboxExtension(system).unregister(owner)
}
}
class MyActor extends Actor {
def receive = {
case msg ⇒
//doStuff(msg)
println(msg)
PeekMailboxExtension.ack()
}
}
object MyApp extends App {
val system = ActorSystem("MySystem", ConfigFactory.parseString("""
peek-dispatcher {
mailbox-type = "example.PeekMailboxType"
}
"""))
val myActor = system.actorOf(Props[MyActor].withDispatcher("peek-dispatcher"), name = "myActor")
myActor ! "Hello"
myActor ! "World"
myActor ! PoisonPill
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment