183 lines
5.1 KiB
Ruby
183 lines
5.1 KiB
Ruby
require 'rspec/core/rake_task'
|
|
require 'bundler/setup'
|
|
require 'micro_migrations'
|
|
require 'travis'
|
|
|
|
ActiveRecord::Base.schema_format = :sql
|
|
Rails.application.config.paths["db/migrate"] = ["db/migrate", "#{Gem.loaded_specs['travis-core'].full_gem_path}/db/migrate"]
|
|
|
|
Rake::Task["db:structure:dump"].clear if ENV['RAILS_ENV=test']
|
|
|
|
desc 'Run specs'
|
|
RSpec::Core::RakeTask.new do |t|
|
|
t.pattern = './spec/**/*_spec.rb'
|
|
end
|
|
|
|
module ActiveRecord
|
|
class Migration
|
|
class << self
|
|
attr_accessor :disable_ddl_transaction
|
|
end
|
|
|
|
# Disable DDL transactions for this migration.
|
|
def self.disable_ddl_transaction!
|
|
@disable_ddl_transaction = true
|
|
end
|
|
|
|
def disable_ddl_transaction # :nodoc:
|
|
self.class.disable_ddl_transaction
|
|
end
|
|
end
|
|
|
|
class Migrator
|
|
def use_transaction?(migration)
|
|
!migration.disable_ddl_transaction && Base.connection.supports_ddl_transactions?
|
|
end
|
|
|
|
def ddl_transaction(migration, &block)
|
|
if use_transaction?(migration)
|
|
Base.transaction { block.call }
|
|
else
|
|
block.call
|
|
end
|
|
end
|
|
|
|
def migrate(&block)
|
|
current = migrations.detect { |m| m.version == current_version }
|
|
target = migrations.detect { |m| m.version == @target_version }
|
|
|
|
if target.nil? && @target_version && @target_version > 0
|
|
raise UnknownMigrationVersionError.new(@target_version)
|
|
end
|
|
|
|
start = up? ? 0 : (migrations.index(current) || 0)
|
|
finish = migrations.index(target) || migrations.size - 1
|
|
runnable = migrations[start..finish]
|
|
|
|
# skip the last migration if we're headed down, but not ALL the way down
|
|
runnable.pop if down? && target
|
|
|
|
ran = []
|
|
runnable.each do |migration|
|
|
if block && !block.call(migration)
|
|
next
|
|
end
|
|
|
|
Base.logger.info "Migrating to #{migration.name} (#{migration.version})" if Base.logger
|
|
|
|
seen = migrated.include?(migration.version.to_i)
|
|
|
|
# On our way up, we skip migrating the ones we've already migrated
|
|
next if up? && seen
|
|
|
|
# On our way down, we skip reverting the ones we've never migrated
|
|
if down? && !seen
|
|
migration.announce 'never migrated, skipping'; migration.write
|
|
next
|
|
end
|
|
|
|
begin
|
|
ddl_transaction(migration) do
|
|
migration.migrate(@direction)
|
|
record_version_state_after_migrating(migration.version)
|
|
end
|
|
ran << migration
|
|
rescue => e
|
|
canceled_msg = Base.connection.supports_ddl_transactions? ? "this and " : ""
|
|
raise StandardError, "An error has occurred, #{canceled_msg}all later migrations canceled:\n\n#{e}", e.backtrace
|
|
end
|
|
end
|
|
ran
|
|
end
|
|
end
|
|
|
|
class MigrationProxy
|
|
delegate :disable_ddl_transaction, to: :migration
|
|
end
|
|
end
|
|
|
|
task :default => :spec
|
|
|
|
module ActiveRecord
|
|
class Migration
|
|
class << self
|
|
attr_accessor :disable_ddl_transaction
|
|
end
|
|
|
|
# Disable DDL transactions for this migration.
|
|
def self.disable_ddl_transaction!
|
|
@disable_ddl_transaction = true
|
|
end
|
|
|
|
def disable_ddl_transaction # :nodoc:
|
|
self.class.disable_ddl_transaction
|
|
end
|
|
end
|
|
|
|
class Migrator
|
|
def use_transaction?(migration)
|
|
!migration.disable_ddl_transaction && Base.connection.supports_ddl_transactions?
|
|
end
|
|
|
|
def ddl_transaction(migration, &block)
|
|
if use_transaction?(migration)
|
|
Base.transaction { block.call }
|
|
else
|
|
block.call
|
|
end
|
|
end
|
|
|
|
def migrate(&block)
|
|
current = migrations.detect { |m| m.version == current_version }
|
|
target = migrations.detect { |m| m.version == @target_version }
|
|
|
|
if target.nil? && @target_version && @target_version > 0
|
|
raise UnknownMigrationVersionError.new(@target_version)
|
|
end
|
|
|
|
start = up? ? 0 : (migrations.index(current) || 0)
|
|
finish = migrations.index(target) || migrations.size - 1
|
|
runnable = migrations[start..finish]
|
|
|
|
# skip the last migration if we're headed down, but not ALL the way down
|
|
runnable.pop if down? && target
|
|
|
|
ran = []
|
|
runnable.each do |migration|
|
|
if block && !block.call(migration)
|
|
next
|
|
end
|
|
|
|
Base.logger.info "Migrating to #{migration.name} (#{migration.version})" if Base.logger
|
|
|
|
seen = migrated.include?(migration.version.to_i)
|
|
|
|
# On our way up, we skip migrating the ones we've already migrated
|
|
next if up? && seen
|
|
|
|
# On our way down, we skip reverting the ones we've never migrated
|
|
if down? && !seen
|
|
migration.announce 'never migrated, skipping'; migration.write
|
|
next
|
|
end
|
|
|
|
begin
|
|
ddl_transaction(migration) do
|
|
migration.migrate(@direction)
|
|
record_version_state_after_migrating(migration.version)
|
|
end
|
|
ran << migration
|
|
rescue => e
|
|
canceled_msg = Base.connection.supports_ddl_transactions? ? "this and " : ""
|
|
raise StandardError, "An error has occurred, #{canceled_msg}all later migrations canceled:\n\n#{e}", e.backtrace
|
|
end
|
|
end
|
|
ran
|
|
end
|
|
end
|
|
|
|
class MigrationProxy
|
|
delegate :disable_ddl_transaction, to: :migration
|
|
end
|
|
end
|