no_fly_list 0.4.0 → 0.5.0

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: 8e5fd47ff5074ba83503a9c1633bada42f7348aafb9a914d2c5f51a70820427d
4
- data.tar.gz: a337a25cdce43d598fd501266ba0d99665c3adb183f994121fd201082e4e874b
3
+ metadata.gz: d5eef702cefcd730112adc305c5e17f8bb1c804d48fa3231d70549929a4f2ad9
4
+ data.tar.gz: 2f845bc5cd816a4c0899ed98f9cb9ac549c9513d5b79e7336876c39362ac72ca
5
5
  SHA512:
6
- metadata.gz: a0063a48bd720527540437b156136c704bd9deea0c4985fb34a9c4a7595c3c932d3c9b13ab5b8032daf277b5e547b358b5343410bdb6bedae1e466bc081edecd
7
- data.tar.gz: 2f430e49bc7a6cc8ed09b1aa5686121c8d90d2262f7a6cce48de48e208804ab77899227e2b48ace8a6b017a9147fd368136ce42b99eeaab9668361e1bf10acfc
6
+ metadata.gz: 4c31ab4627476af364c582db35a5176619d59f3bb96c8205c125f3219eec58c82d394c6b5cd54e8a45787b4c17b3f05bc962d32cb957e8a9befa0175757d88ad
7
+ data.tar.gz: 869883a8b6ffebed467494a2f97597defbc8718fd1f94d3ea33618aa31f391b716335830dd55dcb5eced81b7dbc177a3a62dfdb9db0dda408094422d2a89a2d0
@@ -3,14 +3,14 @@
3
3
  module ApplicationTagTransformer
4
4
  module_function
5
5
 
6
- # @param tags [String|Array<String>]
6
+ # @param [String|Array<String>] tags
7
7
  def parse_tags(tags)
8
8
  tags = recreate_string(tags) if tags.is_a?(Array)
9
9
  tags.split(separator).map(&:strip)
10
10
  end
11
11
 
12
12
  # Recreate a string from an array of tags
13
- # @param tags [Array<String>]
13
+ # @param [Array<String>] tags
14
14
  # @return [String]
15
15
  def recreate_string(tags)
16
16
  tags.join(separator)
@@ -1,93 +1,72 @@
1
- # frozen_string_literal: true
2
-
3
1
  namespace :no_fly_list do
4
- desc "List all taggable records"
2
+ desc "List all models using NoFlyList::TaggableRecord with details about their tagging configurations"
5
3
  task taggable_records: :environment do
6
- Rails.application.eager_load!
7
- taggable_classes = ActiveRecord::Base.descendants.select do |klass|
8
- klass.included_modules.any? { |mod| mod.in?([ NoFlyList::TaggableRecord ]) }
9
- end
10
-
11
- puts "Found #{taggable_classes.size} taggable classes:\n\n"
12
-
13
- taggable_classes.each do |klass|
14
- puts "Class: #{klass.name}"
4
+ classes = NoFlyList::TaskHelpers.find_taggable_classes
5
+ puts "Found #{classes.size} taggable classes:\n\n"
6
+
7
+ classes.each do |klass|
8
+ color = NoFlyList::TaskHelpers.adapter_color(klass)
9
+ puts "#{color}#{klass.name}#{NoFlyList::TaskHelpers::COLORS[:reset]}"
10
+ puts " Tag Contexts: #{klass._no_fly_list&.tag_contexts&.keys&.join(', ')}"
11
+ puts " Tables: #{klass.table_name}"
12
+ puts
15
13
  end
16
14
  end
17
15
 
18
- desc "List all tag records"
16
+ desc "List all tag classes (both global and model-specific) with their inheritance chain"
19
17
  task tag_records: :environment do
20
- Rails.application.eager_load!
21
- tag_classes = ActiveRecord::Base.descendants.select do |klass|
22
- klass.included_modules.any? { |mod| mod.in?([ NoFlyList::ApplicationTag, NoFlyList::TagRecord ]) }
23
- end
18
+ classes = NoFlyList::TaskHelpers.find_tag_classes
19
+ puts "Found #{classes.size} tag classes:\n\n"
24
20
 
25
- puts "Found #{tag_classes.size} tag classes:\n\n"
21
+ classes.each do |klass|
22
+ color = NoFlyList::TaskHelpers.adapter_color(klass)
23
+ type = klass.included_modules.include?(NoFlyList::ApplicationTag) ? 'Global' : 'Model-specific'
26
24
 
27
- tag_classes.each do |klass|
28
- puts "Class: #{klass.name}"
25
+ puts "#{color}#{klass.name}#{NoFlyList::TaskHelpers::COLORS[:reset]}"
26
+ puts " Type: #{type}"
27
+ puts " Table: #{klass.table_name}"
28
+ puts
29
29
  end
30
30
  end
31
31
 
32
- desc "Check taggable records and their associated tables"
32
+ desc "Validate database schema for all taggable models"
33
33
  task check_taggable_records: :environment do
34
- Rails.application.eager_load!
35
- taggable_classes = ActiveRecord::Base.descendants.select do |klass|
36
- klass.included_modules.any? { |mod| mod.in?([ NoFlyList::TaggableRecord ]) }
37
- end
38
-
39
- puts "Checking #{taggable_classes.size} taggable classes:\n\n"
40
-
41
- taggable_classes.each do |klass|
42
- puts "Checking Class: #{klass.name}"
43
-
44
- # ANSI color codes
45
- green = "\e[32m"
46
- red = "\e[31m"
47
- reset = "\e[0m"
48
-
49
- # Check main table exists
50
- begin
51
- klass.table_exists?
52
- puts " #{green}✓#{reset} Main table exists: #{klass.table_name}"
53
- rescue StandardError => e
54
- puts " #{red}✗#{reset} Error checking main table: #{e.message}"
55
- end
56
-
57
- # Dynamically find tag and tagging class names
58
- tag_class_name = "#{klass.name}Tag"
59
- tagging_class_name = "#{klass.name}::Tagging"
60
-
61
- begin
62
- tag_class = Object.const_get(tag_class_name)
63
-
64
- # Check tags table exists
65
- if tag_class.table_exists?
66
- puts " #{green}✓#{reset} Tags table exists: #{tag_class.table_name}"
34
+ classes = NoFlyList::TaskHelpers.find_taggable_classes
35
+ puts "Checking #{classes.size} taggable classes:\n\n"
36
+
37
+ classes.each do |klass|
38
+ color = NoFlyList::TaskHelpers.adapter_color(klass)
39
+ puts "Checking Class: #{color}#{klass.name}#{NoFlyList::TaskHelpers::COLORS[:reset]}"
40
+
41
+ status, message = NoFlyList::TaskHelpers.check_table(klass)
42
+ puts " #{message}"
43
+
44
+ [
45
+ ["#{klass.name}Tag", "Tags", :tag],
46
+ ["#{klass.name}::Tagging", "Taggings", :tagging]
47
+ ].each do |class_name, type, column_type|
48
+ if (check_class = NoFlyList::TaskHelpers.check_class(class_name))
49
+ status, message = NoFlyList::TaskHelpers.check_table(check_class)
50
+ puts " #{message}"
51
+ if status
52
+ puts " #{NoFlyList::TaskHelpers.verify_columns(check_class, column_type)}"
53
+ puts " #{NoFlyList::TaskHelpers.format_columns(check_class)}"
54
+ end
67
55
  else
68
- puts " #{red}✗#{reset} Tags table missing: #{tag_class.table_name}"
56
+ puts " #{NoFlyList::TaskHelpers::colorize('✗', :red)} #{type} class not found: #{class_name}"
69
57
  end
70
- rescue NameError
71
- puts " #{red}✗#{reset} Tag class not found: #{tag_class_name}"
72
- rescue StandardError => e
73
- puts " #{red}✗#{reset} Error checking tag class: #{e.message}"
74
58
  end
75
59
 
76
- begin
77
- tagging_class = Object.const_get(tagging_class_name)
78
-
79
- # Check taggings table exists
80
- if tagging_class.table_exists?
81
- puts " #{green}✓#{reset} Taggings table exists: #{tagging_class.table_name}"
82
- else
83
- puts " #{red}✗#{reset} Taggings table missing: #{tagging_class.table_name}"
60
+ klass._no_fly_list.tag_contexts.each do |context, config|
61
+ puts "\n Context: #{context}"
62
+ bullet = NoFlyList::TaskHelpers::colorize('•', :green)
63
+ puts " #{bullet} Tag class: #{config[:tag_class_name]}"
64
+ puts " #{bullet} Tagging class: #{config[:tagging_class_name]}"
65
+ puts " #{bullet} Polymorphic: #{config[:polymorphic]}"
66
+ if config[:polymorphic]
67
+ puts " #{bullet} Required tagging columns: context, tag_id, taggable_id, taggable_type"
84
68
  end
85
- rescue NameError
86
- puts " #{red}✗#{reset} Tagging class not found: #{tagging_class_name}"
87
- rescue StandardError => e
88
- puts " #{red}✗#{reset} Error checking tagging class: #{e.message}"
89
69
  end
90
-
91
70
  puts "\n"
92
71
  end
93
72
  end
@@ -1,7 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module NoFlyList
4
- # This module provides functionality for a tag table that contains tags for a model.
4
+ # Provides tag table functionality for models
5
+ # Handles tag attributes and delegation of methods
5
6
  #
6
7
  # @example Usage
7
8
  # class User::Tag < ApplicationRecord
@@ -11,6 +12,9 @@ module NoFlyList
11
12
  extend ActiveSupport::Concern
12
13
 
13
14
  included do
15
+ # @!method to_s
16
+ # @return [String] String representation of tag name
17
+ # @!method inspect
14
18
  delegate :to_s, to: :name
15
19
  alias_attribute :tag_name, :name
16
20
  def inspect
@@ -13,9 +13,9 @@ module NoFlyList
13
13
  module_function
14
14
 
15
15
  # Main entry point for setting up tagging functionality on a model
16
- # @param taggable_klass [Class] The model class to make taggable
17
- # @param contexts [Array<Symbol>] The contexts to create tags for (e.g., :tags, :colors)
18
- # @param options [Hash] Configuration options for tagging behavior
16
+ # @param [Class] taggable_klass The model class to make taggable
17
+ # @param [Array<Symbol>] contexts The contexts to create tags for (e.g., :tags, :colors)
18
+ # @param [Hash] options Configuration options for tagging behavior
19
19
  def setup_tagging(taggable_klass, contexts, options = {})
20
20
  contexts.each do |context|
21
21
  setup = build_tag_setup(taggable_klass, context, options)
@@ -5,6 +5,12 @@ module NoFlyList
5
5
  module Mutation
6
6
  module_function
7
7
 
8
+ # Defines mutation methods for tag manipulation
9
+ # @param setup [TagSetup] Tag setup configuration
10
+ # @return [void]
11
+ # @example Generated methods
12
+ # add_tags("red, blue") # Adds tags
13
+ # remove_tags("red") # Removes tags
8
14
  def define_mutation_methods(setup)
9
15
  context = setup.context
10
16
  taggable_klass = setup.taggable_klass
@@ -5,6 +5,12 @@ module NoFlyList
5
5
  module Query
6
6
  module_function
7
7
 
8
+ # Defines query methods based on database adapter
9
+ # @param setup [TagSetup] Tag setup configuration
10
+ # @return [void]
11
+ # @see PostgresqlStrategy#define_query_methods
12
+ # @see MysqlStrategy#define_query_methods
13
+ # @see SqliteStrategy#define_query_methods
8
14
  def define_query_methods(setup)
9
15
  case setup.adapter
10
16
  when :postgresql
@@ -19,10 +25,19 @@ module NoFlyList
19
25
  module BaseStrategy
20
26
  module_function
21
27
 
28
+ # Performs case-insensitive column comparison
29
+ # @param table [Arel::Table] Database table
30
+ # @param column [Symbol] Column name
31
+ # @param values [Array<String>] Values to compare
32
+ # @return [Arel::Node] Query node
33
+ # @abstract
22
34
  def case_insensitive_where(table, column, values)
23
35
  raise NotImplementedError
24
36
  end
25
37
 
38
+ # Defines database-specific query methods
39
+ # @abstract
40
+ # @param setup [TagSetup] Tag setup configuration
26
41
  def define_query_methods(setup)
27
42
  raise NotImplementedError
28
43
  end
@@ -2,11 +2,42 @@
2
2
 
3
3
  module NoFlyList
4
4
  module TaggableRecord
5
+ # Handles setup and configuration of tagging for a model
5
6
  class TagSetup
6
- attr_reader :taggable_klass, :context, :transformer, :polymorphic,
7
- :restrict_to_existing, :limit, :counter_cache,
8
- :tag_class_name, :tagging_class_name, :adapter
7
+ # @return [Class] Model class being made taggable
8
+ attr_reader :taggable_klass
9
9
 
10
+ # @return [Symbol] Tagging context name
11
+ attr_reader :context
12
+
13
+ # @return [Class] Tag string transformer
14
+ attr_reader :transformer
15
+
16
+ # @return [Boolean] Whether tags are polymorphic
17
+ attr_reader :polymorphic
18
+
19
+ # @return [Boolean] Whether to restrict to existing tags
20
+ attr_reader :restrict_to_existing
21
+
22
+ # @return [Integer, nil] Maximum number of tags allowed
23
+ attr_reader :limit
24
+
25
+ # @return [Boolean] Whether to use counter cache
26
+ attr_reader :counter_cache
27
+
28
+ # @return [String] Name of tag class
29
+ attr_reader :tag_class_name
30
+
31
+ # @return [String] Name of tagging class
32
+ attr_reader :tagging_class_name
33
+
34
+ # @return [Symbol] Database adapter type (:postgresql, :mysql, :sqlite)
35
+ attr_reader :adapter
36
+
37
+ # Creates new tag setup configuration
38
+ # @param taggable_klass [Class] Model to configure
39
+ # @param context [Symbol] Tag context name
40
+ # @param options [Hash] Setup options
10
41
  def initialize(taggable_klass, context, options = {})
11
42
  @taggable_klass = taggable_klass
12
43
  @context = context
@@ -4,6 +4,12 @@ module NoFlyList
4
4
  module TaggableRecord
5
5
  extend ActiveSupport::Concern
6
6
 
7
+ # @!attribute _no_fly_list
8
+ # @return [Config] Configuration for this taggable record
9
+ #
10
+ # @!method save_tag_proxies
11
+ # @api private
12
+ # Saves pending tag changes before saving the record
7
13
  included do
8
14
  class_attribute :_no_fly_list, instance_writer: false
9
15
  self._no_fly_list = Config.new self
@@ -12,6 +18,9 @@ module NoFlyList
12
18
  before_validation :validate_tag_proxies
13
19
  end
14
20
 
21
+ # Determines if record needs saving due to tag changes
22
+ # @return [Boolean] True if tags have pending changes
23
+ # @api private
15
24
  def changed_for_autosave?
16
25
  super || tag_proxies_changed?
17
26
  end
@@ -83,6 +92,11 @@ module NoFlyList
83
92
  end
84
93
  end
85
94
 
95
+ # Class methods added to taggable models
96
+ # @!method has_tags(*contexts, **options)
97
+ # Configures tagging contexts for the model
98
+ # @param contexts [Array<Symbol>] Tag context names
99
+ # @param options [Hash] Options for configuring tagging
86
100
  class_methods do
87
101
  def has_tags(*contexts, **options)
88
102
  contexts.each do |context|
@@ -13,6 +13,13 @@ module NoFlyList
13
13
  validate :validate_limit
14
14
  validate :validate_existing_tags
15
15
 
16
+ # Creates a new tagging proxy
17
+ # @param model [ActiveRecord::Base] Model being tagged
18
+ # @param tag_model [Class] Tag model class
19
+ # @param context [Symbol] Tagging context (e.g. :colors)
20
+ # @param transformer [Class] Class for transforming tag strings
21
+ # @param restrict_to_existing [Boolean] Only allow existing tags
22
+ # @param limit [Integer, nil] Maximum number of tags allowed
16
23
  def initialize(model, tag_model, context,
17
24
  transformer: ApplicationTagTransformer,
18
25
  restrict_to_existing: false,
@@ -26,6 +33,9 @@ module NoFlyList
26
33
  @pending_changes = []
27
34
  end
28
35
 
36
+ # Determines if tags have changed from database state
37
+ # @return [Boolean] True if pending changes differ from database
38
+ # @api private
29
39
  def changed?
30
40
  @pending_changes.present? && @pending_changes != current_list_from_database
31
41
  end
@@ -50,6 +60,9 @@ module NoFlyList
50
60
  method_name.to_s =~ /\A(.+)_list(=)?\z/
51
61
  end
52
62
 
63
+ # Handles numeric coercion
64
+ # @param other [Object] Object to coerce with
65
+ # @return [Array] Two-element array for coercion
53
66
  def coerce(other)
54
67
  [ other, to_a ]
55
68
  end
@@ -152,10 +165,20 @@ module NoFlyList
152
165
  "#<#{self.class.name} tags=#{current_list.inspect} transformer_with=#{transformer_name} >"
153
166
  end
154
167
 
168
+ # Adds one or more tags to the current tag list
169
+ # @param *tags [Array<String, Array<String>>] Tags to add:
170
+ # - Single string with comma-separated values ("tag1, tag2")
171
+ # - Single array of strings (["tag1", "tag2"])
172
+ # - Multiple string arguments ("tag1", "tag2")
173
+ # @return [TaggingProxy] Returns self for method chaining
155
174
  def add(*tags)
156
175
  return self if limit_reached?
157
176
 
158
- new_tags = tags.flatten.map { |tag| transformer.parse_tags(tag) }.flatten
177
+ new_tags = if tags.size == 1 && tags.first.is_a?(String)
178
+ transformer.parse_tags(tags.first)
179
+ else
180
+ tags.flatten.map { |tag| transformer.parse_tags(tag) }.flatten
181
+ end
159
182
  return self if new_tags.empty?
160
183
 
161
184
  @pending_changes = current_list + new_tags
@@ -168,9 +191,21 @@ module NoFlyList
168
191
  save
169
192
  end
170
193
 
171
- def remove(tag)
194
+ # Removes one or more tags from the current tag list
195
+ # @param *tags [Array<String, Array<String>>] Tags to remove:
196
+ # - Single string with comma-separated values ("tag1, tag2")
197
+ # - Single array of strings (["tag1", "tag2"])
198
+ # - Multiple string arguments ("tag1", "tag2")
199
+ # @return [TaggingProxy] Returns self for method chaining
200
+ # @raise [ActiveRecord::RecordInvalid] If validation fails
201
+ def remove(*tags)
172
202
  old_list = current_list.dup
173
- @pending_changes = current_list - [ tag.to_s.strip ]
203
+ tags_to_remove = if tags.size == 1 && tags.first.is_a?(String)
204
+ transformer.parse_tags(tags.first)
205
+ else
206
+ tags.flatten.map { |tag| tag.to_s.strip }
207
+ end
208
+ @pending_changes = current_list - tags_to_remove
174
209
  mark_record_dirty if @pending_changes != old_list
175
210
  self
176
211
  end
@@ -180,6 +215,10 @@ module NoFlyList
180
215
  save
181
216
  end
182
217
 
218
+ # Clears all tags
219
+ # @return [TaggingProxy] Returns self for method chaining
220
+ # @example Clear all tags
221
+ # tags.clear #=> []
183
222
  def clear
184
223
  old_list = current_list.dup
185
224
  @pending_changes = []
@@ -188,6 +227,11 @@ module NoFlyList
188
227
  self
189
228
  end
190
229
 
230
+ # Forces clearing all tags by destroying records
231
+ # @return [TaggingProxy] Returns self for method chaining
232
+ # @example Force clear tags
233
+ # tags.clear! #=> []
234
+ # @raise [ActiveRecord::RecordNotDestroyed] If destroy fails
191
235
  def clear!
192
236
  @model.send(@context.to_s).destroy_all
193
237
  @pending_changes = []
@@ -195,14 +239,23 @@ module NoFlyList
195
239
  self
196
240
  end
197
241
 
242
+ # Checks if a tag exists in the list
243
+ # @param tag [String] Tag to check for
244
+ # @return [Boolean] True if tag exists
198
245
  def include?(tag)
199
246
  current_list.include?(tag.to_s.strip)
200
247
  end
201
248
 
249
+ # Checks if tag list is empty
250
+ # @return [Boolean] True if no tags exist
202
251
  def empty?
203
252
  current_list.empty?
204
253
  end
205
254
 
255
+ # Required by ActiveModel::Validations
256
+ # @return [Boolean] Always returns false since proxy isn't persisted
257
+ # @api private
258
+ # @see https://api.rubyonrails.org/classes/ActiveModel/Validations.html
206
259
  def persisted?
207
260
  false
208
261
  end
@@ -0,0 +1,76 @@
1
+ # frozen_string_literal: true
2
+
3
+ module NoFlyList
4
+ module TaskHelpers
5
+ COLORS = {
6
+ mysql2: "\e[38;5;33m",
7
+ postgresql: "\e[38;5;31m",
8
+ sqlite: "\e[38;5;245m",
9
+ reset: "\e[0m",
10
+ green: "\e[32m",
11
+ red: "\e[31m",
12
+ yellow: "\e[33m"
13
+ }.freeze
14
+
15
+ REQUIRED_COLUMNS = {
16
+ tag: ['name'],
17
+ tagging: ['tag_id', 'taggable_id', 'context']
18
+ }.freeze
19
+
20
+ def self.adapter_color(klass)
21
+ color_key = klass.connection.adapter_name.downcase.to_sym
22
+ COLORS[color_key] || COLORS[:sqlite]
23
+ end
24
+
25
+ def self.colorize(text, color)
26
+ "#{COLORS[color]}#{text}#{COLORS[:reset]}"
27
+ end
28
+
29
+ def self.check_table(klass)
30
+ klass.table_exists?
31
+ [true, "#{colorize('✓', :green)} Table exists: #{klass.table_name}"]
32
+ rescue StandardError => e
33
+ [false, "#{colorize('✗', :red)} Error: #{e.message}"]
34
+ end
35
+
36
+ def self.verify_columns(klass, type)
37
+ return unless klass.table_exists?
38
+
39
+ existing_columns = klass.column_names
40
+ required_columns = REQUIRED_COLUMNS[type]
41
+ missing_columns = required_columns - existing_columns
42
+
43
+ if missing_columns.empty?
44
+ "#{colorize('✓', :green)} Required columns present"
45
+ else
46
+ "#{colorize('!', :yellow)} Missing required columns: #{missing_columns.join(', ')}"
47
+ end
48
+ end
49
+
50
+ def self.format_columns(klass)
51
+ return unless klass.table_exists?
52
+ "#{colorize('✓', :green)} All columns: #{klass.column_names.join(', ')}"
53
+ end
54
+
55
+ def self.check_class(class_name)
56
+ Object.const_get(class_name)
57
+ rescue NameError
58
+ nil
59
+ end
60
+
61
+ def self.find_taggable_classes
62
+ Rails.application.eager_load!
63
+ ActiveRecord::Base.descendants.select do |klass|
64
+ klass.included_modules.any? { |mod| mod.in?([NoFlyList::TaggableRecord]) }
65
+ end
66
+ end
67
+
68
+ def self.find_tag_classes
69
+ Rails.application.eager_load!
70
+ ActiveRecord::Base.descendants.select do |klass|
71
+ klass.included_modules.any? { |mod| mod.in?([NoFlyList::ApplicationTag, NoFlyList::TagRecord]) }
72
+ end
73
+ end
74
+ end
75
+ end
76
+
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module NoFlyList
4
- VERSION = "0.4.0"
4
+ VERSION = "0.5.0"
5
5
  end
data/lib/no_fly_list.rb CHANGED
@@ -32,4 +32,5 @@ module NoFlyList
32
32
  autoload :TaggingProxy
33
33
 
34
34
  autoload :TestHelper
35
+ autoload :TaskHelpers
35
36
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: no_fly_list
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.4.0
4
+ version: 0.5.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Abdelkader Boudih
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2024-12-19 00:00:00.000000000 Z
11
+ date: 2024-12-20 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: activerecord
@@ -59,6 +59,7 @@ files:
59
59
  - lib/no_fly_list/taggable_record/tag_setup.rb
60
60
  - lib/no_fly_list/tagging_proxy.rb
61
61
  - lib/no_fly_list/tagging_record.rb
62
+ - lib/no_fly_list/task_helpers.rb
62
63
  - lib/no_fly_list/test_helper.rb
63
64
  - lib/no_fly_list/version.rb
64
65
  homepage: https://github.com/contriboss/no_fly_list