brut 0.0.7 → 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/Dockerfile.dx +1 -1
- 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/component.rb +6 -3
- data/lib/brut/front_end/handling_results.rb +2 -3
- data/lib/brut/front_end/page.rb +3 -0
- data/lib/brut/front_end/request_context.rb +3 -1
- data/lib/brut/front_end/route_hooks/csp_no_inline_scripts.rb +2 -0
- data/lib/brut/front_end/route_hooks/csp_no_inline_styles_or_scripts.rb +8 -0
- data/lib/brut/front_end/route_hooks/setup_request_context.rb +2 -1
- data/lib/brut/front_end/routing.rb +48 -11
- data/lib/brut/spec_support/clock_support.rb +15 -2
- data/lib/brut/spec_support/component_support.rb +11 -3
- data/lib/brut/spec_support/enhanced_node.rb +7 -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 +8 -3
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/Dockerfile.dx
CHANGED
@@ -2,7 +2,7 @@ FROM ruby:3.3
|
|
2
2
|
|
3
3
|
SHELL [ "/bin/bash", "-o", "pipefail", "-c" ]
|
4
4
|
|
5
|
-
ENV DEBIAN_FRONTEND
|
5
|
+
ENV DEBIAN_FRONTEND=noninteractive
|
6
6
|
|
7
7
|
# These packages are needed to set up other repos to install other
|
8
8
|
# packages and/or are useful in installing other software
|
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(
|
@@ -52,8 +52,10 @@ class Brut::FrontEnd::Component
|
|
52
52
|
end
|
53
53
|
end
|
54
54
|
|
55
|
-
# Allows helpers that create components to pass the block they were given to the component
|
56
|
-
|
55
|
+
# Allows helpers that create components to pass the block they were given to the component.
|
56
|
+
# This can be read for the purposes of nested components passing a yielded block to an inner
|
57
|
+
# component
|
58
|
+
attr_accessor :yielded_block
|
57
59
|
|
58
60
|
# Intended to be called by subclasses to render the yielded block wherever it makes sense in their markup.
|
59
61
|
def render_yielded_block
|
@@ -184,11 +186,12 @@ class Brut::FrontEnd::Component
|
|
184
186
|
end
|
185
187
|
|
186
188
|
# Render the {Brut::FrontEnd::Components::ConstraintViolations} component for the given form's input.
|
187
|
-
def constraint_violations(form:, input_name:, message_html_attributes: {}, **html_attributes)
|
189
|
+
def constraint_violations(form:, input_name:, index: nil, message_html_attributes: {}, **html_attributes)
|
188
190
|
component(
|
189
191
|
Brut::FrontEnd::Components::ConstraintViolations.new(
|
190
192
|
form:,
|
191
193
|
input_name:,
|
194
|
+
index:,
|
192
195
|
message_html_attributes:,
|
193
196
|
**html_attributes
|
194
197
|
)
|
@@ -7,8 +7,7 @@ module Brut::FrontEnd::HandlingResults
|
|
7
7
|
# @param [Class] klass A page or handler class whose route should be redirected-to. Note that if parameters are required, they must
|
8
8
|
# be provided in `query_string_params` or this will raise an error. Note that the class must be for a GET route, since you cannot
|
9
9
|
# redirect to a non-GET.
|
10
|
-
# @param [Hash] query_string_params arguments and parameters for the route. Any values that correspond to route parameters will be
|
11
|
-
# used to build the route. Remaining will be used as query parameters.
|
10
|
+
# @param [Hash] query_string_params arguments and parameters for the route. Any values that correspond to route parameters will be used to build the route. A value of 'anchor' will be used as the hash/anchor part of the URL and should not contain a hash sign. Remaining will be used as query parameters.
|
12
11
|
#
|
13
12
|
# @raise [ArgumentError] if `klass` is not a `Class` or if `klass` is not for a `GET`
|
14
13
|
# @raise [Brut::Framework::Errors::MissingParameter] if any required route parameters were not provided
|
@@ -16,7 +15,7 @@ module Brut::FrontEnd::HandlingResults
|
|
16
15
|
if !klass.kind_of?(Class)
|
17
16
|
raise ArgumentError,"redirect_to should be given a Class, not a #{klass.class}"
|
18
17
|
end
|
19
|
-
Brut.container.routing.
|
18
|
+
Brut.container.routing.path(klass,with_method: :get,**query_string_params)
|
20
19
|
end
|
21
20
|
|
22
21
|
# Return this to return an HTTP status code from a number or string containing the code.
|
data/lib/brut/front_end/page.rb
CHANGED
@@ -19,6 +19,9 @@ class Brut::FrontEnd::Page < Brut::FrontEnd::Component
|
|
19
19
|
# Returns the name of the layout for this page. This string is used to find an ERB file in `app/src/front_end/layouts`. Every page
|
20
20
|
# must have a layout. If you wish to render a page with no layout, create an empty layout in your app and use that.
|
21
21
|
#
|
22
|
+
# Note that the layout can be dynamic. It is requested when {#render} is called, so you can override this
|
23
|
+
# method and use any ivar set in your constructor to change what layout is used.
|
24
|
+
#
|
22
25
|
# @return [String] The name of the layout. May not be `nil`.
|
23
26
|
def layout = "default"
|
24
27
|
|
@@ -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
|
}
|
@@ -13,6 +13,8 @@ class Brut::FrontEnd::RouteHooks::CSPNoInlineScripts < Brut::FrontEnd::RouteHook
|
|
13
13
|
private
|
14
14
|
|
15
15
|
def header_value
|
16
|
+
x = "sha256-d21f9b773d3cfad25f041a96ab08376e944e9ad4843f2060ecbdbefb12d91b1d"
|
17
|
+
x = "sha256-0h+bdz08+tJfBBqWqwg3bpROmtSEPyBg7L2++xLZGx0="
|
16
18
|
[
|
17
19
|
"default-src 'self'",
|
18
20
|
"script-src-elem 'self'",
|
@@ -8,6 +8,12 @@ class Brut::FrontEnd::RouteHooks::CSPNoInlineStylesOrScripts < Brut::FrontEnd::R
|
|
8
8
|
continue
|
9
9
|
end
|
10
10
|
|
11
|
+
# TODO: A way for app to pass in stuff to modify this in part.
|
12
|
+
# In particular a hash for a <style>, calculated as follows:
|
13
|
+
# - textContent is sha265'ed
|
14
|
+
# - that is Base64'ed
|
15
|
+
# - that is put as 'sha256–«base64edvalue»` into the directive
|
16
|
+
|
11
17
|
# Sets content security policy headers that only report the use inline scripts and inline styles, but do allow them.
|
12
18
|
# This is useful for existing apps where you want to migrate to a more secure policy, but cannot.
|
13
19
|
# @see Brut::FrontEnd::Handlers::CspReportingHandler
|
@@ -26,6 +32,8 @@ class Brut::FrontEnd::RouteHooks::CSPNoInlineStylesOrScripts < Brut::FrontEnd::R
|
|
26
32
|
private
|
27
33
|
|
28
34
|
def header_value
|
35
|
+
x = "sha256-d21f9b773d3cfad25f041a96ab08376e944e9ad4843f2060ecbdbefb12d91b1d"
|
36
|
+
x = "sha256-0h+bdz08+tJfBBqWqwg3bpROmtSEPyBg7L2++xLZGx0="
|
29
37
|
[
|
30
38
|
"default-src 'self'",
|
31
39
|
"script-src-elem 'self'",
|
@@ -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,25 +146,20 @@ 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
|
142
|
-
if handler_class.respond_to?(:routing) && handler_class.method(:routing).owner != Brut::FrontEnd::Form
|
143
|
-
raise ArgumentError,"#{handler_class} (that handles path #{route.path_template}) got it's ::routing method from #{handler_class.method(:routing).owner}, meaning it has overridden the value fro Brut::FrontEnd::Form"
|
144
|
-
end
|
145
155
|
form_class = route.respond_to?(:form_class) ? route.form_class : nil
|
146
156
|
[ handler_class, form_class ].compact.each do |klass|
|
147
157
|
klass.class_eval do
|
148
158
|
def self.routing(**args)
|
149
|
-
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)
|
150
163
|
end
|
151
164
|
end
|
152
165
|
end
|
@@ -180,6 +193,7 @@ class Brut::FrontEnd::Routing
|
|
180
193
|
end
|
181
194
|
|
182
195
|
def path(**query_string_params)
|
196
|
+
anchor = query_string_params.delete(:anchor) || query_string_params.delete("anchor")
|
183
197
|
path = @path_template.split(/\//).map { |path_part|
|
184
198
|
if path_part =~ /^:(.+)$/
|
185
199
|
param_name = $1.to_sym
|
@@ -199,16 +213,39 @@ class Brut::FrontEnd::Routing
|
|
199
213
|
if joined_path == ""
|
200
214
|
joined_path = "/"
|
201
215
|
end
|
216
|
+
if anchor
|
217
|
+
joined_path = joined_path + "#" + URI.encode_uri_component(anchor)
|
218
|
+
end
|
202
219
|
uri = URI(joined_path)
|
203
220
|
uri.query = URI.encode_www_form(query_string_params)
|
204
221
|
uri
|
205
222
|
end
|
206
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
|
+
|
207
243
|
def ==(other)
|
208
244
|
self.method == other.method && self.path == other.path
|
209
245
|
end
|
210
246
|
|
211
247
|
private
|
248
|
+
|
212
249
|
def locate_handler_class(suffix,preposition, on_missing: :raise)
|
213
250
|
if @path_template == "/"
|
214
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
|
@@ -42,7 +42,11 @@ module Brut::SpecSupport::ComponentSupport
|
|
42
42
|
def render_and_parse(component,&block)
|
43
43
|
rendered_text = render(component,&block)
|
44
44
|
if !rendered_text.kind_of?(String) && !rendered_text.kind_of?(Brut::FrontEnd::Templates::HTMLSafeString)
|
45
|
-
|
45
|
+
if rendered_text.kind_of?(URI::Generic)
|
46
|
+
raise "#{component.class} redirected to #{rendered_text} instead of rendering"
|
47
|
+
else
|
48
|
+
raise "#{component.class} returned a #{rendered_text.class} - you should not attempt to parse this. Instead, call render(component)"
|
49
|
+
end
|
46
50
|
end
|
47
51
|
nokogiri_node = Nokogiri::HTML5(rendered_text)
|
48
52
|
if !component.kind_of?(Brut::FrontEnd::Page)
|
@@ -65,12 +69,16 @@ module Brut::SpecSupport::ComponentSupport
|
|
65
69
|
end
|
66
70
|
nokogiri_node = non_blank_text_elements[0]
|
67
71
|
end
|
68
|
-
|
72
|
+
if nokogiri_node
|
73
|
+
Brut::SpecSupport::EnhancedNode.new(nokogiri_node)
|
74
|
+
else
|
75
|
+
nil
|
76
|
+
end
|
69
77
|
end
|
70
78
|
|
71
79
|
# @!visibility private
|
72
80
|
def routing_for(klass,**args)
|
73
|
-
Brut.container.routing.
|
81
|
+
Brut.container.routing.path(klass,**args)
|
74
82
|
end
|
75
83
|
|
76
84
|
# Escape HTML using the same code Brut uses for rendering templates.
|
@@ -12,7 +12,12 @@ class Brut::SpecSupport::EnhancedNode < SimpleDelegator
|
|
12
12
|
element = css(css_selector)
|
13
13
|
if (element.kind_of?(Nokogiri::XML::NodeSet))
|
14
14
|
expect(element.length).to be < 2
|
15
|
-
|
15
|
+
first_element = element.first
|
16
|
+
if first_element
|
17
|
+
return Brut::SpecSupport::EnhancedNode.new(first_element)
|
18
|
+
else
|
19
|
+
return nil
|
20
|
+
end
|
16
21
|
else
|
17
22
|
expect([Nokogiri::XML::Node, Nokogiri::XML::Element]).to include(element.class)
|
18
23
|
return Brut::SpecSupport::EnhancedNode.new(element)
|
@@ -37,6 +42,7 @@ class Brut::SpecSupport::EnhancedNode < SimpleDelegator
|
|
37
42
|
def first!(css_selector)
|
38
43
|
element = css(css_selector)
|
39
44
|
if (element.kind_of?(Nokogiri::XML::NodeSet))
|
45
|
+
expect(element.first).not_to eq(nil), "No elements matching #{css_selector}"
|
40
46
|
return Brut::SpecSupport::EnhancedNode.new(element.first)
|
41
47
|
else
|
42
48
|
expect([Nokogiri::XML::Node, Nokogiri::XML::Element]).to include(element.class)
|
@@ -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
|
@@ -605,7 +610,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
605
610
|
- !ruby/object:Gem::Version
|
606
611
|
version: '0'
|
607
612
|
requirements: []
|
608
|
-
rubygems_version: 3.6.
|
613
|
+
rubygems_version: 3.6.6
|
609
614
|
specification_version: 4
|
610
615
|
summary: NOT YET RELEASED - Web Framework Built around Ruby, Web Standards, Simplicity,
|
611
616
|
and Object-Orientation
|