0.01
Low commit activity in last 3 years
No release in over a year
Event sourced entity store with a replaceable body
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
 Dependencies

Runtime

~> 4.0
~> 0.2
 Project Readme

Entity Store

Event sourced entity store implementation that ships with a MongoDB store but this can be replaced. Version 0.2 release now allows for running on iOS as part of the entity_stormotion gem.

endorse

Usage

Tracking state change

Rather than directly changing properties of an entity via setter methods, state change happens as a result of executing commands that spawn one or more events. It is these events that are persisted.

A typical entity would look like this

class Tyre
	include EntityStore::Entity

	attr_accessor :pressure

	def inflate(new_pressure)
		record_event TyreInflated.new(new_pressure: new_pressure)
	end
end

The corresponding event would look like this

class TyreInflated
	include EntityStore::Event

	attr_accessor :new_pressure

	def apply(entity)
		entity.pressure = new_pressure
	end
end

The record_event method adds the event to the entity's pending_events queue and applies the event.

The entity is passed the an instance of the entity store via the save method (new entities use add). This results in the pending events being persisted to the entity_events collection in the configured MongoDB repository.

Subscribing to events

In order to denormalise the event subscribers need to be configured to receive events that are published to the internal event bus.

In order to subscribe to an event then a subscriber must expose a instance method matching the event's receiver_name. This is, by default the lower case event class name with underscores between words

eg: a TyreInflated event is received by a tyre_inflated method.

You can also subscribe to all events by implementing an all_events method on a subscriber. If both a matching receiver method and all_events are defined on a subscriber, both will be called.

Entity Values

The EntityValue module provides extensions to support complex objects as values on attributes. For example.

class Address
	include EntityStore::EntityValue

	attr_accessor :street, :town, :county, :post_code, :country
end

class HomeAddressSet
	include EntityStore::Event

	entity_value_attribute :home_address, Address

	def	apply(entity)
		entity.home_address = home_address
	end
end

class Member
	include EntityStore::Entity

	attr_accessor :first_name, :last_name
	entity_value_attribute :home_address, Address

	def set_home_address(address)
		record_event(HomeAddressSet.new(:home_address => address))
	end
end

You'll note that a class method entity_value_attribute is used to mark up the entity and event correctly. Slightly uncomfortable that this isn't a poro (plain old ruby object) class. This is my solution to robust serialisation of these objects. There could well be a better way.

Version Changes

An entity version increment can comprise several events. When a version is incremented, ie after save, an event is raised of class

<entity_class_name>VersionIncremented

In the case of the Member entity above this would be

MemberVersionIncremented

This will contain two properties #entity_id and #version.

Snapshotting

Each time an entity is saved, it's version is incremented.

You can specify a snapshot_threshold while configuring the gem. This will cause a snapshot to be created and attached to the entity record. When an entity is retrieved from the data store, only events post the snapshot version will be retrieved and applied to the entity.

Replay

Replaying of specific events to specific subscribers is possible via the EventBus. This will pull and apply the matching events from the external_event_store in the order they were inserted.

EventBus.new.replay Time.new(2011, 11, 1), 'EventTypeName', SubscriberClass

The first argument is the Time from which you wish to find events from.

Configuration

An initialiser file should assign at minimum a configured store to use.

EntityStore::MongoEntityStore.connection_profile = ENV['MONGO_URL'] || "mongodb://localhost/my_cars_#{Rails.env}"

EntityStore.setup do |config|
	config.store = EntityStore::MongoEntityStore.new
	config.event_subscribers.concat([CarDenormaliser, CarSafetyService])
end

EntityStore.feed_store is configured in a similar way.

You can also override the type loader used by passing a lambda or a Proc. Handy if, as in my case, you moved the entity classes to a new module namespace.

  config.type_loader = lambda {|type_name|
    begin
      type_name.split('::').inject(Object) {|obj, name| obj.const_get(name) }
    rescue NameError => e
      "NewNamespace::#{type_name}".split('::').inject(Object) {|obj, name| obj.const_get(name) }
    end
  }

Replace The Store

The store used is replaceable. The minimum interface requirements for the EntityStore.store. Types should be loaded using the EntityStore.load_type method (bit smelly).

class MyStore

	# Public - adds the entity to the store
	#
	# entity     - An object that behaves an entity, use the EntityStore::Entity mixin
	#
	# Returns String id of the entity
	def add_entity(entity)

	end

	def save_entity(entity)
		# this will be called if the entity has an id
	end

	def get_entities(ids, options)
		# returns an array of entities as empty shells of the appropriate type
		# if a snapshot exists then this should be returned
	end

	def add_events(events)

	end

	def get_events(criteria)
		# returns all events in time sequence since the version if passed otherwise all
	end

	def snapshot_entity(entity)
		# create a snapshot of the entity that can be retrievd without replaying
		# the entire event stream
	end

	def remove_entity_snapshot(id)
		# remove the snapshot so next time the entity is retrieved it replays the event stream
		# to rehhydrate the entity
	end

end

You can also replace the EntityStore.feed_store with

class MyFeedStore

	def add_event(entity_type, event)
		# entity_type is a string
	end

	def get_events(since, type=nil, max_items=nil)
		# retrieve all events since a the DateTime passed as since
	end

end

TODO

  • Concurrency - actually do something with the version of the entity
  • Backup - make copy of all events to external store
  • Restore - restore all backed up events