pg_sql_triggers 1.0.1 → 1.1.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/CHANGELOG.md +83 -0
- data/COVERAGE.md +58 -0
- data/Goal.md +180 -138
- data/README.md +6 -0
- data/app/controllers/pg_sql_triggers/dashboard_controller.rb +4 -1
- data/app/controllers/pg_sql_triggers/generator_controller.rb +67 -5
- data/app/models/pg_sql_triggers/trigger_registry.rb +73 -10
- data/app/views/pg_sql_triggers/generator/new.html.erb +18 -0
- data/app/views/pg_sql_triggers/generator/preview.html.erb +233 -13
- data/app/views/pg_sql_triggers/shared/_confirmation_modal.html.erb +32 -0
- data/config/initializers/pg_sql_triggers.rb +69 -0
- data/db/migrate/20251222000001_create_pg_sql_triggers_tables.rb +2 -0
- data/db/migrate/20251229071916_add_timing_to_pg_sql_triggers_registry.rb +8 -0
- data/docs/api-reference.md +22 -4
- data/docs/usage-guide.md +73 -0
- data/docs/web-ui.md +14 -0
- data/lib/generators/pg_sql_triggers/templates/create_pg_sql_triggers_tables.rb +2 -0
- data/lib/generators/pg_sql_triggers/templates/initializer.rb +8 -0
- data/lib/pg_sql_triggers/drift/db_queries.rb +116 -0
- data/lib/pg_sql_triggers/drift/detector.rb +187 -0
- data/lib/pg_sql_triggers/drift/reporter.rb +179 -0
- data/lib/pg_sql_triggers/drift.rb +14 -11
- data/lib/pg_sql_triggers/dsl/trigger_definition.rb +15 -1
- data/lib/pg_sql_triggers/generator/form.rb +3 -1
- data/lib/pg_sql_triggers/generator/service.rb +81 -25
- data/lib/pg_sql_triggers/migrator/pre_apply_comparator.rb +344 -0
- data/lib/pg_sql_triggers/migrator/pre_apply_diff_reporter.rb +143 -0
- data/lib/pg_sql_triggers/migrator/safety_validator.rb +258 -0
- data/lib/pg_sql_triggers/migrator.rb +58 -0
- data/lib/pg_sql_triggers/registry/manager.rb +96 -9
- data/lib/pg_sql_triggers/testing/function_tester.rb +66 -24
- data/lib/pg_sql_triggers/testing/syntax_validator.rb +24 -1
- data/lib/pg_sql_triggers/version.rb +1 -1
- data/lib/pg_sql_triggers.rb +12 -0
- data/scripts/generate_coverage_report.rb +129 -0
- metadata +12 -2
|
@@ -8,6 +8,7 @@ module PgSqlTriggers
|
|
|
8
8
|
end
|
|
9
9
|
|
|
10
10
|
# Test ONLY the function, not the trigger
|
|
11
|
+
# rubocop:disable Lint/UnusedMethodArgument
|
|
11
12
|
def test_function_only(test_context: {})
|
|
12
13
|
results = {
|
|
13
14
|
function_created: false,
|
|
@@ -16,15 +17,44 @@ module PgSqlTriggers
|
|
|
16
17
|
output: []
|
|
17
18
|
}
|
|
18
19
|
|
|
20
|
+
# Check if function_body is present
|
|
21
|
+
if @trigger.function_body.blank?
|
|
22
|
+
results[:success] = false
|
|
23
|
+
results[:errors] << "Function body is missing"
|
|
24
|
+
return results
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
# Extract function name to verify it matches
|
|
28
|
+
function_name_from_body = nil
|
|
29
|
+
if @trigger.function_body.present?
|
|
30
|
+
pattern = /CREATE\s+(?:OR\s+REPLACE\s+)?FUNCTION\s+([a-zA-Z_][a-zA-Z0-9_]*)\s*\(/i
|
|
31
|
+
match = @trigger.function_body.match(pattern)
|
|
32
|
+
function_name_from_body = match[1] if match
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
# If function_body doesn't contain a valid function definition, fail early
|
|
36
|
+
unless function_name_from_body
|
|
37
|
+
results[:success] = false
|
|
38
|
+
results[:errors] << "Function body does not contain a valid CREATE FUNCTION statement"
|
|
39
|
+
return results
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
# rubocop:disable Metrics/BlockLength
|
|
19
43
|
ActiveRecord::Base.transaction do
|
|
20
44
|
# Create function in transaction
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
45
|
+
begin
|
|
46
|
+
ActiveRecord::Base.connection.execute(@trigger.function_body)
|
|
47
|
+
results[:function_created] = true
|
|
48
|
+
results[:output] << "✓ Function created in test transaction"
|
|
49
|
+
rescue ActiveRecord::StatementInvalid, StandardError => e
|
|
50
|
+
results[:success] = false
|
|
51
|
+
results[:errors] << "Error during function creation: #{e.message}"
|
|
52
|
+
# Don't raise here, let it fall through to ensure block for rollback
|
|
53
|
+
end
|
|
24
54
|
|
|
25
55
|
# Try to invoke function directly (if test context provided)
|
|
26
56
|
# Note: Empty hash {} is not "present" in Rails, so check if it's not nil
|
|
27
|
-
if
|
|
57
|
+
if results[:function_created]
|
|
28
58
|
# This would require custom invocation logic
|
|
29
59
|
# For now, just verify it was created - if function was successfully created,
|
|
30
60
|
# we can assume it exists and is executable within the transaction
|
|
@@ -51,40 +81,50 @@ module PgSqlTriggers
|
|
|
51
81
|
end
|
|
52
82
|
|
|
53
83
|
# Verify function exists in database by checking pg_proc
|
|
54
|
-
# Since the function was created successfully (function_created is true),
|
|
55
|
-
# it exists and is executable
|
|
56
|
-
results[:function_executed] = true
|
|
57
|
-
|
|
58
84
|
# Try to verify via query if function_name is available
|
|
59
85
|
if function_name.present?
|
|
60
|
-
sanitized_name = ActiveRecord::Base.connection.quote_string(function_name)
|
|
61
|
-
check_sql = <<~SQL.squish
|
|
62
|
-
SELECT COUNT(*) as count
|
|
63
|
-
FROM pg_proc p
|
|
64
|
-
JOIN pg_namespace n ON p.pronamespace = n.oid
|
|
65
|
-
WHERE p.proname = '#{sanitized_name}'
|
|
66
|
-
AND n.nspname = 'public'
|
|
67
|
-
SQL
|
|
68
|
-
|
|
69
86
|
begin
|
|
87
|
+
sanitized_name = begin
|
|
88
|
+
ActiveRecord::Base.connection.quote_string(function_name)
|
|
89
|
+
rescue StandardError => e
|
|
90
|
+
# If quote_string fails, use the function name as-is (less safe but allows test to continue)
|
|
91
|
+
results[:errors] << "Error during function name sanitization: #{e.message}"
|
|
92
|
+
function_name
|
|
93
|
+
end
|
|
94
|
+
check_sql = <<~SQL.squish
|
|
95
|
+
SELECT COUNT(*) as count
|
|
96
|
+
FROM pg_proc p
|
|
97
|
+
JOIN pg_namespace n ON p.pronamespace = n.oid
|
|
98
|
+
WHERE p.proname = '#{sanitized_name}'
|
|
99
|
+
AND n.nspname = 'public'
|
|
100
|
+
SQL
|
|
101
|
+
|
|
70
102
|
result = ActiveRecord::Base.connection.execute(check_sql).first
|
|
71
|
-
results[:
|
|
103
|
+
results[:function_executed] = result && result["count"].to_i.positive?
|
|
104
|
+
results[:output] << if results[:function_executed]
|
|
72
105
|
"✓ Function exists and is callable"
|
|
73
106
|
else
|
|
74
107
|
"✓ Function created (verified via successful creation)"
|
|
75
108
|
end
|
|
76
|
-
rescue StandardError
|
|
77
|
-
results[:
|
|
109
|
+
rescue ActiveRecord::StatementInvalid, StandardError => e
|
|
110
|
+
results[:function_executed] = false
|
|
111
|
+
results[:success] = false
|
|
112
|
+
results[:errors] << "Error during function verification: #{e.message}"
|
|
113
|
+
results[:output] << "✓ Function created (verification failed)"
|
|
78
114
|
end
|
|
79
115
|
else
|
|
116
|
+
# If we can't extract function name, assume it was created successfully
|
|
117
|
+
# since function_created is true
|
|
118
|
+
results[:function_executed] = true
|
|
80
119
|
results[:output] << "✓ Function created (execution verified via successful creation)"
|
|
81
120
|
end
|
|
82
121
|
end
|
|
83
122
|
|
|
84
|
-
|
|
85
|
-
|
|
123
|
+
# Set success to true only if no errors occurred and function was created
|
|
124
|
+
results[:success] = results[:errors].empty? && results[:function_created]
|
|
125
|
+
rescue ActiveRecord::StatementInvalid, StandardError => e
|
|
86
126
|
results[:success] = false
|
|
87
|
-
results[:errors] << e.message
|
|
127
|
+
results[:errors] << e.message unless results[:errors].include?(e.message)
|
|
88
128
|
ensure
|
|
89
129
|
raise ActiveRecord::Rollback
|
|
90
130
|
end
|
|
@@ -92,6 +132,7 @@ module PgSqlTriggers
|
|
|
92
132
|
results[:output] << "\n⚠ Function rolled back (test mode)"
|
|
93
133
|
results
|
|
94
134
|
end
|
|
135
|
+
# rubocop:enable Lint/UnusedMethodArgument, Metrics/BlockLength
|
|
95
136
|
|
|
96
137
|
# Check if function already exists in database
|
|
97
138
|
def function_exists?
|
|
@@ -100,7 +141,8 @@ module PgSqlTriggers
|
|
|
100
141
|
rescue StandardError
|
|
101
142
|
{}
|
|
102
143
|
end
|
|
103
|
-
function_name = definition["function_name"]
|
|
144
|
+
function_name = definition["function_name"] || definition["name"] ||
|
|
145
|
+
definition[:function_name] || definition[:name]
|
|
104
146
|
return false if function_name.blank?
|
|
105
147
|
|
|
106
148
|
sanitized_name = ActiveRecord::Base.connection.quote_string(function_name)
|
|
@@ -65,10 +65,33 @@ module PgSqlTriggers
|
|
|
65
65
|
{}
|
|
66
66
|
end
|
|
67
67
|
function_name = definition["function_name"] || "test_validation_function"
|
|
68
|
+
events = Array(definition["events"] || [])
|
|
68
69
|
sanitized_table = ActiveRecord::Base.connection.quote_string(@trigger.table_name)
|
|
69
70
|
sanitized_function = ActiveRecord::Base.connection.quote_string(function_name)
|
|
70
71
|
sanitized_condition = @trigger.condition
|
|
71
72
|
|
|
73
|
+
# Check if condition references OLD values
|
|
74
|
+
condition_uses_old = sanitized_condition.match?(/\bOLD\./i)
|
|
75
|
+
|
|
76
|
+
# Determine which event to use for validation
|
|
77
|
+
# If condition uses OLD, we must use UPDATE or DELETE since INSERT doesn't have OLD
|
|
78
|
+
# If condition doesn't use OLD, we can use INSERT
|
|
79
|
+
if condition_uses_old
|
|
80
|
+
# Condition references OLD, so it can't be used with INSERT
|
|
81
|
+
if events.include?("insert")
|
|
82
|
+
return {
|
|
83
|
+
valid: false,
|
|
84
|
+
error: "WHEN condition cannot reference OLD values for INSERT triggers. " \
|
|
85
|
+
"Use UPDATE or DELETE events, or modify condition to only use NEW values."
|
|
86
|
+
}
|
|
87
|
+
end
|
|
88
|
+
# Use UPDATE for validation if available (it has OLD), otherwise use DELETE
|
|
89
|
+
test_event = events.include?("update") ? "UPDATE" : "DELETE"
|
|
90
|
+
else
|
|
91
|
+
# Condition doesn't reference OLD, so INSERT is fine
|
|
92
|
+
test_event = "INSERT"
|
|
93
|
+
end
|
|
94
|
+
|
|
72
95
|
# Validate condition by creating a temporary trigger with the condition
|
|
73
96
|
# This is the only way to validate WHEN conditions since they use NEW/OLD
|
|
74
97
|
test_function_sql = <<~SQL.squish
|
|
@@ -81,7 +104,7 @@ module PgSqlTriggers
|
|
|
81
104
|
|
|
82
105
|
test_trigger_sql = <<~SQL.squish
|
|
83
106
|
CREATE TRIGGER test_validation_trigger
|
|
84
|
-
BEFORE
|
|
107
|
+
BEFORE #{test_event} ON #{sanitized_table}
|
|
85
108
|
FOR EACH ROW
|
|
86
109
|
WHEN (#{sanitized_condition})
|
|
87
110
|
EXECUTE FUNCTION #{sanitized_function}();
|
data/lib/pg_sql_triggers.rb
CHANGED
|
@@ -9,6 +9,7 @@ module PgSqlTriggers
|
|
|
9
9
|
class DriftError < Error; end
|
|
10
10
|
class KillSwitchError < Error; end
|
|
11
11
|
class ValidationError < Error; end
|
|
12
|
+
class UnsafeMigrationError < Error; end
|
|
12
13
|
|
|
13
14
|
# Configuration
|
|
14
15
|
mattr_accessor :kill_switch_enabled
|
|
@@ -35,6 +36,17 @@ module PgSqlTriggers
|
|
|
35
36
|
mattr_accessor :excluded_tables
|
|
36
37
|
self.excluded_tables = []
|
|
37
38
|
|
|
39
|
+
mattr_accessor :allow_unsafe_migrations
|
|
40
|
+
self.allow_unsafe_migrations = false
|
|
41
|
+
|
|
42
|
+
# Drift states
|
|
43
|
+
DRIFT_STATE_IN_SYNC = "in_sync"
|
|
44
|
+
DRIFT_STATE_DRIFTED = "drifted"
|
|
45
|
+
DRIFT_STATE_MANUAL_OVERRIDE = "manual_override"
|
|
46
|
+
DRIFT_STATE_DISABLED = "disabled"
|
|
47
|
+
DRIFT_STATE_DROPPED = "dropped"
|
|
48
|
+
DRIFT_STATE_UNKNOWN = "unknown"
|
|
49
|
+
|
|
38
50
|
def self.configure
|
|
39
51
|
yield self
|
|
40
52
|
end
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
#!/usr/bin/env ruby
|
|
2
|
+
# frozen_string_literal: true
|
|
3
|
+
|
|
4
|
+
require "json"
|
|
5
|
+
require "pathname"
|
|
6
|
+
require "active_support/core_ext/object/blank"
|
|
7
|
+
|
|
8
|
+
def calculate_file_coverage(lines)
|
|
9
|
+
return 0.0 if lines.blank?
|
|
10
|
+
|
|
11
|
+
relevant_lines = lines.compact
|
|
12
|
+
return 0.0 if relevant_lines.empty?
|
|
13
|
+
|
|
14
|
+
covered = relevant_lines.count { |line| line&.positive? }
|
|
15
|
+
total = relevant_lines.count
|
|
16
|
+
|
|
17
|
+
(covered.to_f / total * 100).round(2)
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
def extract_file_path(full_path)
|
|
21
|
+
# Remove workspace root and get relative path
|
|
22
|
+
workspace_root = Pathname.new(__dir__).parent.realpath
|
|
23
|
+
file_path = Pathname.new(full_path)
|
|
24
|
+
|
|
25
|
+
if file_path.to_s.start_with?(workspace_root.to_s)
|
|
26
|
+
file_path.relative_path_from(workspace_root).to_s
|
|
27
|
+
else
|
|
28
|
+
# Fallback: try to extract relative path
|
|
29
|
+
full_path.sub(%r{.*/(lib|app)/}, '\1/')
|
|
30
|
+
end
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
def parse_coverage_data
|
|
34
|
+
coverage_dir = Pathname.new(__dir__).parent.join("coverage")
|
|
35
|
+
resultset_file = coverage_dir.join(".resultset.json")
|
|
36
|
+
last_run_file = coverage_dir.join(".last_run.json")
|
|
37
|
+
|
|
38
|
+
unless resultset_file.exist?
|
|
39
|
+
puts "Error: Coverage resultset file not found at #{resultset_file}"
|
|
40
|
+
exit 1
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
resultset_data = JSON.parse(File.read(resultset_file))
|
|
44
|
+
last_run_file.exist? ? JSON.parse(File.read(last_run_file)) : {}
|
|
45
|
+
|
|
46
|
+
# Get the first (and usually only) test suite result
|
|
47
|
+
test_suite_key = resultset_data.keys.first
|
|
48
|
+
coverage_data = resultset_data[test_suite_key]["coverage"] || {}
|
|
49
|
+
|
|
50
|
+
file_coverage = {}
|
|
51
|
+
total_lines = 0
|
|
52
|
+
total_covered = 0
|
|
53
|
+
|
|
54
|
+
coverage_data.each do |full_path, file_data|
|
|
55
|
+
next unless file_data["lines"]
|
|
56
|
+
|
|
57
|
+
relative_path = extract_file_path(full_path)
|
|
58
|
+
# Skip files in spec, vendor, etc.
|
|
59
|
+
next if relative_path.include?("/spec/") || relative_path.include?("/vendor/")
|
|
60
|
+
|
|
61
|
+
lines = file_data["lines"]
|
|
62
|
+
coverage_percentage = calculate_file_coverage(lines)
|
|
63
|
+
|
|
64
|
+
# Count relevant lines (non-nil lines)
|
|
65
|
+
relevant_lines = lines.compact
|
|
66
|
+
covered_lines = relevant_lines.count { |line| line&.positive? }
|
|
67
|
+
|
|
68
|
+
file_coverage[relative_path] = {
|
|
69
|
+
percentage: coverage_percentage,
|
|
70
|
+
total_lines: relevant_lines.count,
|
|
71
|
+
covered_lines: covered_lines,
|
|
72
|
+
missed_lines: relevant_lines.count - covered_lines
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
total_lines += relevant_lines.count
|
|
76
|
+
total_covered += covered_lines
|
|
77
|
+
end
|
|
78
|
+
|
|
79
|
+
total_coverage = total_lines.positive? ? (total_covered.to_f / total_lines * 100).round(2) : 0.0
|
|
80
|
+
|
|
81
|
+
{
|
|
82
|
+
file_coverage: file_coverage.sort_by { |_k, v| -v[:percentage] },
|
|
83
|
+
total_coverage: total_coverage,
|
|
84
|
+
total_lines: total_lines,
|
|
85
|
+
total_covered: total_covered
|
|
86
|
+
}
|
|
87
|
+
end
|
|
88
|
+
|
|
89
|
+
def generate_markdown(coverage_info)
|
|
90
|
+
markdown = "# Code Coverage Report\n\n"
|
|
91
|
+
markdown += "**Total Coverage: #{coverage_info[:total_coverage]}%**\n\n"
|
|
92
|
+
markdown += "Covered: #{coverage_info[:total_covered]} / #{coverage_info[:total_lines]} lines\n\n"
|
|
93
|
+
markdown += "---\n\n"
|
|
94
|
+
markdown += "## File Coverage\n\n"
|
|
95
|
+
markdown += "| File | Coverage | Covered Lines | Missed Lines | Total Lines |\n"
|
|
96
|
+
markdown += "|------|----------|---------------|--------------|-------------|\n"
|
|
97
|
+
|
|
98
|
+
coverage_info[:file_coverage].each do |file_path, data|
|
|
99
|
+
status_icon = if data[:percentage] >= 90
|
|
100
|
+
"✅"
|
|
101
|
+
else
|
|
102
|
+
data[:percentage] >= 70 ? "⚠️" : "❌"
|
|
103
|
+
end
|
|
104
|
+
markdown += "| `#{file_path}` | #{data[:percentage]}% #{status_icon} | " \
|
|
105
|
+
"#{data[:covered_lines]} | #{data[:missed_lines]} | #{data[:total_lines]} |\n"
|
|
106
|
+
end
|
|
107
|
+
|
|
108
|
+
markdown += "\n---\n\n"
|
|
109
|
+
markdown += "*Report generated automatically from SimpleCov results*\n"
|
|
110
|
+
markdown += "*To regenerate: Run `bundle exec rspec` and then `ruby scripts/generate_coverage_report.rb`*\n"
|
|
111
|
+
|
|
112
|
+
markdown
|
|
113
|
+
end
|
|
114
|
+
|
|
115
|
+
# Main execution
|
|
116
|
+
begin
|
|
117
|
+
coverage_info = parse_coverage_data
|
|
118
|
+
markdown_content = generate_markdown(coverage_info)
|
|
119
|
+
|
|
120
|
+
output_file = Pathname.new(__dir__).parent.join("COVERAGE.md")
|
|
121
|
+
File.write(output_file, markdown_content)
|
|
122
|
+
|
|
123
|
+
puts "Coverage report generated: #{output_file}"
|
|
124
|
+
puts "Total Coverage: #{coverage_info[:total_coverage]}%"
|
|
125
|
+
rescue StandardError => e
|
|
126
|
+
puts "Error generating coverage report: #{e.message}"
|
|
127
|
+
puts e.backtrace.first(5).join("\n")
|
|
128
|
+
exit 1
|
|
129
|
+
end
|
metadata
CHANGED
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: pg_sql_triggers
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
version: 1.0
|
|
4
|
+
version: 1.1.0
|
|
5
5
|
platform: ruby
|
|
6
6
|
authors:
|
|
7
7
|
- samaswin87
|
|
8
8
|
autorequire:
|
|
9
9
|
bindir: exe
|
|
10
10
|
cert_chain: []
|
|
11
|
-
date: 2025-12-
|
|
11
|
+
date: 2025-12-29 00:00:00.000000000 Z
|
|
12
12
|
dependencies:
|
|
13
13
|
- !ruby/object:Gem::Dependency
|
|
14
14
|
name: pg
|
|
@@ -190,6 +190,7 @@ files:
|
|
|
190
190
|
- ".rspec"
|
|
191
191
|
- ".rubocop.yml"
|
|
192
192
|
- CHANGELOG.md
|
|
193
|
+
- COVERAGE.md
|
|
193
194
|
- Goal.md
|
|
194
195
|
- LICENSE
|
|
195
196
|
- README.md
|
|
@@ -212,8 +213,10 @@ files:
|
|
|
212
213
|
- app/views/pg_sql_triggers/shared/_kill_switch_status.html.erb
|
|
213
214
|
- app/views/pg_sql_triggers/tables/index.html.erb
|
|
214
215
|
- app/views/pg_sql_triggers/tables/show.html.erb
|
|
216
|
+
- config/initializers/pg_sql_triggers.rb
|
|
215
217
|
- config/routes.rb
|
|
216
218
|
- db/migrate/20251222000001_create_pg_sql_triggers_tables.rb
|
|
219
|
+
- db/migrate/20251229071916_add_timing_to_pg_sql_triggers_registry.rb
|
|
217
220
|
- docs/README.md
|
|
218
221
|
- docs/api-reference.md
|
|
219
222
|
- docs/configuration.md
|
|
@@ -236,6 +239,9 @@ files:
|
|
|
236
239
|
- lib/pg_sql_triggers.rb
|
|
237
240
|
- lib/pg_sql_triggers/database_introspection.rb
|
|
238
241
|
- lib/pg_sql_triggers/drift.rb
|
|
242
|
+
- lib/pg_sql_triggers/drift/db_queries.rb
|
|
243
|
+
- lib/pg_sql_triggers/drift/detector.rb
|
|
244
|
+
- lib/pg_sql_triggers/drift/reporter.rb
|
|
239
245
|
- lib/pg_sql_triggers/dsl.rb
|
|
240
246
|
- lib/pg_sql_triggers/dsl/trigger_definition.rb
|
|
241
247
|
- lib/pg_sql_triggers/engine.rb
|
|
@@ -244,6 +250,9 @@ files:
|
|
|
244
250
|
- lib/pg_sql_triggers/generator/service.rb
|
|
245
251
|
- lib/pg_sql_triggers/migration.rb
|
|
246
252
|
- lib/pg_sql_triggers/migrator.rb
|
|
253
|
+
- lib/pg_sql_triggers/migrator/pre_apply_comparator.rb
|
|
254
|
+
- lib/pg_sql_triggers/migrator/pre_apply_diff_reporter.rb
|
|
255
|
+
- lib/pg_sql_triggers/migrator/safety_validator.rb
|
|
247
256
|
- lib/pg_sql_triggers/permissions.rb
|
|
248
257
|
- lib/pg_sql_triggers/permissions/checker.rb
|
|
249
258
|
- lib/pg_sql_triggers/registry.rb
|
|
@@ -258,6 +267,7 @@ files:
|
|
|
258
267
|
- lib/pg_sql_triggers/testing/syntax_validator.rb
|
|
259
268
|
- lib/pg_sql_triggers/version.rb
|
|
260
269
|
- lib/tasks/trigger_migrations.rake
|
|
270
|
+
- scripts/generate_coverage_report.rb
|
|
261
271
|
- sig/pg_sql_triggers.rbs
|
|
262
272
|
homepage: https://github.com/samaswin87/pg_sql_triggers
|
|
263
273
|
licenses:
|