tcb 0.5.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +27 -0
  3. data/LICENSE.txt +21 -0
  4. data/README.md +600 -0
  5. data/lib/generators/tcb/domain/domain_generator.rb +49 -0
  6. data/lib/generators/tcb/domain/templates/command_handler.rb.tt +11 -0
  7. data/lib/generators/tcb/domain/templates/domain_module.rb.tt +34 -0
  8. data/lib/generators/tcb/event_store/event_store_generator.rb +64 -0
  9. data/lib/generators/tcb/event_store/templates/command_handler.rb.tt +18 -0
  10. data/lib/generators/tcb/event_store/templates/domain_module.rb.tt +44 -0
  11. data/lib/generators/tcb/event_store/templates/migration.rb.tt +14 -0
  12. data/lib/generators/tcb/install/install_generator.rb +16 -0
  13. data/lib/generators/tcb/install/templates/tcb.rb.tt +18 -0
  14. data/lib/generators/tcb/shared/command_argument.rb +39 -0
  15. data/lib/tcb/command_bus.rb +26 -0
  16. data/lib/tcb/configuration.rb +118 -0
  17. data/lib/tcb/domain.rb +8 -0
  18. data/lib/tcb/domain_context.rb +29 -0
  19. data/lib/tcb/event_bus/running_strategy.rb +24 -0
  20. data/lib/tcb/event_bus/shutdown_strategy.rb +88 -0
  21. data/lib/tcb/event_bus/subscriber_registry.rb +46 -0
  22. data/lib/tcb/event_bus/termination_signal_handler.rb +55 -0
  23. data/lib/tcb/event_bus.rb +118 -0
  24. data/lib/tcb/event_bus_shutdown.rb +11 -0
  25. data/lib/tcb/event_query.rb +107 -0
  26. data/lib/tcb/event_store/active_record.rb +93 -0
  27. data/lib/tcb/event_store/event_stream_envelope.rb +13 -0
  28. data/lib/tcb/event_store/in_memory.rb +51 -0
  29. data/lib/tcb/handles_commands.rb +31 -0
  30. data/lib/tcb/handles_events.rb +44 -0
  31. data/lib/tcb/minitest_helpers.rb +37 -0
  32. data/lib/tcb/publish.rb +6 -0
  33. data/lib/tcb/record.rb +55 -0
  34. data/lib/tcb/records_events.rb +23 -0
  35. data/lib/tcb/rspec_helpers.rb +61 -0
  36. data/lib/tcb/stream_id.rb +33 -0
  37. data/lib/tcb/subscriber_invocation_failed.rb +31 -0
  38. data/lib/tcb/subscriber_metadata_extractor.rb +66 -0
  39. data/lib/tcb/test_helpers/shared.rb +29 -0
  40. data/lib/tcb/version.rb +5 -0
  41. data/lib/tcb.rb +57 -0
  42. metadata +195 -0
@@ -0,0 +1,31 @@
1
+ # frozen_string_literal: true
2
+
3
+ module TCB
4
+ class SubscriberInvocationFailed < Data.define(
5
+ :original_event,
6
+ :subscriber_type,
7
+ :subscriber_class,
8
+ :subscriber_location,
9
+ :subscriber_source,
10
+ :error_class,
11
+ :error_message,
12
+ :error_backtrace,
13
+ :occurred_at
14
+ )
15
+ def self.build(handler:, original_event:, error:)
16
+ metadata = SubscriberMetadataExtractor.new(handler).extract
17
+
18
+ new(
19
+ original_event: original_event,
20
+ subscriber_type: metadata.subscriber_type,
21
+ subscriber_class: metadata.subscriber_class,
22
+ subscriber_location: metadata.subscriber_location,
23
+ subscriber_source: metadata.subscriber_source,
24
+ error_class: error.class.name,
25
+ error_message: error.message,
26
+ error_backtrace: error.backtrace,
27
+ occurred_at: Time.now
28
+ )
29
+ end
30
+ end
31
+ end
@@ -0,0 +1,66 @@
1
+ # frozen_string_literal: true
2
+
3
+ module TCB
4
+ class SubscriberMetadataExtractor
5
+ SubscriberMetadata = Data.define(
6
+ :subscriber_type,
7
+ :subscriber_class,
8
+ :subscriber_location,
9
+ :subscriber_source
10
+ )
11
+
12
+ def initialize(handler)
13
+ @handler = handler
14
+ end
15
+
16
+ def extract
17
+ if @handler.is_a?(Proc)
18
+ extract_proc_metadata
19
+ else
20
+ extract_class_metadata
21
+ end
22
+ end
23
+
24
+ private
25
+
26
+ def extract_proc_metadata
27
+ file, line = @handler.source_location
28
+ location = file ? "#{file}:#{line}" : nil
29
+
30
+ SubscriberMetadata.new(
31
+ subscriber_type: :block,
32
+ subscriber_class: "Proc",
33
+ subscriber_location: location,
34
+ subscriber_source: extract_source(@handler)
35
+ )
36
+ end
37
+
38
+ def extract_class_metadata
39
+ file, line = @handler.class.source_location
40
+ location = file ? "#{file}:#{line}" : nil
41
+
42
+ SubscriberMetadata.new(
43
+ subscriber_type: :class,
44
+ subscriber_class: @handler.class.name,
45
+ subscriber_location: location,
46
+ subscriber_source: extract_method_source(@handler, :call)
47
+ )
48
+ end
49
+
50
+ def extract_source(proc_or_method)
51
+ return nil unless defined?(MethodSource) # TODO/TBD: We'd need to extract the source of the block
52
+
53
+ proc_or_method.source
54
+ rescue MethodSource::SourceNotFoundError, LoadError
55
+ nil
56
+ end
57
+
58
+ def extract_method_source(handler_instance, method_name)
59
+ return nil unless defined?(MethodSource)
60
+
61
+ handler_instance.method(method_name).source
62
+ rescue MethodSource::SourceNotFoundError, LoadError, NameError
63
+ nil
64
+ end
65
+ end
66
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ module TCB
4
+ module TestHelpers
5
+ module Shared
6
+ def poll_until(within: 1.0, interval: 0.001, &block)
7
+ deadline = Time.now + within
8
+ loop do
9
+ return true if block.call
10
+ return false if Time.now >= deadline
11
+ sleep interval
12
+ end
13
+ end
14
+
15
+ def with_subscriptions(*event_classes)
16
+ captured = Hash.new { |h, k| h[k] = [] }
17
+ subscriptions = event_classes.map do |event_class|
18
+ TCB.config.event_bus.subscribe(event_class) do |event|
19
+ captured[event_class] << event
20
+ end
21
+ end
22
+ yield captured
23
+ ensure
24
+ subscriptions.each { |sub| TCB.config.event_bus.unsubscribe(sub) }
25
+ end
26
+ end
27
+ end
28
+ end
29
+
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module TCB
4
+ VERSION = "0.5.0"
5
+ end
data/lib/tcb.rb ADDED
@@ -0,0 +1,57 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "tcb/minitest_helpers"
4
+ require_relative "tcb/domain_context"
5
+ require_relative "tcb/event_query"
6
+ require_relative "tcb/event_store/active_record"
7
+ require_relative "tcb/event_store/event_stream_envelope"
8
+ require_relative "tcb/event_store/in_memory"
9
+ require_relative "tcb/stream_id"
10
+ require_relative "tcb/handles_events"
11
+ require_relative "tcb/handles_commands"
12
+ require_relative "tcb/records_events"
13
+ require_relative "tcb/record"
14
+ require_relative "tcb/configuration"
15
+ require_relative "tcb/publish"
16
+ require_relative "tcb/command_bus"
17
+ require_relative "tcb/subscriber_metadata_extractor"
18
+ require_relative "tcb/subscriber_invocation_failed"
19
+ require_relative "tcb/event_bus_shutdown"
20
+ require_relative "tcb/domain"
21
+ require_relative "tcb/event_bus"
22
+ require_relative "tcb/version"
23
+
24
+ module TCB
25
+ Envelope = EventStore::EventStreamEnvelope
26
+
27
+ def self.record(events_from: [], events: [], within: nil, &block)
28
+ Record.call(
29
+ events_from: events_from,
30
+ events: events,
31
+ within: within,
32
+ store: config.event_store,
33
+ registrations: config.persist_registrations,
34
+ &block
35
+ )
36
+ end
37
+
38
+ def self.read(domain_module)
39
+ EventQuery.new(
40
+ store: config.event_store,
41
+ context: DomainContext.from_module(domain_module).to_s
42
+ )
43
+ end
44
+
45
+ def self.configure(&block)
46
+ @configure_block = block
47
+ yield config
48
+ config.permitted_serialization_classes
49
+ config.freeze
50
+ end
51
+
52
+ def self.reset!
53
+ @config.event_bus.force_shutdown
54
+ @config = nil
55
+ configure(&@configure_block) if @configure_block
56
+ end
57
+ end
metadata ADDED
@@ -0,0 +1,195 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: tcb
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.5.0
5
+ platform: ruby
6
+ authors:
7
+ - Ljubomir Marković
8
+ bindir: bin
9
+ cert_chain: []
10
+ date: 1980-01-02 00:00:00.000000000 Z
11
+ dependencies:
12
+ - !ruby/object:Gem::Dependency
13
+ name: railties
14
+ requirement: !ruby/object:Gem::Requirement
15
+ requirements:
16
+ - - "~>"
17
+ - !ruby/object:Gem::Version
18
+ version: '8.0'
19
+ type: :development
20
+ prerelease: false
21
+ version_requirements: !ruby/object:Gem::Requirement
22
+ requirements:
23
+ - - "~>"
24
+ - !ruby/object:Gem::Version
25
+ version: '8.0'
26
+ - !ruby/object:Gem::Dependency
27
+ name: activerecord
28
+ requirement: !ruby/object:Gem::Requirement
29
+ requirements:
30
+ - - "~>"
31
+ - !ruby/object:Gem::Version
32
+ version: '8.0'
33
+ type: :development
34
+ prerelease: false
35
+ version_requirements: !ruby/object:Gem::Requirement
36
+ requirements:
37
+ - - "~>"
38
+ - !ruby/object:Gem::Version
39
+ version: '8.0'
40
+ - !ruby/object:Gem::Dependency
41
+ name: sqlite3
42
+ requirement: !ruby/object:Gem::Requirement
43
+ requirements:
44
+ - - "~>"
45
+ - !ruby/object:Gem::Version
46
+ version: '2.9'
47
+ type: :development
48
+ prerelease: false
49
+ version_requirements: !ruby/object:Gem::Requirement
50
+ requirements:
51
+ - - "~>"
52
+ - !ruby/object:Gem::Version
53
+ version: '2.9'
54
+ - !ruby/object:Gem::Dependency
55
+ name: minitest
56
+ requirement: !ruby/object:Gem::Requirement
57
+ requirements:
58
+ - - "~>"
59
+ - !ruby/object:Gem::Version
60
+ version: '5.0'
61
+ type: :development
62
+ prerelease: false
63
+ version_requirements: !ruby/object:Gem::Requirement
64
+ requirements:
65
+ - - "~>"
66
+ - !ruby/object:Gem::Version
67
+ version: '5.0'
68
+ - !ruby/object:Gem::Dependency
69
+ name: rake
70
+ requirement: !ruby/object:Gem::Requirement
71
+ requirements:
72
+ - - "~>"
73
+ - !ruby/object:Gem::Version
74
+ version: '13.0'
75
+ type: :development
76
+ prerelease: false
77
+ version_requirements: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - "~>"
80
+ - !ruby/object:Gem::Version
81
+ version: '13.0'
82
+ - !ruby/object:Gem::Dependency
83
+ name: minitest-reporters
84
+ requirement: !ruby/object:Gem::Requirement
85
+ requirements:
86
+ - - "~>"
87
+ - !ruby/object:Gem::Version
88
+ version: 1.7.1
89
+ type: :development
90
+ prerelease: false
91
+ version_requirements: !ruby/object:Gem::Requirement
92
+ requirements:
93
+ - - "~>"
94
+ - !ruby/object:Gem::Version
95
+ version: 1.7.1
96
+ - !ruby/object:Gem::Dependency
97
+ name: debug
98
+ requirement: !ruby/object:Gem::Requirement
99
+ requirements:
100
+ - - "~>"
101
+ - !ruby/object:Gem::Version
102
+ version: 1.9.2
103
+ type: :development
104
+ prerelease: false
105
+ version_requirements: !ruby/object:Gem::Requirement
106
+ requirements:
107
+ - - "~>"
108
+ - !ruby/object:Gem::Version
109
+ version: 1.9.2
110
+ - !ruby/object:Gem::Dependency
111
+ name: method_source
112
+ requirement: !ruby/object:Gem::Requirement
113
+ requirements:
114
+ - - "~>"
115
+ - !ruby/object:Gem::Version
116
+ version: '1.0'
117
+ type: :runtime
118
+ prerelease: false
119
+ version_requirements: !ruby/object:Gem::Requirement
120
+ requirements:
121
+ - - "~>"
122
+ - !ruby/object:Gem::Version
123
+ version: '1.0'
124
+ description: TCB gives Rails applications a clean domain language for DDD. Thread-safe
125
+ event bus, command routing, aggregate pattern, and opt-in event persistence.
126
+ email:
127
+ - ljubomir@tcb.si
128
+ executables: []
129
+ extensions: []
130
+ extra_rdoc_files: []
131
+ files:
132
+ - CHANGELOG.md
133
+ - LICENSE.txt
134
+ - README.md
135
+ - lib/generators/tcb/domain/domain_generator.rb
136
+ - lib/generators/tcb/domain/templates/command_handler.rb.tt
137
+ - lib/generators/tcb/domain/templates/domain_module.rb.tt
138
+ - lib/generators/tcb/event_store/event_store_generator.rb
139
+ - lib/generators/tcb/event_store/templates/command_handler.rb.tt
140
+ - lib/generators/tcb/event_store/templates/domain_module.rb.tt
141
+ - lib/generators/tcb/event_store/templates/migration.rb.tt
142
+ - lib/generators/tcb/install/install_generator.rb
143
+ - lib/generators/tcb/install/templates/tcb.rb.tt
144
+ - lib/generators/tcb/shared/command_argument.rb
145
+ - lib/tcb.rb
146
+ - lib/tcb/command_bus.rb
147
+ - lib/tcb/configuration.rb
148
+ - lib/tcb/domain.rb
149
+ - lib/tcb/domain_context.rb
150
+ - lib/tcb/event_bus.rb
151
+ - lib/tcb/event_bus/running_strategy.rb
152
+ - lib/tcb/event_bus/shutdown_strategy.rb
153
+ - lib/tcb/event_bus/subscriber_registry.rb
154
+ - lib/tcb/event_bus/termination_signal_handler.rb
155
+ - lib/tcb/event_bus_shutdown.rb
156
+ - lib/tcb/event_query.rb
157
+ - lib/tcb/event_store/active_record.rb
158
+ - lib/tcb/event_store/event_stream_envelope.rb
159
+ - lib/tcb/event_store/in_memory.rb
160
+ - lib/tcb/handles_commands.rb
161
+ - lib/tcb/handles_events.rb
162
+ - lib/tcb/minitest_helpers.rb
163
+ - lib/tcb/publish.rb
164
+ - lib/tcb/record.rb
165
+ - lib/tcb/records_events.rb
166
+ - lib/tcb/rspec_helpers.rb
167
+ - lib/tcb/stream_id.rb
168
+ - lib/tcb/subscriber_invocation_failed.rb
169
+ - lib/tcb/subscriber_metadata_extractor.rb
170
+ - lib/tcb/test_helpers/shared.rb
171
+ - lib/tcb/version.rb
172
+ homepage: https://github.com/tcb-si/tcb
173
+ licenses:
174
+ - MIT
175
+ metadata:
176
+ source_code_uri: https://github.com/tcb-si/tcb
177
+ changelog_uri: https://github.com/tcb-si/tcb/blob/main/CHANGELOG.md
178
+ rdoc_options: []
179
+ require_paths:
180
+ - lib
181
+ required_ruby_version: !ruby/object:Gem::Requirement
182
+ requirements:
183
+ - - ">="
184
+ - !ruby/object:Gem::Version
185
+ version: 3.2.0
186
+ required_rubygems_version: !ruby/object:Gem::Requirement
187
+ requirements:
188
+ - - ">="
189
+ - !ruby/object:Gem::Version
190
+ version: '0'
191
+ requirements: []
192
+ rubygems_version: 3.6.9
193
+ specification_version: 4
194
+ summary: Lightweight DDD runtime for Rails — events, commands, and aggregates
195
+ test_files: []