brick 1.0.149 → 1.0.151

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 91e6b3d87c423a90e6e6cf62a07235e185373d9bb60990579a13ccea83c6a645
4
- data.tar.gz: 8670e080c676eb7c5be3bb62b7ce4e7a310e8d30c98c3d189d513bd9e06a2af3
3
+ metadata.gz: 1d8abfeb02b561a0ac8214de3c4791992947cab9f288f9f57c816a883dc4592a
4
+ data.tar.gz: 1ec0f9d11eea9b1741e454ac04aa2c51380a0d9676f4e84a5de2419d3ef751bd
5
5
  SHA512:
6
- metadata.gz: 58ff4070f313c577b027f2c83babb99c2aeecaf0cfd05edbed89ed6e5a32485a9bdc8849e303d9eb16fa6f5824beb66e4dbf282843bc4e2b1a00c7ec13e0616a
7
- data.tar.gz: 20cd0a6b43f267a653923a86ded57b041cbe49adbd1e239236faac18415eab0e8515490e0828dcc67f4d7e4f809efae9ff7249a1ab5d9a9626a8a24dced8d2d9
6
+ metadata.gz: 758e2ab6e10589689abdb77df64731e3df2fff341998e8f815bd5f180a73df89fb1282155a9520cc43102a49b1f7283114c02146329161e2324fc2fa79c6048e
7
+ data.tar.gz: 9a77d3d4318943225427c59e676c4a067c95a22351eaea88ed14bbc6da60ee834498117704bc0fe1dd493f6e10ab65a27db4bff3b060440f4e7dfaf3b31ad184
@@ -1528,12 +1528,12 @@ class Object
1528
1528
  # end
1529
1529
  end
1530
1530
 
1531
- # Enable Turbo Stream if possible -- equivalent of: broadcasts_to ->(comment) { :comments }
1532
- if Object.const_defined?(:ApplicationCable) && Object.const_defined?(:Turbo) && Turbo.const_defined?(:Broadcastable) && respond_to?(:broadcasts_to)
1533
- relation[:broadcasts] = true
1534
- self.broadcasts_to ->(model) { (model&.class&.name || chosen_name).underscore.pluralize.to_sym }
1535
- code << " broadcasts_to ->(#{chosen_name}) { #{chosen_name}&.class&.name&.underscore&.pluralize&.to_sym }\n"
1536
- end
1531
+ # # %%% Enable Turbo Stream if possible -- equivalent of: broadcasts_to ->(comment) { :comments }
1532
+ # if Object.const_defined?(:ApplicationCable) && Object.const_defined?(:Turbo) && Turbo.const_defined?(:Broadcastable) && respond_to?(:broadcasts_to)
1533
+ # relation[:broadcasts] = true
1534
+ # self.broadcasts_to ->(model) { (model&.class&.name || chosen_name).underscore.pluralize.to_sym }
1535
+ # code << " broadcasts_to ->(#{chosen_name}) { #{chosen_name}&.class&.name&.underscore&.pluralize&.to_sym }\n"
1536
+ # end
1537
1537
  end # class definition
1538
1538
  # Having this separate -- will this now work out better?
1539
1539
  built_model.class_exec do
@@ -2155,13 +2155,16 @@ class Object
2155
2155
  ::Brick.set_db_schema(params)
2156
2156
  if request.format == :csv # Importing CSV?
2157
2157
  require 'csv'
2158
+
2158
2159
  # See if internally it's likely a TSV file (tab-separated)
2159
- tab_counts = []
2160
- 5.times { tab_counts << request.body.readline.count("\t") unless request.body.eof? }
2160
+ likely_separator = Hash.new { |h, k| h[k] = 0 }
2161
+ request.body.readline # Expect first row to have column headers
2162
+ 5.times { ::Brick._find_csv_separator(request.body, likely_separator) unless request.body.eof? }
2161
2163
  request.body.rewind
2162
- separator = "\t" if tab_counts.length > 0 && tab_counts.uniq.length == 1 && tab_counts.first > 0
2163
- result = model.df_import(CSV.parse(request.body, { col_sep: separator || :auto }), model.brick_import_template)
2164
- # render inline: exported_csv, content_type: request.format
2164
+ separator = "\t" if likely_separator["\t"] > likely_separator[',']
2165
+
2166
+ result = model.df_import(CSV.parse(request.body, **{ col_sep: separator || :auto }), model.brick_import_template)
2167
+ render inline: result.to_json, content_type: request.format
2165
2168
  return
2166
2169
  # elsif request.format == :js # Asking for JSON?
2167
2170
  # render inline: model.df_export(true).to_json, content_type: request.format
@@ -2238,14 +2241,9 @@ class Object
2238
2241
 
2239
2242
  if is_need_params
2240
2243
  code << " def #{params_name}\n"
2241
- permits = model.columns_hash.keys.map(&:to_sym)
2242
- permits_txt = permits.map(&:inspect) +
2243
- model.reflect_on_all_associations.select { |assoc| assoc.macro == :has_many && assoc.options[:through] }.map do |assoc|
2244
- permits << { "#{assoc.name.to_s.singularize}_ids".to_sym => [] }
2245
- "#{assoc.name.to_s.singularize}_ids: []"
2246
- end
2244
+ permits_txt = self._brick_find_permits(model, (permits = model.columns_hash.keys.map(&:to_sym)))
2247
2245
  code << " params.require(:#{require_name = model.name.underscore.tr('/', '_')
2248
- }).permit(#{permits_txt.join(', ')})\n"
2246
+ }).permit(#{permits_txt.map(&:inspect).join(', ')})\n"
2249
2247
  code << " end\n"
2250
2248
  self.define_method(params_name) do
2251
2249
  params.require(require_name.to_sym).permit(permits)
@@ -2259,6 +2257,101 @@ class Object
2259
2257
  [built_controller, code]
2260
2258
  end
2261
2259
 
2260
+ def _brick_find_permits(model, current_permits, done_permits = [])
2261
+ unless done_permits.include?(self)
2262
+ done_permits << self
2263
+ model.reflect_on_all_associations.select { |assoc| assoc.macro == :has_many }.each_with_object([]) do |assoc, s|
2264
+ if assoc.options[:through]
2265
+ current_permits << { "#{assoc.name.to_s.singularize}_ids".to_sym => [] }
2266
+ s << "#{assoc.name.to_s.singularize}_ids: []"
2267
+ elsif assoc.active_record.instance_methods.include?(:"#{assoc.name}_attributes=")
2268
+ # Support nested attributes which use the friendly_id gem
2269
+ self._brick_nested_friendly_id if Object.const_defined?('FriendlyId') &&
2270
+ assoc.klass.instance_variable_get(:@friendly_id_config)
2271
+ new_attrib_text = self._brick_find_permits(assoc.klass, (new_permits = assoc.klass.columns_hash.keys.map(&:to_sym)), done_permits)
2272
+ new_permits << :_destroy
2273
+ current_permits << { "#{assoc.name}_attributes".to_sym => new_permits }
2274
+ s << "#{assoc.name}_attributes: #{new_attrib_text}"
2275
+ end
2276
+ end
2277
+ end
2278
+ current_permits
2279
+ end
2280
+
2281
+ def _brick_nested_friendly_id
2282
+ unless @_brick_nested_friendly_id
2283
+ ::ActiveRecord::Base.class_exec do
2284
+ if private_instance_methods.include?(:assign_nested_attributes_for_collection_association)
2285
+ alias _brick_anafca assign_nested_attributes_for_collection_association
2286
+ def assign_nested_attributes_for_collection_association(association_name, attributes_collection)
2287
+ association = association(association_name)
2288
+ slug_column = association.klass.instance_variable_get(:@friendly_id_config)&.slug_column
2289
+ return _brick_anafca unless slug_column
2290
+
2291
+ # Here is the FriendlyId version of #assign_nested_attributes_for_collection_association
2292
+ options = nested_attributes_options[association_name]
2293
+ attributes_collection = attributes_collection.to_h if attributes_collection.respond_to?(:permitted?)
2294
+
2295
+ unless attributes_collection.is_a?(Hash) || attributes_collection.is_a?(Array)
2296
+ raise ArgumentError, "Hash or Array expected for attribute `#{association_name}`, got #{attributes_collection.class.name} (#{attributes_collection.inspect})"
2297
+ end
2298
+
2299
+ check_record_limit!(options[:limit], attributes_collection)
2300
+
2301
+ if attributes_collection.is_a? Hash
2302
+ keys = attributes_collection.keys
2303
+ attributes_collection = if keys.include?("id") || keys.include?(:id)
2304
+ [attributes_collection]
2305
+ else
2306
+ attributes_collection.values
2307
+ end
2308
+ end
2309
+ existing_records = if association.loaded?
2310
+ association.target
2311
+ else
2312
+ attribute_ids = attributes_collection.filter_map { |a| a["id"] || a[:id] }
2313
+ if attribute_ids.empty?
2314
+ []
2315
+ else # Implement the same logic as "friendly" scope
2316
+ association.scope.where(association.klass.primary_key => attribute_ids)
2317
+ .or(association.scope.where(slug_column => attribute_ids))
2318
+ end
2319
+ end
2320
+ attributes_collection.each do |attributes|
2321
+ attributes = attributes.to_h if attributes.respond_to?(:permitted?)
2322
+ attributes = attributes.with_indifferent_access
2323
+ if attributes["id"].blank? && attributes[slug_column].blank?
2324
+ unless reject_new_record?(association_name, attributes)
2325
+ association.reader.build(attributes.except(*::ActiveRecord::Base::UNASSIGNABLE_KEYS))
2326
+ end
2327
+ elsif (attr_id_str = attributes["id"].to_s) &&
2328
+ (existing_record = existing_records.detect { |record| record.id.to_s == attr_id_str ||
2329
+ record.send(slug_column).to_s == attr_id_str })
2330
+ unless call_reject_if(association_name, attributes)
2331
+ # Make sure we are operating on the actual object which is in the association's
2332
+ # proxy_target array (either by finding it, or adding it if not found)
2333
+ # Take into account that the proxy_target may have changed due to callbacks
2334
+ target_record = association.target.detect { |record| record.id.to_s == attr_id_str ||
2335
+ record.send(slug_column).to_s == attr_id_str }
2336
+ if target_record
2337
+ existing_record = target_record
2338
+ else
2339
+ association.add_to_target(existing_record, skip_callbacks: true)
2340
+ end
2341
+
2342
+ assign_to_or_mark_for_destruction(existing_record, attributes, options[:allow_destroy])
2343
+ end
2344
+ else
2345
+ raise_nested_attributes_record_not_found!(association_name, attributes["id"])
2346
+ end
2347
+ end
2348
+ end # anafca
2349
+ end
2350
+ end if ::ActiveRecord::QueryMethods.instance_methods.include?(:or)
2351
+ @_brick_nested_friendly_id = true
2352
+ end
2353
+ end
2354
+
2262
2355
  def _brick_get_hm_assoc_name(relation, hm_assoc, source = nil)
2263
2356
  assoc_name, needs_class = if (relation[:hm_counts][hm_assoc[:inverse_table]]&.> 1) &&
2264
2357
  hm_assoc[:alternate_name] != (source || name.underscore)
@@ -263,11 +263,11 @@ function linkSchemas() {
263
263
 
264
264
  # After we're initialized and before running the rest of stuff, put our configuration in place
265
265
  ActiveSupport.on_load(:after_initialize) do |app|
266
- assets_path = File.expand_path("#{__dir__}/../../../../vendor/assets")
267
- if (app_config = app.config).respond_to?(:assets)
268
- (app_config.assets.precompile ||= []) << "#{assets_path}/images/brick_erd.png"
269
- (app.config.assets.paths ||= []) << assets_path
270
- end
266
+ # assets_path = File.expand_path("#{__dir__}/../../../../vendor/assets")
267
+ # if (app_config = app.config).respond_to?(:assets)
268
+ # (app_config.assets.precompile ||= []) << "#{assets_path}/images/brick_erd.png"
269
+ # (app.config.assets.paths ||= []) << assets_path
270
+ # end
271
271
 
272
272
  # Treat ActiveStorage::Blob metadata as JSON
273
273
  if ::Brick.config.table_name_prefixes.fetch('active_storage_', nil) == 'ActiveStorage' &&
@@ -813,7 +813,7 @@ h1, h3 {
813
813
  margin-bottom: 0;
814
814
  }
815
815
  #imgErd {
816
- background-image:url(/assets/brick_erd.png);
816
+ background-image:url();
817
817
  background-size: 100% 100%;
818
818
  width: 2.2em;
819
819
  height: 2.2em;
@@ -1241,7 +1241,9 @@ erDiagram
1241
1241
  btnImport.style.display = droppedTSV.length > 0 ? \"block\" : \"none\";
1242
1242
  });
1243
1243
  btnImport.addEventListener(\"click\", function () {
1244
- fetch(changeout(<%= #{path_obj_name}_path(-1, format: :csv).inspect.html_safe %>, \"_brick_schema\", brickSchema), {
1244
+ var patchPath = <%= #{path_obj_name}_path(-1, format: :csv).inspect.html_safe %>;
1245
+ if (brickSchema) patchPath = changeout(patchPath, \"_brick_schema\", brickSchema);
1246
+ fetch(patchPath, {
1245
1247
  method: \"PATCH\",
1246
1248
  headers: { \"Content-Type\": \"text/tab-separated-values\" },
1247
1249
  body: droppedTSV
@@ -5,7 +5,7 @@ module Brick
5
5
  module VERSION
6
6
  MAJOR = 1
7
7
  MINOR = 0
8
- TINY = 149
8
+ TINY = 151
9
9
 
10
10
  # PRE is nil unless it's a pre-release (beta, RC, etc.)
11
11
  PRE = nil
data/lib/brick.rb CHANGED
@@ -742,6 +742,28 @@ In config/initializers/brick.rb appropriate entries would look something like:
742
742
  @double_underscore_applied = true
743
743
  end
744
744
  end
745
+
746
+ def _find_csv_separator(stream, likely_separator)
747
+ line = nil
748
+ last_char = nil
749
+ in_quotes = false
750
+ ret = +''
751
+ while true
752
+ (line = stream.readline).each_char do |ch|
753
+ likely_separator[ch] += 2 if !in_quotes && last_char == '"' && ch != "\n"
754
+ if ch == '"'
755
+ in_quotes = !in_quotes
756
+ likely_separator[last_char] += 2 if in_quotes && last_char != "\n"
757
+ end
758
+ last_char = ch
759
+ end
760
+ ret << line
761
+ break if !in_quotes || stream.eof?
762
+ end
763
+ likely_separator[line[0]] += 1
764
+ likely_separator[line[-2]] += 1
765
+ ret
766
+ end
745
767
  end
746
768
 
747
769
  module RouteSet
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: brick
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.0.149
4
+ version: 1.0.151
5
5
  platform: ruby
6
6
  authors:
7
7
  - Lorin Thwaits
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2023-06-01 00:00:00.000000000 Z
11
+ date: 2023-06-08 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: activerecord