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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 37f9d27495d7269447632af6383a1f4e3f24f98bbfcc1838d1a118c09005815f
4
- data.tar.gz: ef05ae07379e79cef2914b16567b5d57856cb402c152e823ce658b4bc4cb8a69
3
+ metadata.gz: b455ac9d87f61b5baa45cc6f4cc11c3b940b015aaf7792ba5ed8229dfd23d2fc
4
+ data.tar.gz: 5e6216b69db294da81ed31041a4b65a4a2b04de229e3d8695e5c9e1ad1625a84
5
5
  SHA512:
6
- metadata.gz: ba995eb4d298f52064d3866eb7712a8cdf62e92e2e1415fb5545a8f8a8e59c3861da28861175c0b83df8b71428a235404b5170e188faab9f1c2ab18d6e3dfc73
7
- data.tar.gz: 2a4ccedddbc0c8ab977124e40a59f12aaae21e7ba140dd7580a709ae17c8de8b8da49d59db473f8270baf8a34c30e36c96e1d937d6b6ce63bec08a6d88e385a3
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 noninteractive
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
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- brut (0.0.7)
4
+ brut (0.0.9)
5
5
  concurrent-ruby
6
6
  i18n
7
7
  irb
@@ -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
@@ -0,0 +1,3 @@
1
+ class Brut::BackEnd::Sidekiq::Middlewares::Server
2
+ autoload(:FlushSpans, "brut/back_end/sidekiq/middlewares/server/flush_spans")
3
+ end
@@ -0,0 +1,3 @@
1
+ class Brut::BackEnd::Sidekiq::Middlewares
2
+ autoload(:Server, "brut/back_end/sidekiq/middlewares/server")
3
+ end
@@ -0,0 +1,3 @@
1
+ class Brut::BackEnd::Sidekiq
2
+ autoload(:Middlewares, "brut/back_end/sidekiq/middlewares")
3
+ 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
- as_execution_result(command.execute)
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
@@ -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
- Sequel::Migrator.run(Brut.container.sequel_db_handle,migrations_dir)
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
@@ -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
- out.puts "Rebuilding test database schema"
44
- system! "bin/db rebuild --env=test"
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
- if args.empty?
47
- out.puts "Running all tests"
48
- system! "#{rspec_command} #{Brut.container.app_specs_dir}/"
49
- else
50
- test_args = args.map { |_|
51
- '"' + Shellwords.escape(_) + '"'
52
- }.join(" ")
53
- system! "#{rspec_command} #{test_args}"
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
- out.puts "Re-Rebuilding test database schema"
57
- system! "bin/db rebuild --env=test"
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}
@@ -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 intance and immediately call `boot!`. This is what happens most of the time, in particular when the app is started
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
- attr_writer :yielded_block
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.uri(klass,with_method: :get,**query_string_params)
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.
@@ -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
- def initialize(env:,session:,flash:,xhr:,body:)
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 uri(handler_class, with_method: :any, **rest)
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.path(**rest)
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.uri(self,**args)
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
- # Return a clock whose value for now is `now`
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
- Clock.new(TZInfo::Timezone.get("UTC"), now: Time.parse(now))
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
- raise "#{component.class} returned a #{rendered_text.class} - you should not attempt to parse this. Instead, call render(component)"
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
- Brut::SpecSupport::EnhancedNode.new(nokogiri_node)
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.uri(klass,**args)
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
- return Brut::SpecSupport::EnhancedNode.new(element.first)
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),"I'ts 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:"
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.uri(klass,**args)
3
+ uri == Brut.container.routing.path(klass,**args)
4
4
  end
5
5
 
6
6
  failure_message do |uri|
7
- expected = Brut.container.routing.uri(klass,**args)
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
@@ -1,4 +1,4 @@
1
1
  module Brut
2
2
  # @!visibility private
3
- VERSION = "0.0.7"
3
+ VERSION = "0.0.9"
4
4
  end
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.7
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-03-02 00:00:00.000000000 Z
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.4
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