no_fly_list 0.4.0 → 0.6.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 +4 -4
- data/lib/generators/no_fly_list/templates/tag_transformer.rb +2 -2
- data/lib/no_fly_list/railties/tasks.rake +51 -72
- data/lib/no_fly_list/tag_record.rb +5 -1
- data/lib/no_fly_list/taggable_record/configuration.rb +3 -3
- data/lib/no_fly_list/taggable_record/mutation.rb +6 -0
- data/lib/no_fly_list/taggable_record/query.rb +15 -0
- data/lib/no_fly_list/taggable_record/tag_setup.rb +34 -3
- data/lib/no_fly_list/taggable_record.rb +14 -0
- data/lib/no_fly_list/tagging_proxy.rb +56 -3
- data/lib/no_fly_list/task_helpers.rb +76 -0
- data/lib/no_fly_list/version.rb +1 -1
- data/lib/no_fly_list.rb +1 -0
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 5f1516f63b4ece62151501ea60f15ae1d5c2c3ea5738f22881ad5e2456a74b68
|
4
|
+
data.tar.gz: c9b88ce48a93a03fd21ade6eccfb0210e788c26ea78150ce92ee5ecf5f0ce8b0
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: ea013a28e4c1246f511ae736951a89a2c292668a6d3ce1ecbf2ada940cee52576144f5de4cfebd2f10a38feb3a80f94082cc1d6ab340d692fe14f75e329b0913
|
7
|
+
data.tar.gz: 9b714d9919bcf14c9fda34bada054f49af0202ab1189f0e487f38d1140df4f27e5580ef3b737adf0a59efc55e19bf381f74cdf2ccdeeb2fcf73d3a49b8135049
|
@@ -3,14 +3,14 @@
|
|
3
3
|
module ApplicationTagTransformer
|
4
4
|
module_function
|
5
5
|
|
6
|
-
# @param
|
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
|
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
|
2
|
+
desc "List all models using NoFlyList::TaggableRecord with details about their tagging configurations"
|
5
3
|
task taggable_records: :environment do
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
puts
|
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
|
16
|
+
desc "List all tag classes (both global and model-specific) with their inheritance chain"
|
19
17
|
task tag_records: :environment do
|
20
|
-
|
21
|
-
|
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
|
-
|
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
|
-
|
28
|
-
puts "
|
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 "
|
32
|
+
desc "Validate database schema for all taggable models"
|
33
33
|
task check_taggable_records: :environment do
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
puts "
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
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}
|
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
|
-
|
77
|
-
|
78
|
-
|
79
|
-
#
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
puts " #{
|
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
|
-
#
|
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
|
17
|
-
# @param
|
18
|
-
# @param
|
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
|
-
|
7
|
-
|
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.
|
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
|
-
|
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
|
-
|
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
|
+
|
data/lib/no_fly_list/version.rb
CHANGED
data/lib/no_fly_list.rb
CHANGED
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
|
+
version: 0.6.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-
|
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
|