euston-eventstore 1.0.2-java
Sign up to get free protection for your applications and to get access to all the features.
- data/Rakefile +126 -0
- data/euston-eventstore.gemspec +68 -0
- data/lib/euston-eventstore/commit.rb +77 -0
- data/lib/euston-eventstore/constants.rb +5 -0
- data/lib/euston-eventstore/dispatcher/asynchronous_dispatcher.rb +37 -0
- data/lib/euston-eventstore/dispatcher/null_dispatcher.rb +11 -0
- data/lib/euston-eventstore/dispatcher/synchronous_dispatcher.rb +21 -0
- data/lib/euston-eventstore/errors.rb +21 -0
- data/lib/euston-eventstore/event_message.rb +26 -0
- data/lib/euston-eventstore/optimistic_event_store.rb +68 -0
- data/lib/euston-eventstore/optimistic_event_stream.rb +106 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_commit.rb +82 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_commit_id.rb +16 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_config.rb +28 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_event_message.rb +31 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_persistence_engine.rb +167 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_persistence_factory.rb +31 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_snapshot.rb +32 -0
- data/lib/euston-eventstore/persistence/mongodb/mongo_stream_head.rb +29 -0
- data/lib/euston-eventstore/persistence/stream_head.rb +23 -0
- data/lib/euston-eventstore/snapshot.rb +21 -0
- data/lib/euston-eventstore/version.rb +5 -0
- data/lib/euston-eventstore.rb +7 -0
- data/spec/event_store/dispatcher/asynchronous_dispatcher_spec.rb +75 -0
- data/spec/event_store/dispatcher/synchronous_dispatcher_spec.rb +39 -0
- data/spec/event_store/optimistic_event_store_spec.rb +292 -0
- data/spec/event_store/optimistic_event_stream_spec.rb +318 -0
- data/spec/event_store/persistence/mongodb_spec.rb +301 -0
- data/spec/event_store/serialization/simple_message.rb +12 -0
- data/spec/spec_helper.rb +39 -0
- data/spec/support/array_enumeration_counter.rb +20 -0
- metadata +178 -0
data/Rakefile
ADDED
@@ -0,0 +1,126 @@
|
|
1
|
+
require 'date'
|
2
|
+
require 'rspec/core/rake_task'
|
3
|
+
|
4
|
+
#############################################################################
|
5
|
+
#
|
6
|
+
# Helper functions
|
7
|
+
#
|
8
|
+
#############################################################################
|
9
|
+
|
10
|
+
def name
|
11
|
+
@name ||= Dir['*.gemspec'].first.split('.').first
|
12
|
+
end
|
13
|
+
|
14
|
+
def version
|
15
|
+
line = File.read("lib/#{name}/version.rb")[/^\s*VERSION\s*=\s*.*/]
|
16
|
+
line.match(/.*VERSION\s*=\s*['"](.*)['"]/)[1]
|
17
|
+
end
|
18
|
+
|
19
|
+
def date
|
20
|
+
Date.today.to_s
|
21
|
+
end
|
22
|
+
|
23
|
+
def rubyforge_project
|
24
|
+
name
|
25
|
+
end
|
26
|
+
|
27
|
+
def gemspec_file
|
28
|
+
"#{name}.gemspec"
|
29
|
+
end
|
30
|
+
|
31
|
+
def gem_file
|
32
|
+
"#{name}-#{version}.gem"
|
33
|
+
end
|
34
|
+
|
35
|
+
def replace_header(head, header_name)
|
36
|
+
head.sub!(/(\.#{header_name}\s*= ').*'/) { "#{$1}#{send(header_name)}'"}
|
37
|
+
end
|
38
|
+
|
39
|
+
def platform
|
40
|
+
RUBY_PLATFORM.to_s == 'java' ? '-java' : ''
|
41
|
+
end
|
42
|
+
|
43
|
+
#############################################################################
|
44
|
+
#
|
45
|
+
# Custom tasks
|
46
|
+
#
|
47
|
+
#############################################################################
|
48
|
+
|
49
|
+
default_rspec_opts = %w[--colour --format Fuubar]
|
50
|
+
|
51
|
+
desc "Run all examples"
|
52
|
+
RSpec::Core::RakeTask.new(:spec) do |t|
|
53
|
+
t.rspec_opts = default_rspec_opts
|
54
|
+
end
|
55
|
+
|
56
|
+
#############################################################################
|
57
|
+
#
|
58
|
+
# Packaging tasks
|
59
|
+
#
|
60
|
+
#############################################################################
|
61
|
+
|
62
|
+
def built_gem
|
63
|
+
Dir["#{name}*.gem"].first
|
64
|
+
end
|
65
|
+
|
66
|
+
desc "Create tag v#{version} and build and push #{gem_file} to Rubygems"
|
67
|
+
task :release => :build do
|
68
|
+
unless `git branch` =~ /^\* master$/
|
69
|
+
puts "You must be on the master branch to release!"
|
70
|
+
exit!
|
71
|
+
end
|
72
|
+
sh "git commit --allow-empty -a -m 'Release #{version}'"
|
73
|
+
sh "git tag v#{version}"
|
74
|
+
sh "git push origin master"
|
75
|
+
sh "git push origin v#{version}"
|
76
|
+
sh "gem push pkg/#{built_gem}"
|
77
|
+
end
|
78
|
+
|
79
|
+
desc "Build #{gem_file} into the pkg directory"
|
80
|
+
task :build => :gemspec do
|
81
|
+
sh "mkdir -p pkg"
|
82
|
+
sh "gem build #{gemspec_file}"
|
83
|
+
sh "mv #{built_gem} pkg"
|
84
|
+
end
|
85
|
+
|
86
|
+
desc "Generate #{gemspec_file}"
|
87
|
+
task :gemspec => :validate do
|
88
|
+
# read spec file and split out manifest section
|
89
|
+
spec = File.read(gemspec_file)
|
90
|
+
head, manifest, tail = spec.split(" # = MANIFEST =\n")
|
91
|
+
|
92
|
+
# replace name version and date
|
93
|
+
replace_header(head, :name)
|
94
|
+
replace_header(head, :version)
|
95
|
+
replace_header(head, :date)
|
96
|
+
#comment this out if your rubyforge_project has a different name
|
97
|
+
#replace_header(head, :rubyforge_project)
|
98
|
+
|
99
|
+
# determine file list from git ls-files
|
100
|
+
files = `git ls-files`.
|
101
|
+
split("\n").
|
102
|
+
sort.
|
103
|
+
reject { |file| file =~ /^\./ }.
|
104
|
+
reject { |file| file =~ /^(rdoc|pkg)/ }.
|
105
|
+
map { |file| " #{file}" }.
|
106
|
+
join("\n")
|
107
|
+
|
108
|
+
# piece file back together and write
|
109
|
+
manifest = " s.files = %w[\n#{files}\n ]\n"
|
110
|
+
spec = [head, manifest, tail].join(" # = MANIFEST =\n")
|
111
|
+
File.open(gemspec_file, 'w') { |io| io.write(spec) }
|
112
|
+
puts "Updated #{gemspec_file}"
|
113
|
+
end
|
114
|
+
|
115
|
+
desc "Validate #{gemspec_file}"
|
116
|
+
task :validate do
|
117
|
+
libfiles = Dir['lib/*'] - ["lib/#{name}.rb", "lib/#{name}"]
|
118
|
+
unless libfiles.empty?
|
119
|
+
puts "Directory `lib` should only contain a `#{name}.rb` file and `#{name}` dir."
|
120
|
+
exit!
|
121
|
+
end
|
122
|
+
unless Dir['VERSION*'].empty?
|
123
|
+
puts "A `VERSION` file at root level violates Gem best practices."
|
124
|
+
exit!
|
125
|
+
end
|
126
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
Gem::Specification.new do |s|
|
2
|
+
s.name = 'euston-eventstore'
|
3
|
+
s.version = '1.0.2'
|
4
|
+
s.date = '2011-09-15'
|
5
|
+
# s.platform = Gem::Platform::CURRENT
|
6
|
+
s.platform = RUBY_PLATFORM.to_s == 'java' ? 'java' : Gem::Platform::RUBY
|
7
|
+
s.authors = ['Lee Henson', 'Guy Boertje']
|
8
|
+
s.email = ['lee.m.henson@gmail.com', 'guyboertje@gmail.com']
|
9
|
+
s.summary = %q{Event store for use with Euston.}
|
10
|
+
s.description = "Ruby port for Jonathan Oliver's EventStore. See https://github.com/joliver/EventStore for details."
|
11
|
+
s.homepage = 'http://github.com/leemhenson/euston-eventstore'
|
12
|
+
# = MANIFEST =
|
13
|
+
s.files = %w[
|
14
|
+
Rakefile
|
15
|
+
euston-eventstore.gemspec
|
16
|
+
lib/euston-eventstore.rb
|
17
|
+
lib/euston-eventstore/commit.rb
|
18
|
+
lib/euston-eventstore/constants.rb
|
19
|
+
lib/euston-eventstore/dispatcher/asynchronous_dispatcher.rb
|
20
|
+
lib/euston-eventstore/dispatcher/null_dispatcher.rb
|
21
|
+
lib/euston-eventstore/dispatcher/synchronous_dispatcher.rb
|
22
|
+
lib/euston-eventstore/errors.rb
|
23
|
+
lib/euston-eventstore/event_message.rb
|
24
|
+
lib/euston-eventstore/optimistic_event_store.rb
|
25
|
+
lib/euston-eventstore/optimistic_event_stream.rb
|
26
|
+
lib/euston-eventstore/persistence/mongodb/mongo_commit.rb
|
27
|
+
lib/euston-eventstore/persistence/mongodb/mongo_commit_id.rb
|
28
|
+
lib/euston-eventstore/persistence/mongodb/mongo_config.rb
|
29
|
+
lib/euston-eventstore/persistence/mongodb/mongo_event_message.rb
|
30
|
+
lib/euston-eventstore/persistence/mongodb/mongo_persistence_engine.rb
|
31
|
+
lib/euston-eventstore/persistence/mongodb/mongo_persistence_factory.rb
|
32
|
+
lib/euston-eventstore/persistence/mongodb/mongo_snapshot.rb
|
33
|
+
lib/euston-eventstore/persistence/mongodb/mongo_stream_head.rb
|
34
|
+
lib/euston-eventstore/persistence/stream_head.rb
|
35
|
+
lib/euston-eventstore/snapshot.rb
|
36
|
+
lib/euston-eventstore/version.rb
|
37
|
+
spec/event_store/dispatcher/asynchronous_dispatcher_spec.rb
|
38
|
+
spec/event_store/dispatcher/synchronous_dispatcher_spec.rb
|
39
|
+
spec/event_store/optimistic_event_store_spec.rb
|
40
|
+
spec/event_store/optimistic_event_stream_spec.rb
|
41
|
+
spec/event_store/persistence/mongodb_spec.rb
|
42
|
+
spec/event_store/serialization/simple_message.rb
|
43
|
+
spec/spec_helper.rb
|
44
|
+
spec/support/array_enumeration_counter.rb
|
45
|
+
]
|
46
|
+
# = MANIFEST =
|
47
|
+
|
48
|
+
s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
|
49
|
+
s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
|
50
|
+
|
51
|
+
s.add_dependency 'activesupport', '~> 3.0.9'
|
52
|
+
s.add_dependency 'hash-keys', '~> 1.0.0'
|
53
|
+
s.add_dependency 'require_all', '~> 1.2.0'
|
54
|
+
s.add_dependency 'uuid', '~> 2.3.0'
|
55
|
+
|
56
|
+
if RUBY_PLATFORM.to_s == 'java'
|
57
|
+
s.add_dependency 'json-jruby', '~> 1.5.0'
|
58
|
+
s.add_dependency 'jmongo', '~> 1.0.0'
|
59
|
+
else
|
60
|
+
s.add_dependency 'bson_ext', '~> 1.1.0'
|
61
|
+
s.add_dependency 'json', '~> 1.5.0'
|
62
|
+
s.add_dependency 'mongo', '~> 1.3.1'
|
63
|
+
end
|
64
|
+
|
65
|
+
s.add_development_dependency 'awesome_print', '~> 0.4.0'
|
66
|
+
s.add_development_dependency 'fuubar', '~> 0.0.0'
|
67
|
+
s.add_development_dependency 'rspec', '~> 2.6.0'
|
68
|
+
end
|
@@ -0,0 +1,77 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
|
4
|
+
# Represents a series of events which have been fully committed as a single unit and which apply to the stream indicated.
|
5
|
+
class Commit
|
6
|
+
def initialize(hash)
|
7
|
+
defaults = {
|
8
|
+
:stream_id => nil,
|
9
|
+
:stream_revision => 1,
|
10
|
+
:commit_id => nil,
|
11
|
+
:commit_sequence => 1,
|
12
|
+
:commit_timestamp => Time.now.utc,
|
13
|
+
:headers => OpenStruct.new,
|
14
|
+
:events => []
|
15
|
+
}
|
16
|
+
values = defaults.merge hash
|
17
|
+
defaults.keys.each { |key| instance_variable_set "@#{key}", values[key] }
|
18
|
+
end
|
19
|
+
|
20
|
+
def to_hash
|
21
|
+
{
|
22
|
+
:stream_id => stream_id,
|
23
|
+
:stream_revision => stream_revision,
|
24
|
+
:commit_id => commit_id,
|
25
|
+
:commit_sequence => commit_sequence,
|
26
|
+
:commit_timestamp => commit_timestamp,
|
27
|
+
:headers => headers.is_a?(OpenStruct) ? headers.instance_variable_get(:@table) : headers,
|
28
|
+
:events => events
|
29
|
+
}
|
30
|
+
end
|
31
|
+
# Gets the value which uniquely identifies the stream to which the commit belongs.
|
32
|
+
attr_reader :stream_id
|
33
|
+
|
34
|
+
# Gets the value which indicates the revision of the most recent event in the stream to which this commit applies.
|
35
|
+
attr_reader :stream_revision
|
36
|
+
|
37
|
+
# Gets the value which uniquely identifies the commit within the stream.
|
38
|
+
attr_reader :commit_id
|
39
|
+
|
40
|
+
# Gets the value which indicates the sequence (or position) in the stream to which this commit applies.
|
41
|
+
attr_reader :commit_sequence
|
42
|
+
|
43
|
+
# Gets the point in time at which the commit was persisted.
|
44
|
+
attr_reader :commit_timestamp
|
45
|
+
|
46
|
+
# Gets the metadata which provides additional, unstructured information about this commit.
|
47
|
+
attr_reader :headers
|
48
|
+
|
49
|
+
# Gets the collection of event messages to be committed as a single unit.
|
50
|
+
attr_reader :events
|
51
|
+
|
52
|
+
def ==(other)
|
53
|
+
(other.is_a? Commit) && (@stream_id == other.stream_id) && (@commit_id == other.commit_id)
|
54
|
+
end
|
55
|
+
|
56
|
+
class << self
|
57
|
+
def empty?(attempt)
|
58
|
+
attempt.nil? || attempt.events.empty?
|
59
|
+
end
|
60
|
+
|
61
|
+
def has_identifier?(attempt)
|
62
|
+
!(attempt.stream_id.nil? || attempt.commit_id.nil?)
|
63
|
+
end
|
64
|
+
|
65
|
+
def valid?(attempt)
|
66
|
+
raise ArgumentError.new('The commit must not be nil.') if attempt.nil?
|
67
|
+
raise ArgumentError.new('The commit must be uniquely identified.') unless Commit.has_identifier? attempt
|
68
|
+
raise ArgumentError.new('The commit sequence must be a positive number.') unless attempt.commit_sequence > 0
|
69
|
+
raise ArgumentError.new('The stream revision must be a positive number.') unless attempt.stream_revision > 0
|
70
|
+
raise ArgumentError.new('The stream revision must always be greater than or equal to the commit sequence.') if (attempt.stream_revision < attempt.commit_sequence)
|
71
|
+
|
72
|
+
true
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
module Dispatcher
|
4
|
+
class AsynchronousDispatcher
|
5
|
+
def initialize(bus, persistence, &block)
|
6
|
+
@bus = bus
|
7
|
+
@persistence = persistence
|
8
|
+
@handle_exception = block_given? ? block : Proc.new {}
|
9
|
+
|
10
|
+
start
|
11
|
+
end
|
12
|
+
|
13
|
+
def dispatch(commit)
|
14
|
+
Thread.fork(commit) { |c| begin_dispatch c }
|
15
|
+
end
|
16
|
+
|
17
|
+
protected
|
18
|
+
|
19
|
+
def begin_dispatch(commit)
|
20
|
+
begin
|
21
|
+
@bus.publish commit
|
22
|
+
@persistence.mark_commit_as_dispatched commit
|
23
|
+
rescue Exception => e
|
24
|
+
@handle_exception.call commit, e
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
def start
|
31
|
+
@persistence.init
|
32
|
+
@persistence.get_undispatched_commits.each { |commit| dispatch commit }
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
module Dispatcher
|
4
|
+
class SynchronousDispatcher
|
5
|
+
def initialize persistence, &block
|
6
|
+
@persistence = persistence
|
7
|
+
@dispatch = block
|
8
|
+
end
|
9
|
+
|
10
|
+
def dispatch commit
|
11
|
+
@dispatch.call commit
|
12
|
+
@persistence.mark_commit_as_dispatched commit
|
13
|
+
end
|
14
|
+
|
15
|
+
def lookup
|
16
|
+
@persistence.get_undispatched_commits.each { |commit| dispatch commit }
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
# Represents an optimistic concurrency conflict between multiple writers.
|
4
|
+
class ConcurrencyError < RuntimeError; end
|
5
|
+
|
6
|
+
# Represents an attempt to commit the same information more than once.
|
7
|
+
class DuplicateCommitError < RuntimeError; end
|
8
|
+
|
9
|
+
# Represents a loss of communications with the storage
|
10
|
+
class StorageUnavailableError < RuntimeError; end
|
11
|
+
|
12
|
+
# Represents a general failure of the storage engine or persistence infrastructure.
|
13
|
+
class StorageError < RuntimeError; end
|
14
|
+
|
15
|
+
# Represents an attempt to commit the same information more than once.
|
16
|
+
class StreamNotFoundError < RuntimeError; end
|
17
|
+
|
18
|
+
# Represents an error when the proxy returns a non 200 code that does not map to any of the above errors.
|
19
|
+
class ProxyCallError < RuntimeError; end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,26 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
# Represents a single element in a stream of events.
|
4
|
+
class EventMessage
|
5
|
+
|
6
|
+
def initialize(arg = nil)
|
7
|
+
if arg.is_a?(Hash) && (arg.keys & ['body','headers']).size == 2
|
8
|
+
@body, @headers = arg.values_at('body','headers')
|
9
|
+
else
|
10
|
+
@headers = {}
|
11
|
+
@body = arg
|
12
|
+
end
|
13
|
+
end
|
14
|
+
|
15
|
+
def to_hash
|
16
|
+
{:headers=>@headers,:body=>@body}
|
17
|
+
end
|
18
|
+
|
19
|
+
# Gets the metadata which provides additional, unstructured information about this message.
|
20
|
+
attr_reader :headers
|
21
|
+
|
22
|
+
# Gets or sets the actual event message body.
|
23
|
+
attr_reader :body
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
class OptimisticEventStore
|
4
|
+
def initialize(persistence)
|
5
|
+
@persistence = persistence
|
6
|
+
end
|
7
|
+
|
8
|
+
def instrumentation
|
9
|
+
return nil unless @persistence.respond_to?(:instrumentation)
|
10
|
+
@persistence.instrumentation
|
11
|
+
end
|
12
|
+
|
13
|
+
def add_snapshot(snapshot)
|
14
|
+
@persistence.add_snapshot snapshot
|
15
|
+
end
|
16
|
+
|
17
|
+
def commit(attempt)
|
18
|
+
return unless Commit.valid?(attempt) && !Commit.empty?(attempt)
|
19
|
+
|
20
|
+
@persistence.commit attempt
|
21
|
+
end
|
22
|
+
|
23
|
+
def create_stream(stream_id)
|
24
|
+
OptimisticEventStream.new(:stream_id => stream_id,
|
25
|
+
:persistence => self)
|
26
|
+
end
|
27
|
+
|
28
|
+
def get_from(stream_id, min_revision, max_revision)
|
29
|
+
@persistence.get_from(:stream_id => stream_id,
|
30
|
+
:min_revision => min_revision,
|
31
|
+
:max_revision => max_revision).to_enum
|
32
|
+
end
|
33
|
+
|
34
|
+
def get_snapshot(stream_id, max_revision)
|
35
|
+
@persistence.get_snapshot stream_id, validate_max_revision(max_revision)
|
36
|
+
end
|
37
|
+
|
38
|
+
def get_streams_to_snapshot(max_threshold)
|
39
|
+
@persistence.get_streams_to_snapshot max_threshold
|
40
|
+
end
|
41
|
+
|
42
|
+
def open_stream(options)
|
43
|
+
options = { :stream_id => nil,
|
44
|
+
:min_revision => 0,
|
45
|
+
:max_revision => 0,
|
46
|
+
:snapshot => nil }.merge(options)
|
47
|
+
|
48
|
+
options = options.merge(:max_revision => validate_max_revision(options[:max_revision]),
|
49
|
+
:persistence => self)
|
50
|
+
|
51
|
+
if options[:snapshot].nil?
|
52
|
+
options.delete :snapshot
|
53
|
+
else
|
54
|
+
options.delete :stream_id
|
55
|
+
options.delete :min_revision
|
56
|
+
end
|
57
|
+
|
58
|
+
OptimisticEventStream.new options
|
59
|
+
end
|
60
|
+
|
61
|
+
private
|
62
|
+
|
63
|
+
def validate_max_revision(max_revision)
|
64
|
+
max_revision <= 0 ? FIXNUM_MAX : max_revision
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
@@ -0,0 +1,106 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
class OptimisticEventStream
|
4
|
+
def initialize(options)
|
5
|
+
@persistence = options[:persistence]
|
6
|
+
@committed_events = []
|
7
|
+
@uncommitted_events = []
|
8
|
+
@uncommitted_headers = {}
|
9
|
+
@commit_sequence = 0
|
10
|
+
@identifiers = []
|
11
|
+
|
12
|
+
if options.has_key? :snapshot
|
13
|
+
snapshot = options[:snapshot]
|
14
|
+
@stream_id = snapshot.stream_id
|
15
|
+
commits = @persistence.get_from @stream_id, snapshot.stream_revision, options[:max_revision]
|
16
|
+
populate_stream snapshot.stream_revision + 1, options[:max_revision], commits
|
17
|
+
@stream_revision = snapshot.stream_revision + committed_events.length
|
18
|
+
else
|
19
|
+
@stream_id = options[:stream_id]
|
20
|
+
@stream_revision = 0
|
21
|
+
min_revision = options[:min_revision] ||= nil
|
22
|
+
max_revision = options[:max_revision] ||= nil
|
23
|
+
|
24
|
+
unless min_revision.nil? || max_revision.nil?
|
25
|
+
commits = @persistence.get_from @stream_id, min_revision, max_revision
|
26
|
+
populate_stream min_revision, max_revision, commits
|
27
|
+
|
28
|
+
raise StreamNotFoundError if (min_revision > 0 && committed_events.empty?)
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
attr_reader :stream_id, :stream_revision, :commit_sequence, :committed_events, :uncommitted_events, :uncommitted_headers
|
34
|
+
|
35
|
+
def <<(event)
|
36
|
+
@uncommitted_events << event unless event.nil? || event.body.nil?
|
37
|
+
end
|
38
|
+
|
39
|
+
def clear_changes
|
40
|
+
@uncommitted_events = []
|
41
|
+
@uncommitted_headers = {}
|
42
|
+
end
|
43
|
+
|
44
|
+
def commit_changes(commit_id)
|
45
|
+
raise Euston::EventStore::DuplicateCommitError if @identifiers.include? commit_id
|
46
|
+
|
47
|
+
return unless has_changes
|
48
|
+
|
49
|
+
begin
|
50
|
+
persist_changes commit_id
|
51
|
+
rescue ConcurrencyError => e
|
52
|
+
commits = @persistence.get_from stream_id, stream_revision + 1, FIXNUM_MAX
|
53
|
+
populate_stream stream_revision + 1, FIXNUM_MAX, commits
|
54
|
+
|
55
|
+
raise e
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
protected
|
60
|
+
|
61
|
+
def copy_values_to_new_commit(commit_id)
|
62
|
+
Euston::EventStore::Commit.new :stream_id => stream_id,
|
63
|
+
:stream_revision => stream_revision + uncommitted_events.length,
|
64
|
+
:commit_id => commit_id,
|
65
|
+
:commit_sequence => commit_sequence + 1,
|
66
|
+
:commit_timestamp => Time.now.utc,
|
67
|
+
:headers => uncommitted_headers,
|
68
|
+
:events => uncommitted_events
|
69
|
+
end
|
70
|
+
|
71
|
+
def has_changes
|
72
|
+
!uncommitted_events.empty?
|
73
|
+
end
|
74
|
+
|
75
|
+
def persist_changes(commit_id)
|
76
|
+
commit = copy_values_to_new_commit commit_id
|
77
|
+
@persistence.commit commit
|
78
|
+
|
79
|
+
populate_stream stream_revision + 1, commit.stream_revision, [ commit ]
|
80
|
+
clear_changes
|
81
|
+
end
|
82
|
+
|
83
|
+
def populate_stream(min_revision, max_revision, commits = [])
|
84
|
+
commits.each do |commit|
|
85
|
+
@identifiers << commit.commit_id
|
86
|
+
@commit_sequence = commit.commit_sequence
|
87
|
+
|
88
|
+
current_revision = commit.stream_revision - commit.events.length + 1
|
89
|
+
|
90
|
+
return if current_revision > max_revision
|
91
|
+
|
92
|
+
commit.events.each do |event|
|
93
|
+
break if current_revision > max_revision
|
94
|
+
|
95
|
+
unless current_revision < min_revision
|
96
|
+
@committed_events << event
|
97
|
+
@stream_revision = current_revision
|
98
|
+
end
|
99
|
+
|
100
|
+
current_revision += 1
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
106
|
+
end
|
@@ -0,0 +1,82 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
module Persistence
|
4
|
+
module Mongodb
|
5
|
+
module MongoCommit
|
6
|
+
extend ::ActiveSupport::Concern
|
7
|
+
|
8
|
+
included do
|
9
|
+
alias_method :original_initialize, :initialize
|
10
|
+
alias_method :initialize, :mongo_initialize
|
11
|
+
end
|
12
|
+
|
13
|
+
class << self
|
14
|
+
def from_hash(hash)
|
15
|
+
return nil if hash.nil?
|
16
|
+
|
17
|
+
id = hash['_id']
|
18
|
+
events = hash['events'].sort_by { |e| e["stream_revision"] }.to_a
|
19
|
+
stream_revision = events.last['stream_revision']
|
20
|
+
events = events.map { |e| Euston::EventStore::Persistence::Mongodb::MongoEventMessage.from_hash e['payload'] }
|
21
|
+
|
22
|
+
Euston::EventStore::Commit.new :stream_id => id['stream_id'],
|
23
|
+
:stream_revision => stream_revision,
|
24
|
+
:commit_id => hash['commit_id'],
|
25
|
+
:commit_sequence => id['commit_sequence'],
|
26
|
+
:commit_timestamp => hash['commit_timestamp'],
|
27
|
+
:headers => hash['headers'].recursive_symbolize_keys!,
|
28
|
+
:events => events
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
def mongo_initialize(hash)
|
33
|
+
original_initialize(hash)
|
34
|
+
@dispatched = hash[:dispatched]
|
35
|
+
end
|
36
|
+
|
37
|
+
attr_reader :dispatched
|
38
|
+
|
39
|
+
def to_hash
|
40
|
+
{
|
41
|
+
:_id => { :stream_id => stream_id, :commit_sequence => commit_sequence },
|
42
|
+
:commit_id => commit_id,
|
43
|
+
:commit_timestamp => commit_timestamp.to_f,
|
44
|
+
:dispatched => dispatched || false,
|
45
|
+
:events => events.map { |e| e.to_hash },
|
46
|
+
:headers => headers
|
47
|
+
}
|
48
|
+
end
|
49
|
+
|
50
|
+
def to_mongo_commit
|
51
|
+
mongo_stream_revision = stream_revision - (events.length - 1)
|
52
|
+
mongo_events = events.map do |e|
|
53
|
+
hash = { :stream_revision => mongo_stream_revision, :payload => e.to_hash }
|
54
|
+
mongo_stream_revision += 1
|
55
|
+
hash
|
56
|
+
end
|
57
|
+
|
58
|
+
{
|
59
|
+
:_id => { :stream_id => stream_id, :commit_sequence => commit_sequence },
|
60
|
+
:commit_id => commit_id,
|
61
|
+
:commit_timestamp => commit_timestamp.to_f,
|
62
|
+
:headers => headers,
|
63
|
+
:events => mongo_events,
|
64
|
+
:dispatched => false
|
65
|
+
}
|
66
|
+
end
|
67
|
+
|
68
|
+
def to_id_query
|
69
|
+
{
|
70
|
+
'_id.commit_sequence' => commit_sequence,
|
71
|
+
'_id.stream_id' => stream_id
|
72
|
+
}
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
class Commit
|
79
|
+
include Persistence::Mongodb::MongoCommit
|
80
|
+
end
|
81
|
+
end
|
82
|
+
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module Euston
|
2
|
+
module EventStore
|
3
|
+
module Persistence
|
4
|
+
module Mongodb
|
5
|
+
module MongoCommitId
|
6
|
+
def initialize(stream_id, commit_sequence)
|
7
|
+
@stream_id = stream_id
|
8
|
+
@commit_sequence = commit_sequence
|
9
|
+
end
|
10
|
+
|
11
|
+
attr_reader :stream_id, :commit_sequence
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|