brick 1.0.190 → 1.0.192

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.
@@ -15,4 +15,10 @@ module ::Brick::Rails
15
15
 
16
16
  AA_PNG = "<img src=\"\">
17
17
  "
18
+
19
+ IN_APP = "<svg height=\"36px\" viewBox=\"0 0 214 274\" xmlns=\"http://www.w3.org/2000/svg\">
20
+ <g transform=\"matrix(4.16667,0,0,4.16667,-1049.47,-789.371)\">
21
+ <path d=\"M281.386,193.134L287.086,193.134C287.433,193.134 287.716,193.417 287.716,193.764C287.716,194.11 287.433,194.394 287.086,194.394L281.386,194.394C281.04,194.394 280.756,194.11 280.756,193.764C280.756,193.417 281.04,193.134 281.386,193.134ZM284.252,245.638C282.456,245.638 281.008,247.086 281.008,248.851C281.008,250.645 282.456,252.094 284.252,252.094C286.047,252.094 287.496,250.645 287.496,248.851C287.496,247.086 286.047,245.638 284.252,245.638ZM284.252,246.331C282.835,246.331 281.701,247.465 281.701,248.851C281.701,250.268 282.835,251.401 284.252,251.401C285.638,251.401 286.803,250.268 286.803,248.851C286.803,247.465 285.638,246.331 284.252,246.331ZM275.843,208.63L287.559,218.11L275.843,227.622L275.843,221.858L251.874,221.858L251.874,214.394L275.843,214.394L275.843,208.63ZM278.866,239.906L278.866,233.102C278.866,232.851 278.677,232.693 278.456,232.693L271.622,232.693C271.401,232.693 271.212,232.851 271.212,233.102L271.212,239.906C271.212,240.157 271.401,240.314 271.622,240.314L278.456,240.314C278.677,240.314 278.866,240.157 278.866,239.906ZM280.41,233.102L280.41,239.906C280.41,240.157 280.599,240.314 280.819,240.314L287.653,240.314C287.874,240.314 288.063,240.157 288.063,239.906L288.063,233.102C288.063,232.851 287.874,232.693 287.653,232.693L280.819,232.693C280.599,232.693 280.41,232.851 280.41,233.102ZM289.606,233.102L289.606,239.906C289.606,240.157 289.795,240.314 290.047,240.314L296.851,240.314C297.071,240.314 297.26,240.157 297.26,239.906L297.26,233.102C297.26,232.851 297.071,232.693 296.851,232.693L290.047,232.693C289.795,232.693 289.606,232.851 289.606,233.102ZM269.197,189.449L299.276,189.449C301.354,189.449 303.055,191.149 303.055,193.197L303.055,251.275C303.055,253.355 301.354,255.055 299.276,255.055L269.197,255.055C267.118,255.055 265.449,253.355 265.449,251.275L265.449,223.496L267.842,223.496L267.842,242.488L300.63,242.488L300.63,198.394L267.842,198.394L267.842,212.725L265.449,212.725L265.449,193.197C265.449,191.149 267.118,189.449 269.197,189.449Z\" style=\"fill:rgb(31,147,209);\"/>
22
+ </g>
23
+ </svg>"
18
24
  end
@@ -0,0 +1,348 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Brick
4
+ class << self
5
+ attr_accessor :routes_done
6
+ end
7
+
8
+ module RouteMapper
9
+ def add_brick_routes
10
+ routeset_to_use = ::Rails.application.routes
11
+ path_prefix = ::Brick.config.path_prefix
12
+ existing_controllers = routeset_to_use.routes.each_with_object({}) do |r, s|
13
+ if (r.verb == 'GET' || (r.verb.is_a?(Regexp) && r.verb.source == '^GET$')) &&
14
+ (controller_name = r.defaults[:controller])
15
+ path = r.path.ast.to_s
16
+ path = path[0..((path.index('(') || 0) - 1)]
17
+ # Skip adding this if it's the default_route_fallback set from the initializers/brick.rb file
18
+ next if "#{path}##{r.defaults[:action]}" == ::Brick.established_drf ||
19
+ # or not a GET request
20
+ [:index, :show, :new, :edit].exclude?(action = r.defaults[:action].to_sym)
21
+
22
+ # Attempt to backtrack to original
23
+ c_parts = controller_name.split('/')
24
+ while c_parts.length > 0
25
+ c_dotted = c_parts.join('.')
26
+ if (relation = ::Brick.relations.fetch(c_dotted, nil)) # Does it match up with an existing Brick table / resource name?
27
+ # puts path
28
+ # puts " #{c_dotted}##{r.defaults[:action]}"
29
+ if (route_name = r.name&.to_sym) != :root
30
+ relation[:existing][action] = route_name
31
+ else
32
+ relation[:existing][action] ||= path
33
+ end
34
+ s[c_dotted.tr('.', '/')] = nil
35
+ break
36
+ end
37
+ c_parts.shift
38
+ end
39
+ s[controller_name] = nil if c_parts.length.zero?
40
+ end
41
+ end
42
+
43
+ tables = []
44
+ views = []
45
+ table_class_length = 38 # Length of "Classes that can be built from tables:"
46
+ view_class_length = 37 # Length of "Classes that can be built from views:"
47
+
48
+ brick_namespace_create = lambda do |path_names, res_name, options|
49
+ if path_names&.present?
50
+ if (path_name = path_names.pop).is_a?(Array)
51
+ module_name = path_name[1]
52
+ path_name = path_name.first
53
+ end
54
+ send(:scope, { module: module_name || path_name, path: path_name, as: path_name }) do
55
+ brick_namespace_create.call(path_names, res_name, options)
56
+ end
57
+ else
58
+ send(:resources, res_name.to_sym, **options)
59
+ end
60
+ end
61
+
62
+ # %%% TODO: If no auto-controllers then enumerate the controllers folder in order to build matching routes
63
+ # If auto-controllers and auto-models are both enabled then this makes sense:
64
+ controller_prefix = (path_prefix ? "#{path_prefix}/" : '')
65
+ sti_subclasses = ::Brick.config.sti_namespace_prefixes.each_with_object(Hash.new { |h, k| h[k] = [] }) do |v, s|
66
+ # Turn something like {"::Spouse"=>"Person", "::Friend"=>"Person"} into {"Person"=>["Spouse", "Friend"]}
67
+ s[v.last] << v.first[2..-1] unless v.first.end_with?('::')
68
+ end
69
+ versioned_views = {} # Track which views have already been done for each api_root
70
+ ::Brick.relations.each do |k, v|
71
+ next if k.is_a?(Symbol)
72
+
73
+ if (schema_name = v.fetch(:schema, nil))
74
+ schema_prefix = "#{schema_name}."
75
+ end
76
+
77
+ resource_name = v.fetch(:resource, nil)
78
+ next if !resource_name ||
79
+ existing_controllers.key?(
80
+ controller_prefix + (resource_name = "#{schema_prefix&.tr('.', '/')}#{resource_name}".pluralize)
81
+ )
82
+
83
+ object_name = k.split('.').last # Take off any first schema part
84
+
85
+ full_schema_prefix = if (full_aps = aps = v.fetch(:auto_prefixed_schema, nil))
86
+ aps = aps[0..-2] if aps[-1] == '_'
87
+ (schema_prefix&.dup || +'') << "#{aps}."
88
+ else
89
+ schema_prefix
90
+ end
91
+
92
+ # Track routes being built
93
+ resource_name = v.fetch(:resource, nil) || k
94
+ if (class_name = v.fetch(:class_name, nil))
95
+ if v.key?(:isView)
96
+ view_class_length = class_name.length if class_name.length > view_class_length
97
+ views
98
+ else
99
+ table_class_length = class_name.length if class_name.length > table_class_length
100
+ tables
101
+ end << [class_name, aps, "#{"#{schema_name}/" if schema_name}#{resource_name[full_aps&.length || 0 .. -1]}"]
102
+ end
103
+
104
+ options = {}
105
+ options[:only] = [:index, :show] if v.key?(:isView)
106
+
107
+ # First do the normal routes
108
+ prefixes = []
109
+ prefixes << [aps, v[:class_name]&.split('::')[-2]&.underscore] if aps
110
+ prefixes << schema_name if schema_name
111
+ prefixes << path_prefix if path_prefix
112
+ brick_namespace_create.call(prefixes, resource_name, options)
113
+ sti_subclasses.fetch(class_name, nil)&.each do |sc| # Add any STI subclass routes for this relation
114
+ brick_namespace_create.call(prefixes, sc.underscore.tr('/', '_').pluralize, options)
115
+ end
116
+
117
+ # Now the API routes if necessary
118
+ full_resource = nil
119
+ ::Brick.api_roots&.each do |api_root|
120
+ api_done_views = (versioned_views[api_root] ||= {})
121
+ found = nil
122
+ test_ver_num = nil
123
+ view_relation = nil
124
+ # If it's a view then see if there's a versioned one available by searching for resource names
125
+ # versioned with the closest number (equal to or less than) compared with our API version number.
126
+ if v.key?(:isView)
127
+ if (ver = object_name.match(/^v([\d_]*)/)&.captures&.first) && ver[-1] == '_'
128
+ core_object_name = object_name[ver.length + 1..-1]
129
+ next if api_done_views.key?(unversioned = "#{schema_prefix}v_#{core_object_name}")
130
+
131
+ # Expect that the last item in the path generally holds versioning information
132
+ api_ver = api_root.split('/')[-1]&.gsub('_', '.')
133
+ vn_idx = api_ver.rindex(/[^\d._]/) # Position of the first numeric digit at the end of the version number
134
+ # Was: .to_d
135
+ test_ver_num = api_ver_num = api_ver[vn_idx + 1..-1].gsub('_', '.').to_i # Attempt to turn something like "v3" into the decimal value 3
136
+ # puts [api_ver, vn_idx, api_ver_num, unversioned].inspect
137
+
138
+ next if ver.to_i > api_ver_num # Don't surface any newer views in an older API
139
+
140
+ test_ver_num -= 1 until test_ver_num.zero? ||
141
+ (view_relation = ::Brick.relations.fetch(
142
+ found = "#{schema_prefix}v#{test_ver_num}_#{core_object_name}", nil
143
+ ))
144
+ api_done_views[unversioned] = nil # Mark that for this API version this view is done
145
+
146
+ # puts "Found #{found}" if view_relation
147
+ # If we haven't found "v3_view_name" or "v2_view_name" or so forth, at the last
148
+ # fall back to simply looking for "v_view_name", and then finally "view_name".
149
+ no_v_prefix_name = "#{schema_prefix}#{core_object_name}"
150
+ standard_prefix = 'v_'
151
+ else
152
+ core_object_name = object_name
153
+ end
154
+ if (rvp = ::Brick.config.api_remove_view_prefix) && core_object_name.start_with?(rvp)
155
+ core_object_name.slice!(0, rvp.length)
156
+ end
157
+ no_prefix_name = "#{schema_prefix}#{core_object_name}"
158
+ unversioned = "#{schema_prefix}#{standard_prefix}#{::Brick.config.api_add_view_prefix}#{core_object_name}"
159
+ else
160
+ unversioned = k
161
+ end
162
+
163
+ view_relation ||= ::Brick.relations.fetch(found = unversioned, nil) ||
164
+ (no_v_prefix_name && ::Brick.relations.fetch(found = no_v_prefix_name, nil)) ||
165
+ (no_prefix_name && ::Brick.relations.fetch(found = no_prefix_name, nil))
166
+ if view_relation
167
+ actions = view_relation.key?(:isView) ? [:index, :show] : ::Brick::ALL_API_ACTIONS # By default all actions are allowed
168
+ # Call proc that limits which endpoints get surfaced based on version, table or view name, method (get list / get one / post / patch / delete)
169
+ # Returning nil makes it do nothing, false makes it skip creating this endpoint, and an array of up to
170
+ # these 3 things controls and changes the nature of the endpoint that gets built:
171
+ # (updated api_name, name of different relation to route to, allowed actions such as :index, :show, :create, etc)
172
+ proc_result = if (filter = ::Brick.config.api_filter).is_a?(Proc)
173
+ begin
174
+ num_args = filter.arity.negative? ? 6 : filter.arity
175
+ filter.call(*[unversioned, k, view_relation, actions, api_ver_num, found, test_ver_num][0...num_args])
176
+ rescue StandardError => e
177
+ puts "::Brick.api_filter Proc error: #{e.message}"
178
+ end
179
+ end
180
+ # proc_result expects to receive back: [updated_api_name, to_other_relation, allowed_actions]
181
+
182
+ case proc_result
183
+ when NilClass
184
+ # Do nothing differently than what normal behaviour would be
185
+ when FalseClass # Skip implementing this endpoint
186
+ view_relation[:api][api_ver_num] = nil
187
+ next
188
+ when Array # Did they give back an array of actions?
189
+ unless proc_result.any? { |pr| ::Brick::ALL_API_ACTIONS.exclude?(pr) }
190
+ proc_result = [unversioned, to_relation, proc_result]
191
+ end
192
+ # Otherwise don't change this array because it's probably legit
193
+ when String
194
+ proc_result = [proc_result] # Treat this as the surfaced api_name (path) they want to use for this endpoint
195
+ else
196
+ puts "::Brick.api_filter Proc warning: Unable to parse this result returned: \n #{proc_result.inspect}"
197
+ proc_result = nil # Couldn't understand what in the world was returned
198
+ end
199
+
200
+ if proc_result&.present?
201
+ if proc_result[1] # to_other_relation
202
+ if (new_view_relation = ::Brick.relations.fetch(proc_result[1], nil))
203
+ k = proc_result[1] # Route this call over to this different relation
204
+ view_relation = new_view_relation
205
+ else
206
+ puts "::Brick.api_filter Proc warning: Unable to find new suggested relation with name #{proc_result[1]} -- sticking with #{k} instead."
207
+ end
208
+ end
209
+ if proc_result.first&.!=(k) # updated_api_name -- a different name than this relation would normally have
210
+ found = proc_result.first
211
+ end
212
+ actions &= proc_result[2] if proc_result[2] # allowed_actions
213
+ end
214
+ (view_relation[:api][api_ver_num] ||= {})[unversioned] = actions # Add to the list of API paths this resource responds to
215
+
216
+ # view_ver_num = if (first_part = k.split('_').first) =~ /^v[\d_]+/
217
+ # first_part[1..-1].gsub('_', '.').to_i
218
+ # end
219
+ controller_name = if (last = view_relation.fetch(:resource, nil)&.pluralize)
220
+ "#{full_schema_prefix}#{last}"
221
+ else
222
+ found
223
+ end.tr('.', '/')
224
+
225
+ { :index => 'get', :create => 'post' }.each do |action, method|
226
+ if actions.include?(action)
227
+ # Normally goes to something like: /api/v1/employees
228
+ send(method, "#{api_root}#{unversioned.tr('.', '/')}", { to: "#{controller_prefix}#{controller_name}##{action}" })
229
+ end
230
+ end
231
+ # %%% We do not yet surface the #show action
232
+ if (id_col = view_relation[:pk]&.first) # ID-dependent stuff
233
+ { :update => ['put', 'patch'], :destroy => ['delete'] }.each do |action, methods|
234
+ if actions.include?(action)
235
+ methods.each do |method|
236
+ send(method, "#{api_root}#{unversioned.tr('.', '/')}/:#{id_col}", { to: "#{controller_prefix}#{controller_name}##{action}" })
237
+ end
238
+ end
239
+ end
240
+ end
241
+ end
242
+ end
243
+
244
+ # Trestle compatibility
245
+ if Object.const_defined?('Trestle') && ::Trestle.config.options&.key?(:site_title) &&
246
+ !Object.const_defined?("#{(res_name = resource_name.tr('/', '_')).camelize}Admin")
247
+ begin
248
+ ::Trestle.resource(res_sym = res_name.to_sym, model: class_name&.constantize) do
249
+ menu { item res_sym, icon: "fa fa-star" }
250
+ end
251
+ rescue
252
+ end
253
+ end
254
+ end
255
+
256
+ if (named_routes = instance_variable_get(:@set).named_routes).respond_to?(:find)
257
+ if ::Brick.config.add_status && (status_as = "#{controller_prefix.tr('/', '_')}brick_status".to_sym)
258
+ (
259
+ !(status_route = instance_variable_get(:@set).named_routes.find { |route| route.first == status_as }&.last) ||
260
+ !status_route.ast.to_s.include?("/#{controller_prefix}brick_status/")
261
+ )
262
+ get("/#{controller_prefix}brick_status", to: 'brick_gem#status', as: status_as.to_s)
263
+ end
264
+
265
+ # ::Brick.config.add_schema &&
266
+ if (schema_as = "#{controller_prefix.tr('/', '_')}brick_schema".to_sym)
267
+ (
268
+ !(schema_route = instance_variable_get(:@set).named_routes.find { |route| route.first == schema_as }&.last) ||
269
+ !schema_route.ast.to_s.include?("/#{controller_prefix}brick_schema/")
270
+ )
271
+ post("/#{controller_prefix}brick_schema", to: 'brick_gem#schema_create', as: schema_as.to_s)
272
+ end
273
+
274
+ if ::Brick.config.add_orphans && (orphans_as = "#{controller_prefix.tr('/', '_')}brick_orphans".to_sym)
275
+ (
276
+ !(orphans_route = instance_variable_get(:@set).named_routes.find { |route| route.first == orphans_as }&.last) ||
277
+ !orphans_route.ast.to_s.include?("/#{controller_prefix}brick_orphans/")
278
+ )
279
+ get("/#{controller_prefix}brick_orphans", to: 'brick_gem#orphans', as: 'brick_orphans')
280
+ end
281
+ end
282
+
283
+ if instance_variable_get(:@set).named_routes.names.exclude?(:brick_crosstab)
284
+ get("/#{controller_prefix}brick_crosstab", to: 'brick_gem#crosstab', as: 'brick_crosstab')
285
+ get("/#{controller_prefix}brick_crosstab/data", to: 'brick_gem#crosstab_data')
286
+ end
287
+
288
+ if ((rswag_ui_present = Object.const_defined?('Rswag::Ui')) &&
289
+ (rswag_path = routeset_to_use.routes.find { |r| r.app.app == ::Rswag::Ui::Engine }
290
+ &.instance_variable_get(:@path_formatter)
291
+ &.instance_variable_get(:@parts)&.join) &&
292
+ (doc_endpoints = ::Rswag::Ui.config.config_object[:urls])) ||
293
+ (doc_endpoints = ::Brick.instance_variable_get(:@swagger_endpoints))
294
+ last_endpoint_parts = nil
295
+ doc_endpoints.each do |doc_endpoint|
296
+ puts "Mounting OpenApi 3.0 documentation endpoint for \"#{doc_endpoint[:name]}\" on #{doc_endpoint[:url]}" unless ::Brick.routes_done
297
+ send(:get, doc_endpoint[:url], { to: 'brick_openapi#index' })
298
+ endpoint_parts = doc_endpoint[:url]&.split('/')
299
+ last_endpoint_parts = endpoint_parts
300
+ end
301
+ end
302
+ return if ::Brick.routes_done
303
+
304
+ if doc_endpoints.present?
305
+ if rswag_ui_present
306
+ if rswag_path
307
+ puts "API documentation now available when navigating to: /#{last_endpoint_parts&.find(&:present?)}/index.html"
308
+ else
309
+ puts "In order to make documentation available you can put this into your routes.rb:"
310
+ puts " mount Rswag::Ui::Engine => '/#{last_endpoint_parts&.find(&:present?) || 'api-docs'}'"
311
+ end
312
+ else
313
+ puts "Having this exposed, one easy way to leverage this to create HTML-based API documentation is to use Scalar.
314
+ It will jump to life when you put these two lines into a view template or other HTML resource:
315
+ <script id=\"api-reference\" data-url=\"#{last_endpoint_parts.join('/')}\"></script>
316
+ <script src=\"https://cdn.jsdelivr.net/@scalar/api-reference\"></script>
317
+ Alternatively you can add the rswag-ui gem."
318
+ end
319
+ elsif rswag_ui_present
320
+ sample_path = rswag_path || '/api-docs'
321
+ puts
322
+ puts "Brick: rswag-ui gem detected -- to make OpenAPI 3.0 documentation available from a path such as '#{sample_path}/v1/swagger.json',"
323
+ puts ' put code such as this in an initializer:'
324
+ puts ' Rswag::Ui.configure do |config|'
325
+ puts " config.swagger_endpoint '#{sample_path}/v1/swagger.json', 'API V1 Docs'"
326
+ puts ' end'
327
+ unless rswag_path
328
+ puts
329
+ puts ' and put this into your routes.rb:'
330
+ puts " mount Rswag::Ui::Engine => '/api-docs'"
331
+ end
332
+ end
333
+
334
+ puts "\n" if tables.present? || views.present?
335
+ if tables.present?
336
+ puts "Classes that can be built from tables:#{' ' * (table_class_length - 38)} Path:"
337
+ puts "======================================#{' ' * (table_class_length - 38)} ====="
338
+ ::Brick.display_classes(controller_prefix, tables, table_class_length)
339
+ end
340
+ if views.present?
341
+ puts "Classes that can be built from views:#{' ' * (view_class_length - 37)} Path:"
342
+ puts "=====================================#{' ' * (view_class_length - 37)} ====="
343
+ ::Brick.display_classes(controller_prefix, views, view_class_length)
344
+ end
345
+ ::Brick.routes_done = true
346
+ end
347
+ end
348
+ end
@@ -5,7 +5,7 @@ module Brick
5
5
  module VERSION
6
6
  MAJOR = 1
7
7
  MINOR = 0
8
- TINY = 190
8
+ TINY = 192
9
9
 
10
10
  # PRE is nil unless it's a pre-release (beta, RC, etc.)
11
11
  PRE = nil