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 +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/config.rb +2 -1
- data/lib/no_fly_list/taggable_record/configuration.rb +7 -4
- 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 +36 -3
- data/lib/no_fly_list/taggable_record.rb +14 -0
- data/lib/no_fly_list/tagging_proxy.rb +66 -6
- 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: d5eef702cefcd730112adc305c5e17f8bb1c804d48fa3231d70549929a4f2ad9
|
4
|
+
data.tar.gz: 2f845bc5cd816a4c0899ed98f9cb9ac549c9513d5b79e7336876c39362ac72ca
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
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
|
@@ -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
|
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)
|
@@ -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
|
-
|
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
|
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
|
-
|
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 =
|
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!(
|
162
|
-
add(
|
189
|
+
def add!(*tags)
|
190
|
+
add(*tags)
|
163
191
|
save
|
164
192
|
end
|
165
193
|
|
166
|
-
|
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
|
-
|
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
|
+
|
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.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-
|
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
|