Created
July 1, 2012 18:55
-
-
Save erikrozendaal/3029244 to your computer and use it in GitHub Desktop.
Event sourcing example - part 1
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
sealed trait PostEvent { | |
def postId: PostId | |
} | |
case class PostAdded(postId: PostId, content: PostContent) extends PostEvent | |
case class PostEdited(postId: PostId, content: PostContent) extends PostEvent | |
case class PostDeleted(postId: PostId) extends PostEvent |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
case class PostContent(author: String, title: String, content: String) | |
case class PostId(uuid: UUID) | |
object PostId { | |
def generate(): PostId = PostId(UUID.randomUUID()) | |
def fromString(s: String): Option[PostId] = s match { | |
case PostIdRegex(uuid) => catching(classOf[RuntimeException]) opt { PostId(UUID.fromString(uuid)) } | |
case _ => None | |
} | |
private val PostIdRegex = """PostId\(([a-fA-F0-9-]{36})\)""".r | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* A specific blog post with its current content. | |
*/ | |
case class Post(id: PostId, content: PostContent) | |
/** | |
* The current state of blog posts, derived from all committed PostEvents. | |
*/ | |
case class Posts(byId: Map[PostId, Post] = Map.empty, orderedByTimeAdded: Seq[PostId] = Vector.empty) { | |
def get(id: PostId): Option[Post] = byId.get(id) | |
def mostRecent(n: Int): Seq[Post] = orderedByTimeAdded.takeRight(n).reverse.map(byId) | |
def apply(event: PostEvent): Posts = event match { | |
case PostAdded(id, content) => | |
this.copy(byId = byId.updated(id, Post(id, content)), orderedByTimeAdded = orderedByTimeAdded :+ id) | |
case PostEdited(id, content) => | |
this.copy(byId = byId.updated(id, byId(id).copy(content = content))) | |
case PostDeleted(id) => | |
this.copy(byId = byId - id, orderedByTimeAdded = orderedByTimeAdded.filterNot(_ == id)) | |
} | |
} | |
object Posts { | |
def fromHistory(events: PostEvent*): Posts = events.foldLeft(Posts())(_ apply _) | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
object PostsController extends Controller { | |
/** | |
* A Scala STM reference holding the current state of the application, | |
* which is derived from all committed events. | |
*/ | |
val posts = Ref(Posts()).single |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* Commits an event and applies it to the current state. | |
*/ | |
def commit(event: PostEvent): Unit = { | |
posts.transform(_.apply(event)) | |
Logger.debug("Committed event: " + event) | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* Show an overview of the most recent blog posts. | |
*/ | |
def index = Action { implicit request => | |
Ok(views.html.posts.index(posts().mostRecent(20))) | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/* | |
* Blog content form definition. | |
*/ | |
private val postContentForm = Form(mapping( | |
"author" -> trimmedText.verifying(minLength(3)), | |
"title" -> trimmedText.verifying(minLength(3)), | |
"content" -> trimmedText.verifying(minLength(3)))(PostContent.apply)(PostContent.unapply)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
def show(id: PostId) = Action { implicit request => | |
posts().get(id) match { | |
case Some(post) => Ok(views.html.posts.edit(id, postContentForm.fill(post.content))) | |
case None => NotFound(notFound(request, None)) | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
def submit(id: PostId) = Action { implicit request => | |
postContentForm.bindFromRequest.fold( | |
formWithErrors => BadRequest(views.html.posts.edit(id, formWithErrors)), | |
postContent => { | |
commit(PostEdited(id, postContent)) | |
Redirect(routes.PostsController.show(id)).flashing("info" -> "Post saved.") | |
}) | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment