TL; DR Veuillez déplacer votre code de migration de données dans les tâches Rake ou utiliser des gemmes de style schéma à part entière. Couvrez cette logique avec des tests.
Je travaille en tant que développeur backend chez FunBox. Pour un certain nombre de projets, nous écrivons un backend Ruby On Rails. Nous nous efforçons de construire des processus de développement adéquats, par conséquent, face à un problème, nous essayons de le comprendre et d'élaborer des recommandations méthodologiques. C'est donc arrivé avec le problème de la migration des données. Une fois, j'ai effectué une migration de données dans une tâche Rake distincte couverte de tests, et l'équipe s'est posée une question: "Pourquoi pas dans la migration de schéma?" J'ai demandé aux développeurs dans le chat interne et, à ma grande surprise, les opinions étaient partagées. Il est devenu clair que la question était ambiguë et méritait une analyse approfondie et un article. Le programme maximum d'objectifs par article sera rempli pour moi lorsque quelqu'un cite un lien vers ce texte dans la revue de code en réponse à la question de savoir pourquoi une migration de données particulière a été supprimée ou, au contraire, pas retirée de la migration de schéma.
Digression lyrique
IT . Ruby, - . , , . , : , , , , , , .
, , . , , , , .
— , (views), , . .
— (, , , .) . . , . CI, , ( ) SQL , .
— . , DML- UPDATE
SQL. . .
(Continuous Delivery) — , .
Rails , , DDL-. , . , Rails omakase- . , .
, , . , . , , . .
. , . (, ) , .
, . . -, , .
deadlocks .
, , , Zero Downtime Migrations Strong Migrations.
— DSL (Domain Specific Language) Ruby DDL- SQL . DSL, , . , .
DSL, , SRP. . , , …
( , )
Ruby On Rails Data Migration , . , . Rails-, . .
SQL ORM ActiveRecord.
:
- . .
- , .
- callbacks , .
«» . , .
# db/migrate/20100513121110_add_flag_to_product.rb
class AddFlagToProduct < ActiveRecord::Migration
class Product < ActiveRecord::Base
end
def change
add_column :products, :flag, :boolean
Product.reset_column_information
Product.all.each do |product|
product.update_attributes!(:flag => false)
end
end
end
.
, each
find_each
c batch-.
SQL
, , SQL, :
- , . , , , (SQL), .
- JOIN-, , .
- , deadlock.
,
, .
, nullable- .
, .
, :
UPDATE table SET field = 'f' WHERE field IS NULL
:
class ClientDemandsMakeApprovedNullable < ActiveRecord::Migration
def up
change_column_null :client_demands, :approved, true
change_column_default :client_demands, :approved, nil
end
def down
execute("UPDATE client_demands SET approved = 'f' WHERE approved IS NULL")
change_column_null :client_demands, :approved, false
change_column_default :client_demands, :approved, false
end
end
, . , , . Dan Mayer Managing DB Schema & Data Changes Modifying Large Tables.
. «», . . , , , .
, .
, , .
, . . REPL .
, :
- ;
- ;
- .
. , . . .
, , , , .
Rake-
, — Rake-. . -.
Rake- . , . . . — .
, , Rake, Thoughtbot:
# lib/tasks/temporary/users.rake
namespace :users do
desc "Actualize achievements counter cache"
task actualize_achievements_counter_cache: :environment do
# C (ActiveRelation)
users = User.with_achievements
#
puts "Going to update #{users.count} users"
# , ,
# .
ActiveRecord::Base.transaction do
# Batch- find_each
users.find_each do |user|
#
user.actualize_achievements_counter_cache!
#
print "."
end
end
puts "Done!"
end
end
each
find_each
, . memory bloats. Akshay Mohite.
. , Rake- .
, . . , , . .
Mark Qualie up, . «» . :
class AddLastSmiledAtColumnToUsers < ActiveRecord::Migration[5.1]
def change
add_column :users, :last_smiled_at, :datetime
add_index :users, :last_smiled_at
end
class Data
def up
User.all.find_in_batches(batch_size: 250).each do |group|
ActiveRecord::Base.transaction do
group.each do |user|
user.last_smiled_at = user.smiles.last.created_at
user.save if user.changed?
end
end
end
end
end
end
:
Dir.glob("#{Rails.root}/db/migrate/*.rb").each { |file| require file }
AddLastSmiledAtColumnToUsers::Data.new.up
Job, .
, - , .
, , .
, . , .
data-migrate
(> 670), , Readme. Rails 5+.
, Rails 4+:
rails-data-migrations
(> 93 )
nonschema_migrations
(> 53 )
. .
, Rake-. . , .
db/data
, db/migrate
c :
rails g data_migration add_this_to_that
:
rake data:migrate
rake db:migrate:with_data
rake db:rollback:with_data
rake db:migrate:status:with_data
, .
Rails | Rake | ||||
---|---|---|---|---|---|
+ | - | - | - | - | |
Zero Downtime Deployment | - | + | + | + | + |
Test First | - | - | + | + | |
+ | - | - | + | ||
+ | - | + | + | + | |
+ | - | - | + | + |
.
— , :
- — ;
- Zero Downtime Deployment — , ;
- Test First — ;
- — ;
- — , ;
- — , , .
, Rake- — .
, . .
, — , . , .
- Rails Guides.
- Thoughtbot. Data Migrations in Rails.
- AtomicObject. Testing Data Migrations in Rails.
- Marcqualie. Rails Data Migrations.
- Ombulabs. Three Useful Data Migration Patterns for Rails.
- Strong Migrations.
- Zero Downtime Migrations.
- Dan Mayer. Managing DB Schema & Data Change.
Akshay Mohite find_each
each
.- . « . ». 1, 8 . , , .
UPD 2020-08-06: Extrapolator « ».
- , Test First .
.
UPD 2020-08-07: . , . . , .