com.github.mboogerd.mmap.InMemMonotonicMap.scala Maven / Gradle / Ivy
/*
* Copyright 2017 Merlijn Boogerd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.github.mboogerd.mmap
import akka.actor.{ ActorRef, ActorRefFactory }
import algebra.lattice.JoinSemilattice
import org.reactivestreams.{ Publisher, Subscriber }
import InMemMonotonicMapMessages._
/**
* Constructs an in-memory mmap.
* @tparam K The key type
*/
class InMemMonotonicMap[K <: AnyRef](storeActor: ActorRef) extends MonotonicMap[K] {
/**
* Attempts to read a stream of updates for `key` as type `V`
*
* @param key The key to read an update-stream from
* @tparam V The value type, which should be a JoinSemilattice (only such values can be written and thus cause updates)
* @return A Publisher with updates for `key` of type `V`. The stream is expected to fail iif any element
* cannot be handled as if being of type `V`. If no Subscriber is created for the Publisher, or the
* subscriber terminates, the Publisher is expected to clean up after itself.
*/
override def read[V <: AnyRef](key: K): Publisher[V] =
(s: Subscriber[_ >: V]) => storeActor ! Read(key, s.asInstanceOf[Subscriber[AnyRef]])
/**
* Attempts to write `value` to `key`. We expect a `JoinSemilattice` for `V` as we ought to be able to merge any
* value, if one were to exist.
*
* @param key The key to write the new value to
* @param value The value to be written
* @tparam V The type of the value, which should be compatible with any existing value
* @return A Publisher that propagates updates of the writes. This Publisher should never fail except for fatal JVM
* exceptions. All other errors should be transformed to instances of `WriteNotification`. Implementations
* are expected to clean up after themselves if subscribers terminate.
*/
override def write[V <: AnyRef: JoinSemilattice](key: K, value: V): Publisher[WriteNotification] =
(s: Subscriber[_ >: WriteNotification]) =>
storeActor ! Write(key, value, implicitly[JoinSemilattice[V]], s.asInstanceOf[Subscriber[WriteNotification]])
}
object InMemMonotonicMap {
/**
*
* @param actorRefFactory The factory to use for construction of the memory-state-wrapping actor
* @param initialState The initial state that the actor will hold
* @tparam K
* @return
*/
def apply[K <: AnyRef](initialState: Map[K, AnyRef] = Map.empty[K, AnyRef])(implicit actorRefFactory: ActorRefFactory): InMemMonotonicMap[K] =
new InMemMonotonicMap(actorRefFactory.actorOf(InMemMonotonicMapActor.props(initialState)))
}