strong_migrations 0.7.0 → 2.2.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.
@@ -1,226 +1,61 @@
1
1
  # dependencies
2
2
  require "active_support"
3
3
 
4
+ # adapters
5
+ require_relative "strong_migrations/adapters/abstract_adapter"
6
+ require_relative "strong_migrations/adapters/mysql_adapter"
7
+ require_relative "strong_migrations/adapters/mariadb_adapter"
8
+ require_relative "strong_migrations/adapters/postgresql_adapter"
9
+
4
10
  # modules
5
- require "strong_migrations/checker"
6
- require "strong_migrations/database_tasks"
7
- require "strong_migrations/migration"
8
- require "strong_migrations/version"
11
+ require_relative "strong_migrations/checks"
12
+ require_relative "strong_migrations/safe_methods"
13
+ require_relative "strong_migrations/checker"
14
+ require_relative "strong_migrations/migration"
15
+ require_relative "strong_migrations/migration_context"
16
+ require_relative "strong_migrations/migrator"
17
+ require_relative "strong_migrations/version"
9
18
 
10
19
  # integrations
11
- require "strong_migrations/railtie" if defined?(Rails)
20
+ require_relative "strong_migrations/railtie" if defined?(Rails)
12
21
 
13
22
  module StrongMigrations
14
23
  class Error < StandardError; end
15
24
  class UnsafeMigration < Error; end
25
+ class UnsupportedVersion < Error; end
16
26
 
17
27
  class << self
18
28
  attr_accessor :auto_analyze, :start_after, :checks, :error_messages,
19
29
  :target_postgresql_version, :target_mysql_version, :target_mariadb_version,
20
- :enabled_checks, :lock_timeout, :statement_timeout, :check_down
30
+ :enabled_checks, :lock_timeout, :statement_timeout, :check_down, :target_version,
31
+ :safe_by_default, :target_sql_mode, :lock_timeout_retries, :lock_timeout_retry_delay,
32
+ :alphabetize_schema, :skipped_databases, :remove_invalid_indexes
21
33
  attr_writer :lock_timeout_limit
22
34
  end
23
35
  self.auto_analyze = false
24
36
  self.start_after = 0
37
+ self.lock_timeout_retries = 0
38
+ self.lock_timeout_retry_delay = 10 # seconds
25
39
  self.checks = []
26
- self.error_messages = {
27
- add_column_default:
28
- "Adding a column with a non-null default blocks %{rewrite_blocks} while the entire table is rewritten.
29
- Instead, add the column without a default value, then change the default.
30
-
31
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
32
- def up
33
- %{add_command}
34
- %{change_command}
35
- end
36
-
37
- def down
38
- %{remove_command}
39
- end
40
- end
41
-
42
- Then backfill the existing rows in the Rails console or a separate migration with disable_ddl_transaction!.
43
-
44
- class Backfill%{migration_name} < ActiveRecord::Migration%{migration_suffix}
45
- disable_ddl_transaction!
46
-
47
- def up
48
- %{code}
49
- end
50
- end",
51
-
52
- add_column_json:
53
- "There's no equality operator for the json column type, which can cause errors for
54
- existing SELECT DISTINCT queries in your application. Use jsonb instead.
55
-
56
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
57
- def change
58
- %{command}
59
- end
60
- end",
61
-
62
- change_column:
63
- "Changing the type of an existing column blocks %{rewrite_blocks}
64
- while the entire table is rewritten. A safer approach is to:
65
-
66
- 1. Create a new column
67
- 2. Write to both columns
68
- 3. Backfill data from the old column to the new column
69
- 4. Move reads from the old column to the new column
70
- 5. Stop writing to the old column
71
- 6. Drop the old column",
72
-
73
- change_column_with_not_null:
74
- "Changing the type is safe, but setting NOT NULL is not.",
75
-
76
- remove_column: "Active Record caches attributes, which causes problems
77
- when removing columns. Be sure to ignore the column%{column_suffix}:
78
-
79
- class %{model} < %{base_model}
80
- %{code}
81
- end
82
-
83
- Deploy the code, then wrap this step in a safety_assured { ... } block.
84
-
85
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
86
- def change
87
- safety_assured { %{command} }
88
- end
89
- end",
90
-
91
- rename_column:
92
- "Renaming a column that's in use will cause errors
93
- in your application. A safer approach is to:
94
-
95
- 1. Create a new column
96
- 2. Write to both columns
97
- 3. Backfill data from the old column to new column
98
- 4. Move reads from the old column to the new column
99
- 5. Stop writing to the old column
100
- 6. Drop the old column",
101
-
102
- rename_table:
103
- "Renaming a table that's in use will cause errors
104
- in your application. A safer approach is to:
105
-
106
- 1. Create a new table. Don't forget to recreate indexes from the old table
107
- 2. Write to both tables
108
- 3. Backfill data from the old table to new table
109
- 4. Move reads from the old table to the new table
110
- 5. Stop writing to the old table
111
- 6. Drop the old table",
112
-
113
- add_reference:
114
- "%{headline} Instead, use:
115
-
116
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
117
- disable_ddl_transaction!
118
-
119
- def change
120
- %{command}
121
- end
122
- end",
123
-
124
- add_index:
125
- "Adding an index non-concurrently blocks writes. Instead, use:
126
-
127
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
128
- disable_ddl_transaction!
129
-
130
- def change
131
- %{command}
132
- end
133
- end",
134
-
135
- remove_index:
136
- "Removing an index non-concurrently blocks writes. Instead, use:
137
-
138
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
139
- disable_ddl_transaction!
140
-
141
- def change
142
- %{command}
143
- end
144
- end",
145
-
146
- add_index_columns:
147
- "Adding a non-unique index with more than three columns rarely improves performance.
148
- Instead, start an index with columns that narrow down the results the most.",
149
-
150
- change_table:
151
- "Strong Migrations does not support inspecting what happens inside a
152
- change_table block, so cannot help you here. Please make really sure that what
153
- you're doing is safe before proceeding, then wrap it in a safety_assured { ... } block.",
154
-
155
- create_table:
156
- "The force option will destroy existing tables.
157
- If this is intended, drop the existing table first.
158
- Otherwise, remove the force option.",
159
-
160
- execute:
161
- "Strong Migrations does not support inspecting what happens inside an
162
- execute call, so cannot help you here. Please make really sure that what
163
- you're doing is safe before proceeding, then wrap it in a safety_assured { ... } block.",
164
-
165
- change_column_null:
166
- "Passing a default value to change_column_null runs a single UPDATE query,
167
- which can cause downtime. Instead, backfill the existing rows in the
168
- Rails console or a separate migration with disable_ddl_transaction!.
169
-
170
- class Backfill%{migration_name} < ActiveRecord::Migration%{migration_suffix}
171
- disable_ddl_transaction!
172
-
173
- def up
174
- %{code}
175
- end
176
- end",
177
-
178
- change_column_null_postgresql:
179
- "Setting NOT NULL on an existing column blocks reads and writes while every row is checked.
180
- Instead, add a check constraint and validate it in a separate migration.
181
-
182
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
183
- def change
184
- %{add_constraint_code}
185
- end
186
- end
187
-
188
- class Validate%{migration_name} < ActiveRecord::Migration%{migration_suffix}
189
- def change
190
- %{validate_constraint_code}
191
- end
192
- end",
193
-
194
- change_column_null_mysql:
195
- "Setting NOT NULL on an existing column is not safe with your database engine.",
196
-
197
- add_foreign_key:
198
- "Adding a foreign key blocks writes on both tables. Instead,
199
- add the foreign key without validating existing rows,
200
- then validate them in a separate migration.
201
-
202
- class %{migration_name} < ActiveRecord::Migration%{migration_suffix}
203
- def change
204
- %{add_foreign_key_code}
205
- end
206
- end
207
-
208
- class Validate%{migration_name} < ActiveRecord::Migration%{migration_suffix}
209
- def change
210
- %{validate_foreign_key_code}
211
- end
212
- end",
213
-
214
- validate_foreign_key:
215
- "Validating a foreign key while writes are blocked is dangerous.
216
- Use disable_ddl_transaction! or a separate migration."
217
- }
218
- self.enabled_checks = (error_messages.keys - [:remove_index]).map { |k| [k, {}] }.to_h
40
+ self.safe_by_default = false
219
41
  self.check_down = false
42
+ self.alphabetize_schema = false
43
+ self.skipped_databases = []
44
+ self.remove_invalid_indexes = false
220
45
 
221
46
  # private
222
47
  def self.developer_env?
223
- defined?(Rails) && (Rails.env.development? || Rails.env.test?)
48
+ env == "development" || env == "test"
49
+ end
50
+
51
+ # private
52
+ def self.env
53
+ if defined?(Rails.env)
54
+ Rails.env
55
+ else
56
+ # default to production for safety
57
+ ENV["RACK_ENV"] || "production"
58
+ end
224
59
  end
225
60
 
226
61
  def self.lock_timeout_limit
@@ -250,12 +85,20 @@ Use disable_ddl_transaction! or a separate migration."
250
85
  false
251
86
  end
252
87
  end
88
+
89
+ def self.skip_database(database)
90
+ self.skipped_databases << database
91
+ end
253
92
  end
254
93
 
94
+ # load error messages
95
+ require_relative "strong_migrations/error_messages"
96
+
255
97
  ActiveSupport.on_load(:active_record) do
256
98
  ActiveRecord::Migration.prepend(StrongMigrations::Migration)
99
+ ActiveRecord::MigrationContext.prepend(StrongMigrations::MigrationContext)
100
+ ActiveRecord::Migrator.prepend(StrongMigrations::Migrator)
257
101
 
258
- if defined?(ActiveRecord::Tasks::DatabaseTasks)
259
- ActiveRecord::Tasks::DatabaseTasks.singleton_class.prepend(StrongMigrations::DatabaseTasks)
260
- end
102
+ require_relative "strong_migrations/schema_dumper"
103
+ ActiveRecord::SchemaDumper.prepend(StrongMigrations::SchemaDumper)
261
104
  end
@@ -1,14 +1,9 @@
1
1
  namespace :strong_migrations do
2
- # https://www.pgrs.net/2008/03/13/alphabetize-schema-rb-columns/
2
+ # https://www.pgrs.net/2008/03/12/alphabetize-schema-rb-columns/
3
3
  task :alphabetize_columns do
4
4
  $stderr.puts "Dumping schema"
5
5
  ActiveRecord::Base.logger.level = Logger::INFO
6
6
 
7
- require "strong_migrations/alphabetize_columns"
8
- ActiveRecord::Base.connection.class.prepend StrongMigrations::AlphabetizeColumns
9
- if ActiveRecord::ConnectionAdapters.const_defined?('PostGISAdapter')
10
- ActiveRecord::ConnectionAdapters::PostGISAdapter.prepend StrongMigrations::AlphabetizeColumns
11
- end
12
- ActiveRecord::ConnectionAdapters::AbstractAdapter.prepend StrongMigrations::AlphabetizeColumns
7
+ StrongMigrations.alphabetize_schema = true
13
8
  end
14
9
  end
metadata CHANGED
@@ -1,16 +1,15 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: strong_migrations
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.7.0
4
+ version: 2.2.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Andrew Kane
8
8
  - Bob Remeika
9
9
  - David Waller
10
- autorequire:
11
10
  bindir: bin
12
11
  cert_chain: []
13
- date: 2020-07-23 00:00:00.000000000 Z
12
+ date: 2025-02-01 00:00:00.000000000 Z
14
13
  dependencies:
15
14
  - !ruby/object:Gem::Dependency
16
15
  name: activerecord
@@ -18,87 +17,16 @@ dependencies:
18
17
  requirements:
19
18
  - - ">="
20
19
  - !ruby/object:Gem::Version
21
- version: '5'
20
+ version: '7'
22
21
  type: :runtime
23
22
  prerelease: false
24
23
  version_requirements: !ruby/object:Gem::Requirement
25
24
  requirements:
26
25
  - - ">="
27
26
  - !ruby/object:Gem::Version
28
- version: '5'
29
- - !ruby/object:Gem::Dependency
30
- name: bundler
31
- requirement: !ruby/object:Gem::Requirement
32
- requirements:
33
- - - ">="
34
- - !ruby/object:Gem::Version
35
- version: '0'
36
- type: :development
37
- prerelease: false
38
- version_requirements: !ruby/object:Gem::Requirement
39
- requirements:
40
- - - ">="
41
- - !ruby/object:Gem::Version
42
- version: '0'
43
- - !ruby/object:Gem::Dependency
44
- name: rake
45
- requirement: !ruby/object:Gem::Requirement
46
- requirements:
47
- - - ">="
48
- - !ruby/object:Gem::Version
49
- version: '0'
50
- type: :development
51
- prerelease: false
52
- version_requirements: !ruby/object:Gem::Requirement
53
- requirements:
54
- - - ">="
55
- - !ruby/object:Gem::Version
56
- version: '0'
57
- - !ruby/object:Gem::Dependency
58
- name: minitest
59
- requirement: !ruby/object:Gem::Requirement
60
- requirements:
61
- - - ">="
62
- - !ruby/object:Gem::Version
63
- version: '0'
64
- type: :development
65
- prerelease: false
66
- version_requirements: !ruby/object:Gem::Requirement
67
- requirements:
68
- - - ">="
69
- - !ruby/object:Gem::Version
70
- version: '0'
71
- - !ruby/object:Gem::Dependency
72
- name: pg
73
- requirement: !ruby/object:Gem::Requirement
74
- requirements:
75
- - - ">="
76
- - !ruby/object:Gem::Version
77
- version: '0'
78
- type: :development
79
- prerelease: false
80
- version_requirements: !ruby/object:Gem::Requirement
81
- requirements:
82
- - - ">="
83
- - !ruby/object:Gem::Version
84
- version: '0'
85
- - !ruby/object:Gem::Dependency
86
- name: mysql2
87
- requirement: !ruby/object:Gem::Requirement
88
- requirements:
89
- - - ">="
90
- - !ruby/object:Gem::Version
91
- version: '0'
92
- type: :development
93
- prerelease: false
94
- version_requirements: !ruby/object:Gem::Requirement
95
- requirements:
96
- - - ">="
97
- - !ruby/object:Gem::Version
98
- version: '0'
99
- description:
27
+ version: '7'
100
28
  email:
101
- - andrew@chartkick.com
29
+ - andrew@ankane.org
102
30
  - bob.remeika@gmail.com
103
31
  executables: []
104
32
  extensions: []
@@ -111,18 +39,25 @@ files:
111
39
  - lib/generators/strong_migrations/install_generator.rb
112
40
  - lib/generators/strong_migrations/templates/initializer.rb.tt
113
41
  - lib/strong_migrations.rb
114
- - lib/strong_migrations/alphabetize_columns.rb
42
+ - lib/strong_migrations/adapters/abstract_adapter.rb
43
+ - lib/strong_migrations/adapters/mariadb_adapter.rb
44
+ - lib/strong_migrations/adapters/mysql_adapter.rb
45
+ - lib/strong_migrations/adapters/postgresql_adapter.rb
115
46
  - lib/strong_migrations/checker.rb
116
- - lib/strong_migrations/database_tasks.rb
47
+ - lib/strong_migrations/checks.rb
48
+ - lib/strong_migrations/error_messages.rb
117
49
  - lib/strong_migrations/migration.rb
50
+ - lib/strong_migrations/migration_context.rb
51
+ - lib/strong_migrations/migrator.rb
118
52
  - lib/strong_migrations/railtie.rb
53
+ - lib/strong_migrations/safe_methods.rb
54
+ - lib/strong_migrations/schema_dumper.rb
119
55
  - lib/strong_migrations/version.rb
120
56
  - lib/tasks/strong_migrations.rake
121
57
  homepage: https://github.com/ankane/strong_migrations
122
58
  licenses:
123
59
  - MIT
124
60
  metadata: {}
125
- post_install_message:
126
61
  rdoc_options: []
127
62
  require_paths:
128
63
  - lib
@@ -130,15 +65,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
130
65
  requirements:
131
66
  - - ">="
132
67
  - !ruby/object:Gem::Version
133
- version: '2.4'
68
+ version: '3.1'
134
69
  required_rubygems_version: !ruby/object:Gem::Requirement
135
70
  requirements:
136
71
  - - ">="
137
72
  - !ruby/object:Gem::Version
138
73
  version: '0'
139
74
  requirements: []
140
- rubygems_version: 3.1.2
141
- signing_key:
75
+ rubygems_version: 3.6.2
142
76
  specification_version: 4
143
77
  summary: Catch unsafe migrations in development
144
78
  test_files: []
@@ -1,11 +0,0 @@
1
- module StrongMigrations
2
- module AlphabetizeColumns
3
- def columns(*args)
4
- super.sort_by(&:name)
5
- end
6
-
7
- def extensions(*args)
8
- super.sort
9
- end
10
- end
11
- end