brut 0.0.8 → 0.0.9
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.
- checksums.yaml +4 -4
- data/Gemfile.lock +1 -1
- data/lib/brut/back_end/sidekiq/middlewares/server/flush_spans.rb +17 -0
- data/lib/brut/back_end/sidekiq/middlewares/server.rb +3 -0
- data/lib/brut/back_end/sidekiq/middlewares.rb +3 -0
- data/lib/brut/back_end/sidekiq.rb +3 -0
- data/lib/brut/cli/app.rb +8 -1
- data/lib/brut/cli/apps/db.rb +3 -1
- data/lib/brut/cli/apps/test.rb +20 -12
- data/lib/brut/cli.rb +6 -1
- data/lib/brut/framework/config.rb +9 -0
- data/lib/brut/framework/errors/missing_configuration.rb +11 -0
- data/lib/brut/framework/errors.rb +1 -0
- data/lib/brut/framework/mcp.rb +10 -1
- data/lib/brut/front_end/handling_results.rb +1 -1
- data/lib/brut/front_end/request_context.rb +3 -1
- data/lib/brut/front_end/route_hooks/setup_request_context.rb +2 -1
- data/lib/brut/front_end/routing.rb +44 -8
- data/lib/brut/spec_support/clock_support.rb +15 -2
- data/lib/brut/spec_support/component_support.rb +1 -1
- data/lib/brut/spec_support/general_support.rb +11 -1
- data/lib/brut/spec_support/matchers/be_routing_for.rb +2 -2
- data/lib/brut/spec_support/rspec_setup.rb +1 -0
- data/lib/brut/version.rb +1 -1
- data/lib/brut.rb +1 -0
- metadata +7 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: b455ac9d87f61b5baa45cc6f4cc11c3b940b015aaf7792ba5ed8229dfd23d2fc
|
4
|
+
data.tar.gz: 5e6216b69db294da81ed31041a4b65a4a2b04de229e3d8695e5c9e1ad1625a84
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 688aba783c6b979976ddf40417a6220a10295effef357aa9ebc0b1cd772fac759cae4ddaa2ce09c5bb4b1559d65ee04261d57a3c4253559b7f22a4c9dfffa42b
|
7
|
+
data.tar.gz: 67dbc5b31bf1c6e02f4f5b505d3f69e1eb084f862ccb78372377a3b49052e71711af4868eec6c5fd3bc68a4994bc10bbecb5ce42a260677c6a42c765f8a21e0a
|
data/Gemfile.lock
CHANGED
@@ -0,0 +1,17 @@
|
|
1
|
+
# Designed to flush all OTel spans after each job is processed. You likely only
|
2
|
+
# want this to be configured in development so you can see the results of individual
|
3
|
+
# job executions. Do not enable in production.
|
4
|
+
#
|
5
|
+
# When using, you want this to be inserted before OTel's sidekiq middleware:
|
6
|
+
#
|
7
|
+
# config.server_middleware do |chain|
|
8
|
+
# chain.insert_before OpenTelemetry::Instrumentation::Sidekiq::Middlewares::Server::TracerMiddleware,
|
9
|
+
# Brut::BackEnd::Sidekiq::Middlewares::Server::FlushSpans
|
10
|
+
# end
|
11
|
+
class Brut::BackEnd::Sidekiq::Middlewares::Server::FlushSpans
|
12
|
+
def call(worker, job, queue)
|
13
|
+
yield
|
14
|
+
ensure
|
15
|
+
OpenTelemetry.tracer_provider.force_flush
|
16
|
+
end
|
17
|
+
end
|
data/lib/brut/cli/app.rb
CHANGED
@@ -208,7 +208,14 @@ class Brut::CLI::App
|
|
208
208
|
return bootstrap_result
|
209
209
|
end
|
210
210
|
after_bootstrap
|
211
|
-
|
211
|
+
if self.class.configure_only?
|
212
|
+
as_execution_result(command.execute)
|
213
|
+
else
|
214
|
+
result = Brut.container.instrumentation.span("CLI.#{$0}", class: self.class.name) do |span|
|
215
|
+
as_execution_result(command.execute)
|
216
|
+
end
|
217
|
+
result
|
218
|
+
end
|
212
219
|
rescue Brut::CLI::Error => ex
|
213
220
|
abort_execution(ex.message)
|
214
221
|
end
|
data/lib/brut/cli/apps/db.rb
CHANGED
@@ -189,7 +189,9 @@ class Brut::CLI::Apps::DB < Brut::CLI::App
|
|
189
189
|
formatted
|
190
190
|
}
|
191
191
|
Brut.container.sequel_db_handle.logger = logger
|
192
|
-
|
192
|
+
Brut.container.instrumentation.span("migrations.run") do
|
193
|
+
Sequel::Migrator.run(Brut.container.sequel_db_handle,migrations_dir)
|
194
|
+
end
|
193
195
|
out.puts "Migrations applied"
|
194
196
|
end
|
195
197
|
end
|
data/lib/brut/cli/apps/test.rb
CHANGED
@@ -40,21 +40,29 @@ class Brut::CLI::Apps::Test < Brut::CLI::App
|
|
40
40
|
def execute
|
41
41
|
Brut.container.sequel_db_handle.disconnect
|
42
42
|
if options.rebuild?(default: rebuild_by_default?)
|
43
|
-
|
44
|
-
|
43
|
+
Brut.container.instrumentation.span("schema.rebuild.before") do
|
44
|
+
out.puts "Rebuilding test database schema"
|
45
|
+
system! "bin/db rebuild --env=test"
|
46
|
+
end
|
45
47
|
end
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
48
|
+
Brut.container.instrumentation.span("tests.run") do |span|
|
49
|
+
if args.empty?
|
50
|
+
span.add_attributes(tests: :all)
|
51
|
+
out.puts "Running all tests"
|
52
|
+
system! "#{rspec_command} #{Brut.container.app_specs_dir}/"
|
53
|
+
else
|
54
|
+
span.add_attributes(tests: args.length)
|
55
|
+
test_args = args.map { |_|
|
56
|
+
'"' + Shellwords.escape(_) + '"'
|
57
|
+
}.join(" ")
|
58
|
+
system! "#{rspec_command} #{test_args}"
|
59
|
+
end
|
54
60
|
end
|
55
61
|
if options.rebuild_after?(default: rebuild_after_by_default?)
|
56
|
-
|
57
|
-
|
62
|
+
Brut.container.instrumentation.span("schema.rebuild.after") do
|
63
|
+
out.puts "Re-Rebuilding test database schema"
|
64
|
+
system! "bin/db rebuild --env=test"
|
65
|
+
end
|
58
66
|
end
|
59
67
|
0
|
60
68
|
end
|
data/lib/brut/cli.rb
CHANGED
@@ -26,7 +26,12 @@ module Brut
|
|
26
26
|
# @param project_root [Pathname] the path to the root of your project. This is needed before the Brut framework is initialized so
|
27
27
|
# it must be specified explicitly.
|
28
28
|
def self.app(app_klass, project_root:)
|
29
|
-
Brut::CLI::AppRunner.new(app_klass:,project_root:).run
|
29
|
+
Brut::CLI::AppRunner.new(app_klass:,project_root:).run!.tap {
|
30
|
+
otel_configured = OpenTelemetry.tracer_provider.is_a?(OpenTelemetry::SDK::Trace::TracerProvider)
|
31
|
+
if otel_configured
|
32
|
+
OpenTelemetry.tracer_provider.shutdown
|
33
|
+
end
|
34
|
+
}
|
30
35
|
end
|
31
36
|
autoload(:App, "brut/cli/app")
|
32
37
|
autoload(:Command, "brut/cli/command")
|
@@ -424,6 +424,15 @@ class Brut::Framework::Config
|
|
424
424
|
allow_nil: true,
|
425
425
|
)
|
426
426
|
|
427
|
+
Brut.container.store(
|
428
|
+
"fallback_host",
|
429
|
+
URI,
|
430
|
+
"Hostname to use in situations where the request is not available",
|
431
|
+
nil,
|
432
|
+
allow_app_override: true,
|
433
|
+
allow_nil: true
|
434
|
+
)
|
435
|
+
|
427
436
|
c.store(
|
428
437
|
"local_hostname",
|
429
438
|
String,
|
@@ -0,0 +1,11 @@
|
|
1
|
+
# Raised when Brut configuration is missing an expected value. This is mostly raised when values that must be set per app
|
2
|
+
# have not been set.
|
3
|
+
class Brut::Framework::Errors::MissingConfiguration < Brut::Framework::Error
|
4
|
+
# Create the exception
|
5
|
+
#
|
6
|
+
# @param [String|Symbol] config_name the name of the missing configuration parameter
|
7
|
+
# @param [String] context Any additional context to understand the error
|
8
|
+
def initialize(config_name, context)
|
9
|
+
super("Configuration parameter '#{config_name}' did not have a value, but was expected to. #{context}")
|
10
|
+
end
|
11
|
+
end
|
@@ -7,6 +7,7 @@ module Brut
|
|
7
7
|
autoload(:NotImplemented,"brut/framework/errors/not_implemented")
|
8
8
|
autoload(:NotFound,"brut/framework/errors/not_found")
|
9
9
|
autoload(:MissingParameter,"brut/framework/errors/missing_parameter")
|
10
|
+
autoload(:MissingConfiguration,"brut/framework/errors/missing_configuration")
|
10
11
|
autoload(:AbstractMethod,"brut/framework/errors/abstract_method")
|
11
12
|
autoload(:NoClassForPath,"brut/framework/errors/no_class_for_path")
|
12
13
|
# Raises {Brut::Framework::Errors::Bug}
|
data/lib/brut/framework/mcp.rb
CHANGED
@@ -33,7 +33,7 @@ class Brut::Framework::MCP
|
|
33
33
|
#
|
34
34
|
# * Create the instance and *do not* call `boot!`. This is what you'd do if you can't or don't want to connect to external services
|
35
35
|
# like the database. For example, when Brut builds assets, it does not call `boot!`.
|
36
|
-
# * Create the
|
36
|
+
# * Create the instance and immediately call `boot!`. This is what happens most of the time, in particular when the app is started
|
37
37
|
# up by Puma to start serving requests.
|
38
38
|
#
|
39
39
|
# What you should avoid doing is creating an instance of this class and performing logic before later calling `boot!`.
|
@@ -70,6 +70,7 @@ class Brut::Framework::MCP
|
|
70
70
|
end
|
71
71
|
end
|
72
72
|
Sequel::Database.extension :pg_array
|
73
|
+
Sequel::Database.extension :pg_json
|
73
74
|
|
74
75
|
sequel_db = Brut.container.sequel_db_handle
|
75
76
|
|
@@ -100,6 +101,14 @@ class Brut::Framework::MCP
|
|
100
101
|
)
|
101
102
|
)
|
102
103
|
end
|
104
|
+
|
105
|
+
if defined?(OpenTelemetry::Instrumentation::Sidekiq)
|
106
|
+
c.use 'OpenTelemetry::Instrumentation::Sidekiq', {
|
107
|
+
span_naming: :job_class,
|
108
|
+
}
|
109
|
+
else
|
110
|
+
SemanticLogger[self.class].info "OpenTelemetry::Instrumentation::Sidekiq is not loaded, so Sidekiq traces will not be captured"
|
111
|
+
end
|
103
112
|
end
|
104
113
|
|
105
114
|
Brut.container.store(
|
@@ -15,7 +15,7 @@ module Brut::FrontEnd::HandlingResults
|
|
15
15
|
if !klass.kind_of?(Class)
|
16
16
|
raise ArgumentError,"redirect_to should be given a Class, not a #{klass.class}"
|
17
17
|
end
|
18
|
-
Brut.container.routing.
|
18
|
+
Brut.container.routing.path(klass,with_method: :get,**query_string_params)
|
19
19
|
end
|
20
20
|
|
21
21
|
# Return this to return an HTTP status code from a number or string containing the code.
|
@@ -15,13 +15,15 @@ class Brut::FrontEnd::RequestContext
|
|
15
15
|
# @param [Brut::FrontEnd::Flash] flash the current flash
|
16
16
|
# @param [true|false] xhr true if this is an XHR request.
|
17
17
|
# @param [Object] body the `request.body` as provided by Rack
|
18
|
-
|
18
|
+
# @param [URI] host URI the `request.host` and `request.scheme`, and `request.port` as provided by Rack
|
19
|
+
def initialize(env:,session:,flash:,xhr:,body:,host:)
|
19
20
|
@hash = {
|
20
21
|
env:,
|
21
22
|
session:,
|
22
23
|
flash:,
|
23
24
|
xhr:,
|
24
25
|
body:,
|
26
|
+
host:,
|
25
27
|
csrf_token: Rack::Protection::AuthenticityToken.token(env["rack.session"]),
|
26
28
|
clock: Clock.new(session.timezone),
|
27
29
|
}
|
@@ -5,9 +5,10 @@ class Brut::FrontEnd::RouteHooks::SetupRequestContext < Brut::FrontEnd::RouteHoo
|
|
5
5
|
def before(session:,request:,env:)
|
6
6
|
flash = session.flash
|
7
7
|
session[:_flash] ||= flash
|
8
|
+
host_uri = URI.parse("#{request.scheme}://#{request.host}:#{request.port}")
|
8
9
|
Thread.current.thread_variable_set(
|
9
10
|
:request_context,
|
10
|
-
Brut::FrontEnd::RequestContext.new(env:,session:session,flash:,xhr: request.xhr?,body: request.body)
|
11
|
+
Brut::FrontEnd::RequestContext.new(env:,session:session,flash:,xhr: request.xhr?,body: request.body, host: host_uri)
|
11
12
|
)
|
12
13
|
continue
|
13
14
|
end
|
@@ -116,7 +116,25 @@ class Brut::FrontEnd::Routing
|
|
116
116
|
route
|
117
117
|
end
|
118
118
|
|
119
|
-
def
|
119
|
+
def path(handler_class, with_method: :any, **rest)
|
120
|
+
route = self.route_for(handler_class, with_method:)
|
121
|
+
route.path(**rest)
|
122
|
+
end
|
123
|
+
|
124
|
+
def url(handler_class, with_method: :any, **rest)
|
125
|
+
route = self.route_for(handler_class, with_method:)
|
126
|
+
route.url(**rest)
|
127
|
+
end
|
128
|
+
|
129
|
+
def inspect
|
130
|
+
@routes.map { |route|
|
131
|
+
"#{route.http_method}:#{route.path_template} - #{route.handler_class.name}"
|
132
|
+
}.join("\n")
|
133
|
+
end
|
134
|
+
|
135
|
+
private
|
136
|
+
|
137
|
+
def route_for(handler_class, with_method: :any)
|
120
138
|
route = self.route(handler_class)
|
121
139
|
route_allowed_for_method = if with_method == :any
|
122
140
|
true
|
@@ -128,14 +146,9 @@ class Brut::FrontEnd::Routing
|
|
128
146
|
if !route_allowed_for_method
|
129
147
|
raise ArgumentError,"The route for '#{handler_class}' (#{route.path}) is not supported by HTTP method '#{with_method}'"
|
130
148
|
end
|
131
|
-
route
|
149
|
+
route
|
132
150
|
end
|
133
151
|
|
134
|
-
def inspect
|
135
|
-
@routes.map { |route|
|
136
|
-
"#{route.http_method}:#{route.path_template} - #{route.handler_class.name}"
|
137
|
-
}.join("\n")
|
138
|
-
end
|
139
152
|
|
140
153
|
def add_routing_method(route)
|
141
154
|
handler_class = route.handler_class
|
@@ -143,7 +156,10 @@ class Brut::FrontEnd::Routing
|
|
143
156
|
[ handler_class, form_class ].compact.each do |klass|
|
144
157
|
klass.class_eval do
|
145
158
|
def self.routing(**args)
|
146
|
-
Brut.container.routing.
|
159
|
+
Brut.container.routing.path(self,**args)
|
160
|
+
end
|
161
|
+
def self.full_routing(**args)
|
162
|
+
Brut.container.routing.url(self,**args)
|
147
163
|
end
|
148
164
|
end
|
149
165
|
end
|
@@ -205,11 +221,31 @@ class Brut::FrontEnd::Routing
|
|
205
221
|
uri
|
206
222
|
end
|
207
223
|
|
224
|
+
def url(**query_string_params)
|
225
|
+
request_context = Thread.current.thread_variable_get(:request_context)
|
226
|
+
path = self.path(**query_string_params)
|
227
|
+
host = if request_context
|
228
|
+
request_context[:host]
|
229
|
+
end
|
230
|
+
host ||= Brut.container.fallback_host
|
231
|
+
host.merge(path)
|
232
|
+
rescue ArgumentError => ex
|
233
|
+
request_context_note = if request_context
|
234
|
+
"the RequestContext did not contain request.host, which is unusual"
|
235
|
+
else
|
236
|
+
"the RequestContext was not available (likely due to calling `full_routing` outside an HTTP request)"
|
237
|
+
end
|
238
|
+
raise Brut::Framework::Errors::MissingConfiguration(
|
239
|
+
:fallback_host,
|
240
|
+
"Attempting to get the full URL for route #{self.path_template}, #{request_context_note}, and Brut.container.fallback_host was not set. You must set this value if you expect to generate complete URLs outside of an HTTP request")
|
241
|
+
end
|
242
|
+
|
208
243
|
def ==(other)
|
209
244
|
self.method == other.method && self.path == other.path
|
210
245
|
end
|
211
246
|
|
212
247
|
private
|
248
|
+
|
213
249
|
def locate_handler_class(suffix,preposition, on_missing: :raise)
|
214
250
|
if @path_template == "/"
|
215
251
|
return Module.const_get("HomePage") # XXX Needs error handling
|
@@ -2,10 +2,23 @@
|
|
2
2
|
module Brut::SpecSupport::ClockSupport
|
3
3
|
# Return a real lock in UTC
|
4
4
|
def real_clock = Clock.new(TZInfo::Timezone.get("UTC"))
|
5
|
-
|
5
|
+
|
6
|
+
# Return a clock whose value for now is `now`, in UTC
|
6
7
|
#
|
7
8
|
# @param [String] now a string containing the value you want for {Clock#now} to return.
|
8
9
|
def clock_at(now:)
|
9
|
-
|
10
|
+
self.clock_in_timezone_at(timezone_name: "UTC", now: now)
|
11
|
+
end
|
12
|
+
|
13
|
+
# Return a clock whose value for now is `now` in the given timezone
|
14
|
+
#
|
15
|
+
# @param [String] timezone_name a string that is the name of the timezone to use.
|
16
|
+
# @param [String] now a string containing the value you want for {Clock#now} to return.
|
17
|
+
def clock_in_timezone_at(timezone_name:, now:)
|
18
|
+
time = Time.parse(now)
|
19
|
+
timezone = TZInfo::Timezone.get(timezone_name)
|
20
|
+
same_time_in_timezone = timezone.local_time(time.year, time.month, time.day, time.hour, time.min, time.sec)
|
21
|
+
|
22
|
+
Clock.new(TZInfo::Timezone.get(timezone_name), now: same_time_in_timezone)
|
10
23
|
end
|
11
24
|
end
|
@@ -78,7 +78,7 @@ module Brut::SpecSupport::ComponentSupport
|
|
78
78
|
|
79
79
|
# @!visibility private
|
80
80
|
def routing_for(klass,**args)
|
81
|
-
Brut.container.routing.
|
81
|
+
Brut.container.routing.path(klass,**args)
|
82
82
|
end
|
83
83
|
|
84
84
|
# Escape HTML using the same code Brut uses for rendering templates.
|
@@ -5,6 +5,16 @@ module Brut::SpecSupport::GeneralSupport
|
|
5
5
|
end
|
6
6
|
|
7
7
|
module ClassMethods
|
8
|
+
def implementation_is_needed(check_again_at:)
|
9
|
+
check_again_at = if check_again_at.kind_of?(Time)
|
10
|
+
check_again_at
|
11
|
+
else
|
12
|
+
check_again_at = Date.parse(check_again_at).to_time
|
13
|
+
end
|
14
|
+
it "has no tests for now, but they are needed eventually" do
|
15
|
+
expect(Time.now < check_again_at).to eq(true),"It's after #{check_again_at}. Implementation is needed"
|
16
|
+
end
|
17
|
+
end
|
8
18
|
# To pass bin/test audit with a class whose implementation is trivial, call this inside the RSpec `describe` block. This is better
|
9
19
|
# than an empty test as it makes it more explicit that you believe the implementation is trivial enough to not require a test. You
|
10
20
|
# can also set an expiration for this thinking.
|
@@ -23,7 +33,7 @@ module Brut::SpecSupport::GeneralSupport
|
|
23
33
|
if check_again_at.nil?
|
24
34
|
expect(true).to eq(true)
|
25
35
|
else
|
26
|
-
expect(Time.now < check_again_at).to eq(true),"
|
36
|
+
expect(Time.now < check_again_at).to eq(true),"It's after #{check_again_at}. Check that the implementation of the class under test is still trivial. If it is, update or remove check_again_at:"
|
27
37
|
end
|
28
38
|
end
|
29
39
|
end
|
@@ -1,10 +1,10 @@
|
|
1
1
|
RSpec::Matchers.define :be_routing_for do |klass,**args|
|
2
2
|
match do |uri|
|
3
|
-
uri == Brut.container.routing.
|
3
|
+
uri == Brut.container.routing.path(klass,**args)
|
4
4
|
end
|
5
5
|
|
6
6
|
failure_message do |uri|
|
7
|
-
expected = Brut.container.routing.
|
7
|
+
expected = Brut.container.routing.path(klass,**args)
|
8
8
|
"Expected route for #{klass}: #{expected}, but got #{uri}"
|
9
9
|
end
|
10
10
|
|
@@ -103,6 +103,7 @@ class Brut::SpecSupport::RSpecSetup
|
|
103
103
|
flash: empty_flash,
|
104
104
|
body: nil,
|
105
105
|
xhr: false,
|
106
|
+
host: URI("https://example.com")
|
106
107
|
)
|
107
108
|
Thread.current.thread_variable_set(:request_context, request_context)
|
108
109
|
example.example_group.let(:request_context) { request_context }
|
data/lib/brut/version.rb
CHANGED
data/lib/brut.rb
CHANGED
@@ -50,6 +50,7 @@ module Brut
|
|
50
50
|
# will be in the back end, Brut is far less prescriptive about how to manage that than it is the front end.
|
51
51
|
module BackEnd
|
52
52
|
autoload(:Validators, "brut/back_end/validator")
|
53
|
+
autoload(:Sidekiq, "brut/back_end/sidekiq")
|
53
54
|
# Do not put SeedData here - it must be loaded only when needed
|
54
55
|
end
|
55
56
|
# I18n is where internationalization and localization support lives.
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: brut
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.0.
|
4
|
+
version: 0.0.9
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- David Bryant Copeland
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-
|
10
|
+
date: 2025-04-02 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: irb
|
@@ -455,6 +455,10 @@ files:
|
|
455
455
|
- dx/stop
|
456
456
|
- lib/brut.rb
|
457
457
|
- lib/brut/back_end/seed_data.rb
|
458
|
+
- lib/brut/back_end/sidekiq.rb
|
459
|
+
- lib/brut/back_end/sidekiq/middlewares.rb
|
460
|
+
- lib/brut/back_end/sidekiq/middlewares/server.rb
|
461
|
+
- lib/brut/back_end/sidekiq/middlewares/server/flush_spans.rb
|
458
462
|
- lib/brut/back_end/validator.rb
|
459
463
|
- lib/brut/back_end/validators/form_validator.rb
|
460
464
|
- lib/brut/cli.rb
|
@@ -478,6 +482,7 @@ files:
|
|
478
482
|
- lib/brut/framework/errors.rb
|
479
483
|
- lib/brut/framework/errors/abstract_method.rb
|
480
484
|
- lib/brut/framework/errors/bug.rb
|
485
|
+
- lib/brut/framework/errors/missing_configuration.rb
|
481
486
|
- lib/brut/framework/errors/missing_parameter.rb
|
482
487
|
- lib/brut/framework/errors/no_class_for_path.rb
|
483
488
|
- lib/brut/framework/errors/not_found.rb
|