migration_revert 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
data/.gitignore ADDED
@@ -0,0 +1,17 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in migration_revert.gemspec
4
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2012 Marc-Andre Lafortune
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,66 @@
1
+ # MigrationRevert
2
+
3
+ Reversible migrations are very cool.
4
+
5
+ Some commands are not reversible, for example the removal of a table or a column.
6
+
7
+ Also, one downside is that it is now more difficult to write the reverse of a migration if `change` is used. When `up` and `down` were used, one could simply swap the code around.
8
+
9
+ These commits introduce `Migration#revert` that makes it trivial to revert a past migration, in part or in whole, or do the doing a reversible removal of a table/column.
10
+
11
+ Note that `revert` can even be called from legacy migrations using `up` & `down` and that it can revert legacy-style migrations too. For anyone changing their mind every second day, `revert` is fully nestable.
12
+
13
+ To have complete revertible capability, I would like to introduce a modified syntax for `change_column` that would allow it to be revertible; pull request upcoming when I get a chance...
14
+
15
+ ## Usage
16
+
17
+ Reverses the migration commands for the given block and
18
+ the given migrations.
19
+
20
+ The following migration will remove the table 'horses'
21
+ and create the table 'apples' on the way up, and the reverse
22
+ on the way down.
23
+
24
+ This command can be nested.
25
+
26
+ class FixTLMigration < ActiveRecord::Migration
27
+ def change
28
+ revert do
29
+ create_table(:horses) do |t|
30
+ t.text :content
31
+ t.datetime :remind_at
32
+ end
33
+ end
34
+ create_table(:apples) do |t|
35
+ t.string :variety
36
+ end
37
+ end
38
+ end
39
+
40
+ Or equivalently, if +TenderloveMigration+ is defined as in the
41
+ documentation for Migration:
42
+
43
+ class FixupTLMigration < ActiveRecord::Migration
44
+ def change
45
+ revert TenderloveMigration
46
+
47
+ create_table(:apples) do |t|
48
+ t.string :variety
49
+ end
50
+ end
51
+ end
52
+
53
+
54
+ ## Installation
55
+
56
+ Add this line to your application's Gemfile:
57
+
58
+ gem 'migration_revert'
59
+
60
+ And then execute:
61
+
62
+ $ bundle
63
+
64
+ Or install it yourself as:
65
+
66
+ $ gem install migration_revert
data/Rakefile ADDED
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env rake
2
+ require "bundler/gem_tasks"
@@ -0,0 +1,6 @@
1
+ require "migration_revert/version"
2
+
3
+ unless ActiveRecord::Migration::CommandRecorder.method_defined? :revert
4
+ require "migration_revert/ext/migration"
5
+ require "migration_revert/ext/command_recorder"
6
+ end
@@ -0,0 +1,52 @@
1
+ module ActiveRecord
2
+ class Migration
3
+ class CommandRecorder
4
+ def revert
5
+ @reverting = !@reverting
6
+ previous = @commands
7
+ @commands = []
8
+ yield
9
+ ensure
10
+ @commands = previous.concat(@commands.reverse)
11
+ @reverting = !@reverting
12
+ end
13
+
14
+ # record +command+. +command+ should be a method name and arguments.
15
+ # For example:
16
+ #
17
+ # recorder.record(:method_name, [:arg1, :arg2])
18
+ #
19
+ def record(*command, &block)
20
+ if @reverting
21
+ @commands << inverse_of(*command)
22
+ else
23
+ @commands << (command << block)
24
+ end
25
+ end
26
+
27
+ [:create_table, :create_join_table, :change_table, :rename_table, :add_column, :remove_column, :rename_index, :rename_column, :add_index, :remove_index, :add_timestamps, :remove_timestamps, :change_column, :change_column_default, :add_reference, :remove_reference].each do |method|
28
+ class_eval <<-EOV, __FILE__, __LINE__ + 1
29
+ def #{method}(*args, &block) # def create_table(*args, &block)
30
+ record(:"#{method}", args, &block) # record(:create_table, args, &block)
31
+ end # end
32
+ EOV
33
+ end
34
+ alias :add_belongs_to :add_reference
35
+ alias :remove_belongs_to :remove_reference
36
+
37
+ # Returns the inverse of the given command
38
+ #
39
+ # recorder.inverse_of(:rename_table, [:old, :new])
40
+ # # => [:rename_table, [:new, :old]]
41
+ #
42
+ # This method will raise an +IrreversibleMigration+ exception if it cannot
43
+ # invert the +commands+.
44
+ #
45
+ def inverse_of(name, args)
46
+ method = :"invert_#{name}"
47
+ raise IrreversibleMigration unless respond_to?(method, true)
48
+ send(method, args)
49
+ end
50
+ end
51
+ end
52
+ end
@@ -0,0 +1,118 @@
1
+ module ActiveRecord
2
+ class Migration
3
+ # Reverses the migration commands for the given block and
4
+ # the given migrations.
5
+ #
6
+ # The following migration will remove the table 'horses'
7
+ # and create the table 'apples' on the way up, and the reverse
8
+ # on the way down.
9
+ #
10
+ # This command can be nested.
11
+ #
12
+ # class FixTLMigration < ActiveRecord::Migration
13
+ # def change
14
+ # revert do
15
+ # create_table(:horses) do |t|
16
+ # t.text :content
17
+ # t.datetime :remind_at
18
+ # end
19
+ # end
20
+ # create_table(:apples) do |t|
21
+ # t.string :variety
22
+ # end
23
+ # end
24
+ # end
25
+ #
26
+ # Or equivalently, if +TenderloveMigration+ is defined as in the
27
+ # documentation for Migration:
28
+ #
29
+ # class FixupTLMigration < ActiveRecord::Migration
30
+ # def change
31
+ # revert TenderloveMigration
32
+ #
33
+ # create_table(:apples) do |t|
34
+ # t.string :variety
35
+ # end
36
+ # end
37
+ # end
38
+ #
39
+ def revert(*migration_classes)
40
+ run(*migration_classes.reverse, :revert => true) unless migration_classes.empty?
41
+ if block_given?
42
+ if @connection.respond_to? :revert
43
+ @connection.revert { yield }
44
+ else
45
+ recorder = CommandRecorder.new(@connection)
46
+ @connection = recorder
47
+ suppress_messages do
48
+ @connection.revert { yield }
49
+ end
50
+ @connection = recorder.delegate
51
+ recorder.commands.each do |cmd, args, block|
52
+ send(cmd, *args, &block)
53
+ end
54
+ end
55
+ end
56
+ end
57
+
58
+ def reverting?
59
+ @connection.respond_to?(:reverting) && @connection.reverting
60
+ end
61
+
62
+ # Runs the given migration classes.
63
+ # Last argument can specify options:
64
+ # - :direction (default is :up)
65
+ # - :revert (default is false)
66
+ #
67
+ def run(*migration_classes)
68
+ opts = migration_classes.extract_options!
69
+ dir = opts[:direction] || :up
70
+ dir = (dir == :down ? :up : :down) if opts[:revert]
71
+ if reverting?
72
+ # If in revert and going :up, say, we want to execute :down without reverting, so
73
+ revert { run(*migration_classes, direction: dir, revert: true) }
74
+ else
75
+ migration_classes.each do |migration_class|
76
+ migration_class.new.exec_migration(@connection, dir)
77
+ end
78
+ end
79
+ end
80
+
81
+ # Execute this migration in the named direction
82
+ def migrate(direction)
83
+ return unless respond_to?(direction)
84
+
85
+ case direction
86
+ when :up then announce "migrating"
87
+ when :down then announce "reverting"
88
+ end
89
+
90
+ time = nil
91
+ ActiveRecord::Base.connection_pool.with_connection do |conn|
92
+ time = Benchmark.measure do
93
+ exec_migration(conn, direction)
94
+ end
95
+ end
96
+
97
+ case direction
98
+ when :up then announce "migrated (%.4fs)" % time.real; write
99
+ when :down then announce "reverted (%.4fs)" % time.real; write
100
+ end
101
+ end
102
+
103
+ def exec_migration(conn, direction)
104
+ @connection = conn
105
+ if respond_to?(:change)
106
+ if direction == :down
107
+ revert { change }
108
+ else
109
+ change
110
+ end
111
+ else
112
+ send(direction)
113
+ end
114
+ ensure
115
+ @connection = nil
116
+ end
117
+ end
118
+ end
@@ -0,0 +1,3 @@
1
+ module MigrationRevert
2
+ VERSION = "1.0.0"
3
+ end
@@ -0,0 +1,18 @@
1
+ # -*- encoding: utf-8 -*-
2
+ require File.expand_path('../lib/migration_revert/version', __FILE__)
3
+
4
+ Gem::Specification.new do |gem|
5
+ gem.authors = ["Marc-Andre Lafortune"]
6
+ gem.email = ["github@marc-andre.ca"]
7
+ gem.description = %q{Revert ActiveRecord migrations}
8
+ gem.summary = %q{Revert ActiveRecord migrations in part or in whole}
9
+ gem.homepage = ""
10
+
11
+ gem.files = `git ls-files`.split($\)
12
+ gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
13
+ gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
14
+ gem.name = "migration_revert"
15
+ gem.require_paths = ["lib"]
16
+ gem.version = MigrationRevert::VERSION
17
+ gem.add_runtime_dependency('activerecord', [">= 3.1.0"])
18
+ end
metadata ADDED
@@ -0,0 +1,72 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: migration_revert
3
+ version: !ruby/object:Gem::Version
4
+ version: 1.0.0
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Marc-Andre Lafortune
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2012-08-07 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: activerecord
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ! '>='
20
+ - !ruby/object:Gem::Version
21
+ version: 3.1.0
22
+ type: :runtime
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ! '>='
28
+ - !ruby/object:Gem::Version
29
+ version: 3.1.0
30
+ description: Revert ActiveRecord migrations
31
+ email:
32
+ - github@marc-andre.ca
33
+ executables: []
34
+ extensions: []
35
+ extra_rdoc_files: []
36
+ files:
37
+ - .gitignore
38
+ - Gemfile
39
+ - LICENSE
40
+ - README.md
41
+ - Rakefile
42
+ - lib/migration_revert.rb
43
+ - lib/migration_revert/ext/command_recorder.rb
44
+ - lib/migration_revert/ext/migration.rb
45
+ - lib/migration_revert/version.rb
46
+ - migration_revert.gemspec
47
+ homepage: ''
48
+ licenses: []
49
+ post_install_message:
50
+ rdoc_options: []
51
+ require_paths:
52
+ - lib
53
+ required_ruby_version: !ruby/object:Gem::Requirement
54
+ none: false
55
+ requirements:
56
+ - - ! '>='
57
+ - !ruby/object:Gem::Version
58
+ version: '0'
59
+ required_rubygems_version: !ruby/object:Gem::Requirement
60
+ none: false
61
+ requirements:
62
+ - - ! '>='
63
+ - !ruby/object:Gem::Version
64
+ version: '0'
65
+ requirements: []
66
+ rubyforge_project:
67
+ rubygems_version: 1.8.24
68
+ signing_key:
69
+ specification_version: 3
70
+ summary: Revert ActiveRecord migrations in part or in whole
71
+ test_files: []
72
+ has_rdoc: