Writing tests for a Projection
Like other Akka libraries, Projections ships with a TestKit that a user can include to assert the correctness of their Projection handler implementation. Add the Projections TestKit dependency to your project:
Import the ProjectionTestKit
and other utilities into a new ScalaTest test spec .
- Scala
-
source
import akka.projection.testkit.scaladsl.ProjectionTestKit import akka.projection.testkit.scaladsl.TestProjection import akka.projection.testkit.scaladsl.TestSourceProvider
- Java
The TestKit includes several utilities to run the Projection handler in isolation so that a full projection implementation and source provider are not required.
ProjectionTestKit
runs a projection with the testActorSystem
.TestSourceProvider
allows the user to mock out test dataEnvelopes
that will be processed by the Projection Handler.TestProjection
is a test Projection implementation that uses an in-memory internal offset store.
Using these tools we can assert that our Projection handler meets the following requirements of the ItemPopularityProjectionHandler
.
- Process each shopping cart item event, correctly calculate the item count delta, and update the database.
- Log the popularity of every 10th shopping cart item event that is processed.
- Scala
-
source
package docs.guide import java.time.Instant import scala.concurrent.Future import akka.Done import akka.actor.testkit.typed.scaladsl.LoggingTestKit import akka.actor.testkit.typed.scaladsl.ScalaTestWithActorTestKit import akka.persistence.query.Offset import akka.projection.ProjectionId import akka.projection.eventsourced.EventEnvelope import akka.stream.scaladsl.Source import akka.projection.testkit.scaladsl.ProjectionTestKit import akka.projection.testkit.scaladsl.TestProjection import akka.projection.testkit.scaladsl.TestSourceProvider import org.scalatest.wordspec.AnyWordSpecLike object ShoppingCartAppSpec { // mock out the Cassandra data layer and simulate recording item count updates class MockItemPopularityRepository extends ItemPopularityProjectionRepository { var counts: Map[String, Long] = Map.empty override def update(itemId: String, delta: Int): Future[Done] = Future.successful { counts = counts + (itemId -> (counts.getOrElse(itemId, 0L) + delta)) Done } override def getItem(itemId: String): Future[Option[Long]] = Future.successful(counts.get(itemId)) } } class ShoppingCartAppSpec extends ScalaTestWithActorTestKit() with AnyWordSpecLike { import ShoppingCartAppSpec._ private val projectionTestKit = ProjectionTestKit(system) def createEnvelope(event: ShoppingCartEvents.Event, seqNo: Long, timestamp: Long = 0L) = EventEnvelope(Offset.sequence(seqNo), "persistenceId", seqNo, event, timestamp) "The ItemPopularityProjectionHandler" should { "process item events correctly" in { val repo = new MockItemPopularityRepository val handler = new ItemPopularityProjectionHandler("tag", system, repo) val events = Source( List[EventEnvelope[ShoppingCartEvents.Event]]( createEnvelope(ShoppingCartEvents.ItemAdded("a7098", "bowling shoes", 1), 0L), createEnvelope(ShoppingCartEvents.ItemQuantityAdjusted("a7098", "bowling shoes", 2, 1), 1L), createEnvelope(ShoppingCartEvents.CheckedOut("a7098", Instant.parse("2020-01-01T12:00:00.00Z")), 2L), createEnvelope(ShoppingCartEvents.ItemAdded("0d12d", "akka t-shirt", 1), 3L), createEnvelope(ShoppingCartEvents.ItemAdded("0d12d", "skis", 1), 4L), createEnvelope(ShoppingCartEvents.ItemRemoved("0d12d", "skis", 1), 5L), createEnvelope(ShoppingCartEvents.CheckedOut("0d12d", Instant.parse("2020-01-01T12:05:00.00Z")), 6L))) val projectionId = ProjectionId("name", "key") val sourceProvider = TestSourceProvider[Offset, EventEnvelope[ShoppingCartEvents.Event]](events, extractOffset = env => env.offset) val projection = TestProjection[Offset, EventEnvelope[ShoppingCartEvents.Event]](projectionId, sourceProvider, () => handler) projectionTestKit.run(projection) { repo.counts shouldBe Map("bowling shoes" -> 2, "akka t-shirt" -> 1, "skis" -> 0) } } "log item popularity for day every 10 item events" in { val repo = new MockItemPopularityRepository val handler = new ItemPopularityProjectionHandler("tag", system, repo) val events = (0L until 10L).map { i => createEnvelope(ShoppingCartEvents.ItemAdded("a7098", "bowling shoes", 1), i) } val projectionId = ProjectionId("name", "key") val sourceProvider = TestSourceProvider[Offset, EventEnvelope[ShoppingCartEvents.Event]]( Source(events), extractOffset = env => env.offset) val projection = TestProjection[Offset, EventEnvelope[ShoppingCartEvents.Event]](projectionId, sourceProvider, () => handler) LoggingTestKit .info("ItemPopularityProjectionHandler(tag) item popularity for 'bowling shoes': [10]") .expect { projectionTestKit.runWithTestSink(projection) { testSink => testSink.request(events.length) testSink.expectNextN(events.length) } } } } }
- Java