no_fly_list 0.3.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: da62f66e9694575999da6c6864650b05036fe0f32ed762255d4552e11fc53ce4
4
- data.tar.gz: 965687001501fec670c74c1a89d95340776b3b8ffa061517d34c430439471572
3
+ metadata.gz: d5eef702cefcd730112adc305c5e17f8bb1c804d48fa3231d70549929a4f2ad9
4
+ data.tar.gz: 2f845bc5cd816a4c0899ed98f9cb9ac549c9513d5b79e7336876c39362ac72ca
5
5
  SHA512:
6
- metadata.gz: 3c347ef635cb9c1adeb14df10b7ce8a6b6220104f93b0fc4b7debccf30c0537da86e232a4b62000fb3498253c8ae5535c5ca9ddb9290371d3afd4dd021c894ab
7
- data.tar.gz: 3b1685788f672eadfe558f2bbceacdfd287dbcb865b7e2bbd8b718e710cf3870a575900bcae69452ebc9a720f8ca77c4c0f2b4488d56431e6c5a30946f4d630b
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
@@ -24,7 +24,8 @@ module NoFlyList
24
24
  restrict_to_existing: options.fetch(:restrict_to_existing, false),
25
25
  limit: options.fetch(:limit, nil),
26
26
  case_sensitive: options.fetch(:case_sensitive, true),
27
- adapter: @adapter
27
+ adapter: @adapter,
28
+ counter_cache: options.fetch(:counter_cache, false)
28
29
  }
29
30
  end
30
31
 
@@ -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)
@@ -94,7 +94,8 @@ module NoFlyList
94
94
 
95
95
  belongs_to :taggable,
96
96
  class_name: setup.taggable_klass.name,
97
- foreign_key: "taggable_id"
97
+ foreign_key: "taggable_id",
98
+ counter_cache: setup.counter_cache ? "#{setup.context}_count" : nil
98
99
 
99
100
  include NoFlyList::TaggingRecord
100
101
  end
@@ -156,12 +157,14 @@ module NoFlyList
156
157
 
157
158
  # Sets up associations for local (non-global) tags
158
159
  def setup_local_tag_associations(setup, singular_name)
160
+ plural_name = setup.context.to_s
159
161
  # Set up tag class associations
160
162
  setup.tag_class_name.constantize.class_eval do
161
163
  has_many :"#{singular_name}_taggings",
162
164
  -> { where(context: singular_name) },
163
165
  class_name: setup.tagging_class_name,
164
166
  foreign_key: "tag_id",
167
+ counter_cache: setup.counter_cache ? "#{plural_name}_count" : nil,
165
168
  dependent: :destroy
166
169
 
167
170
  has_many :"#{singular_name}_taggables",
@@ -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,17 +2,50 @@
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,
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
13
44
  @transformer = options.fetch(:transformer, ApplicationTagTransformer)
14
45
  @polymorphic = options.fetch(:polymorphic, false)
15
46
  @restrict_to_existing = options.fetch(:restrict_to_existing, false)
47
+ @counter_cache = options.fetch(:counter_cache, false)
48
+ @counter_cache_column = "#{context}_count"
16
49
  @limit = options.fetch(:limit, nil)
17
50
  @tag_class_name = determine_tag_class_name(taggable_klass, options)
18
51
  @tagging_class_name = determine_tagging_class_name(taggable_klass, options)
@@ -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
@@ -74,8 +87,11 @@ module NoFlyList
74
87
  end
75
88
 
76
89
  # Clear existing tags
90
+ old_count = model.send(context_taggings).count
77
91
  model.send(context_taggings).delete_all
78
92
 
93
+ # Update counter
94
+ model.update_column("#{@context}_count", 0) if setup[:counter_cache]
79
95
  # Create new tags
80
96
  @pending_changes.each do |tag_name|
81
97
  tag = find_or_create_tag(tag_name)
@@ -95,6 +111,8 @@ module NoFlyList
95
111
  model.send(context_taggings).create!(attributes)
96
112
  end
97
113
  end
114
+ # Update counter to match the actual count
115
+ model.update_column("#{@context}_count", @pending_changes.size) if setup[:counter_cache]
98
116
 
99
117
  refresh_from_database
100
118
  true
@@ -147,10 +165,20 @@ module NoFlyList
147
165
  "#<#{self.class.name} tags=#{current_list.inspect} transformer_with=#{transformer_name} >"
148
166
  end
149
167
 
150
- def add(tag)
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
174
+ def add(*tags)
151
175
  return self if limit_reached?
152
176
 
153
- new_tags = transformer.parse_tags(tag)
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
154
182
  return self if new_tags.empty?
155
183
 
156
184
  @pending_changes = current_list + new_tags
@@ -158,14 +186,26 @@ module NoFlyList
158
186
  self
159
187
  end
160
188
 
161
- def add!(tag)
162
- add(tag)
189
+ def add!(*tags)
190
+ add(*tags)
163
191
  save
164
192
  end
165
193
 
166
- 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)
167
202
  old_list = current_list.dup
168
- @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
169
209
  mark_record_dirty if @pending_changes != old_list
170
210
  self
171
211
  end
@@ -175,27 +215,47 @@ module NoFlyList
175
215
  save
176
216
  end
177
217
 
218
+ # Clears all tags
219
+ # @return [TaggingProxy] Returns self for method chaining
220
+ # @example Clear all tags
221
+ # tags.clear #=> []
178
222
  def clear
179
223
  old_list = current_list.dup
180
224
  @pending_changes = []
181
225
  mark_record_dirty if @pending_changes != old_list
226
+ model.write_attribute("#{@context}_count", 0) if setup[:counter_cache]
182
227
  self
183
228
  end
184
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
185
235
  def clear!
186
236
  @model.send(@context.to_s).destroy_all
187
237
  @pending_changes = []
238
+ @model.update_column("#{@context}_count", 0) if setup[:counter_cache]
188
239
  self
189
240
  end
190
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
191
245
  def include?(tag)
192
246
  current_list.include?(tag.to_s.strip)
193
247
  end
194
248
 
249
+ # Checks if tag list is empty
250
+ # @return [Boolean] True if no tags exist
195
251
  def empty?
196
252
  current_list.empty?
197
253
  end
198
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
199
259
  def persisted?
200
260
  false
201
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.3.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.3.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