brick 1.0.190 → 1.0.192

Sign up to get free protection for your applications and to get access to all the features.
@@ -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