francois-piston 2.0.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.
Files changed (73) hide show
  1. data/History.txt +19 -0
  2. data/License.txt +20 -0
  3. data/Manifest.txt +109 -0
  4. data/README.txt +136 -0
  5. data/VERSION.yml +4 -0
  6. data/bin/piston +5 -0
  7. data/lib/piston.rb +18 -0
  8. data/lib/piston/cli.rb +391 -0
  9. data/lib/piston/commands.rb +4 -0
  10. data/lib/piston/commands/base.rb +44 -0
  11. data/lib/piston/commands/convert.rb +26 -0
  12. data/lib/piston/commands/diff.rb +12 -0
  13. data/lib/piston/commands/import.rb +43 -0
  14. data/lib/piston/commands/info.rb +14 -0
  15. data/lib/piston/commands/lock_unlock.rb +21 -0
  16. data/lib/piston/commands/status.rb +40 -0
  17. data/lib/piston/commands/update.rb +34 -0
  18. data/lib/piston/commands/upgrade.rb +20 -0
  19. data/lib/piston/git.rb +13 -0
  20. data/lib/piston/git/client.rb +76 -0
  21. data/lib/piston/git/commit.rb +114 -0
  22. data/lib/piston/git/repository.rb +63 -0
  23. data/lib/piston/git/working_copy.rb +142 -0
  24. data/lib/piston/repository.rb +61 -0
  25. data/lib/piston/revision.rb +83 -0
  26. data/lib/piston/svn.rb +15 -0
  27. data/lib/piston/svn/client.rb +88 -0
  28. data/lib/piston/svn/repository.rb +67 -0
  29. data/lib/piston/svn/revision.rb +112 -0
  30. data/lib/piston/svn/working_copy.rb +182 -0
  31. data/lib/piston/version.rb +9 -0
  32. data/lib/piston/working_copy.rb +334 -0
  33. data/lib/subclass_responsibility_error.rb +2 -0
  34. data/test/integration_helpers.rb +35 -0
  35. data/test/spec_suite.rb +4 -0
  36. data/test/test_helper.rb +83 -0
  37. data/test/unit/git/commit/test_checkout.rb +31 -0
  38. data/test/unit/git/commit/test_each.rb +30 -0
  39. data/test/unit/git/commit/test_rememberance.rb +22 -0
  40. data/test/unit/git/commit/test_validation.rb +34 -0
  41. data/test/unit/git/repository/test_at.rb +23 -0
  42. data/test/unit/git/repository/test_basename.rb +12 -0
  43. data/test/unit/git/repository/test_branchanme.rb +15 -0
  44. data/test/unit/git/repository/test_guessing.rb +32 -0
  45. data/test/unit/git/working_copy/test_copying.rb +25 -0
  46. data/test/unit/git/working_copy/test_creation.rb +22 -0
  47. data/test/unit/git/working_copy/test_existence.rb +18 -0
  48. data/test/unit/git/working_copy/test_finalization.rb +15 -0
  49. data/test/unit/git/working_copy/test_guessing.rb +35 -0
  50. data/test/unit/git/working_copy/test_rememberance.rb +22 -0
  51. data/test/unit/svn/repository/test_at.rb +19 -0
  52. data/test/unit/svn/repository/test_basename.rb +24 -0
  53. data/test/unit/svn/repository/test_guessing.rb +45 -0
  54. data/test/unit/svn/revision/test_checkout.rb +28 -0
  55. data/test/unit/svn/revision/test_each.rb +22 -0
  56. data/test/unit/svn/revision/test_rememberance.rb +38 -0
  57. data/test/unit/svn/revision/test_validation.rb +50 -0
  58. data/test/unit/svn/working_copy/test_copying.rb +26 -0
  59. data/test/unit/svn/working_copy/test_creation.rb +16 -0
  60. data/test/unit/svn/working_copy/test_existence.rb +23 -0
  61. data/test/unit/svn/working_copy/test_externals.rb +56 -0
  62. data/test/unit/svn/working_copy/test_finalization.rb +17 -0
  63. data/test/unit/svn/working_copy/test_guessing.rb +18 -0
  64. data/test/unit/svn/working_copy/test_rememberance.rb +26 -0
  65. data/test/unit/test_info.rb +37 -0
  66. data/test/unit/test_lock_unlock.rb +47 -0
  67. data/test/unit/test_repository.rb +51 -0
  68. data/test/unit/test_revision.rb +31 -0
  69. data/test/unit/working_copy/test_guessing.rb +35 -0
  70. data/test/unit/working_copy/test_info.rb +14 -0
  71. data/test/unit/working_copy/test_rememberance.rb +42 -0
  72. data/test/unit/working_copy/test_validate.rb +63 -0
  73. metadata +178 -0
@@ -0,0 +1,37 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../test_helper")
2
+
3
+ class TestInfo < Piston::TestCase
4
+ def setup
5
+ super
6
+ @values = {"lock" => false}
7
+ @wcdir = "tmp/wcdir"
8
+ @wc = mock("WorkingCopy")
9
+ @wc.stubs(:validate!)
10
+ end
11
+
12
+ def test_info
13
+ run_and_verify do
14
+ @wc.expects(:info)
15
+ end
16
+ end
17
+
18
+ def test_validates_working_copy_before_working
19
+ assert_raise(Piston::WorkingCopy::NotWorkingCopy) do
20
+ run_and_verify do
21
+ @wc.expects(:validate!).raises(Piston::WorkingCopy::NotWorkingCopy)
22
+ end
23
+ end
24
+ end
25
+
26
+ private
27
+ def run_and_verify
28
+ yield
29
+ info_command.run(@wcdir)
30
+ end
31
+
32
+ def info_command
33
+ Piston::WorkingCopy.expects(:guess).with(@wcdir).returns(@wc)
34
+ Piston::Commands::Info.new(:verbose => "verbose",
35
+ :quiet => "quiet", :force => "force")
36
+ end
37
+ end
@@ -0,0 +1,47 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../test_helper")
2
+
3
+ class TestLockUnlock < Piston::TestCase
4
+ def setup
5
+ super
6
+ @values = {"lock" => false}
7
+ @wcdir = "tmp/wcdir"
8
+ @wc = mock("WorkingCopy")
9
+ @wc.stubs(:validate!)
10
+ end
11
+
12
+ def test_lock_working_copy
13
+ run_and_verify(true) do
14
+ @wc.expects(:recall).returns(@values)
15
+ @wc.expects(:finalize).returns(@values)
16
+ @wc.expects(:remember).with(@values.merge("lock" => true), @values["handler"]).returns(@values)
17
+ end
18
+ end
19
+
20
+ def test_unlock_working_copy
21
+ run_and_verify(false) do
22
+ @wc.expects(:recall).returns(@values)
23
+ @wc.expects(:finalize).returns(@values)
24
+ @wc.expects(:remember).with(@values.merge("lock" => false), @values["handler"]).returns(@values)
25
+ end
26
+ end
27
+
28
+ def test_validates_working_copy_before_working
29
+ assert_raise(Piston::WorkingCopy::NotWorkingCopy) do
30
+ run_and_verify do
31
+ @wc.expects(:validate!).raises(Piston::WorkingCopy::NotWorkingCopy)
32
+ end
33
+ end
34
+ end
35
+
36
+ private
37
+ def run_and_verify(lock=true)
38
+ yield
39
+ lock_unlock_command.run(lock)
40
+ end
41
+
42
+ def lock_unlock_command
43
+ Piston::WorkingCopy.expects(:guess).with(File.expand_path(@wcdir)).returns(@wc)
44
+ Piston::Commands::LockUnlock.new(:verbose => "verbose", :wcdir => @wcdir,
45
+ :quiet => "quiet", :force => "force")
46
+ end
47
+ end
@@ -0,0 +1,51 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../test_helper")
2
+
3
+ class TestRepository < Piston::TestCase
4
+ def setup
5
+ super
6
+ Piston::Repository.send(:handlers).clear
7
+ end
8
+
9
+ def test_guess_asks_each_handler_in_turn
10
+ Piston::Repository.add_handler(handler = mock("handler"))
11
+ handler.expects(:understands_url?).with("http://a.repos.com/trunk").returns(false)
12
+ assert_raise Piston::Repository::UnhandledUrl do
13
+ Piston::Repository.guess("http://a.repos.com/trunk")
14
+ end
15
+ end
16
+
17
+ def test_guess_returns_first_handler_that_understands_the_url
18
+ url = "svn://a.repos.com/projects/libcalc/trunk"
19
+
20
+ handler = mock("handler")
21
+ handler.expects(:understands_url?).with(url).returns(true)
22
+ handler_instance = mock("handler_instance")
23
+ handler.expects(:new).with(url).returns(handler_instance)
24
+
25
+ Piston::Repository.add_handler handler
26
+ assert_equal handler_instance, Piston::Repository.guess(url)
27
+ end
28
+
29
+ def test_guess_raises_unhandled_url_exception_when_no_repository_handler_found
30
+ assert_raise(Piston::Repository::UnhandledUrl) do
31
+ Piston::Repository.guess("invalid")
32
+ end
33
+ end
34
+
35
+ def test_add_handler
36
+ Piston::Repository.add_handler(handler = mock("handler"))
37
+ assert_equal [handler], Piston::Repository.send(:handlers)
38
+ end
39
+
40
+ def test_initialize_stores_url_parameter_in_url_accessor
41
+ @repository = Piston::Repository.new("url")
42
+ assert_equal "url", @repository.url
43
+ end
44
+
45
+ def test_at_is_a_subclass_responsibility
46
+ @repository = Piston::Repository.new("url")
47
+ assert_raise(SubclassResponsibilityError) do
48
+ @repository.at(:any)
49
+ end
50
+ end
51
+ end
@@ -0,0 +1,31 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../test_helper")
2
+
3
+ class TestRevision < Piston::TestCase
4
+ def setup
5
+ super
6
+ @repository = mock("repository")
7
+ @revision_number = "1"
8
+ @revision = Piston::Revision.new(@repository, @revision_number)
9
+ end
10
+
11
+ def test_initialize
12
+ assert_equal @repository, @revision.repository
13
+ assert_equal @revision_number, @revision.revision
14
+ end
15
+
16
+ def test_repository
17
+ assert_equal @repository, @revision.repository
18
+ end
19
+
20
+ def test_revision
21
+ assert_equal @revision_number, @revision.revision
22
+ end
23
+
24
+ def test_name
25
+ assert_equal @revision_number, @revision.name
26
+ end
27
+
28
+ def test_default_recalled_values_are_an_empty_hash
29
+ assert_equal Hash.new, @revision.recalled_values
30
+ end
31
+ end
@@ -0,0 +1,35 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../../test_helper")
2
+
3
+ class TestWorkingCopyGuessing < Piston::TestCase
4
+ def setup
5
+ super
6
+ Piston::WorkingCopy.send(:handlers).clear
7
+ @dir = mkpath("tmp/wc")
8
+ end
9
+
10
+ def test_guess_when_no_handlers_raises
11
+ assert_raise Piston::WorkingCopy::UnhandledWorkingCopy do
12
+ Piston::WorkingCopy.guess(@dir)
13
+ end
14
+ end
15
+
16
+ def test_guess_asks_each_handler_in_turn
17
+ Piston::WorkingCopy.add_handler(handler = mock("handler"))
18
+ handler.stubs(:name).returns("aname")
19
+ handler.expects(:understands_dir?).with(@dir).returns(false)
20
+ assert_raise Piston::WorkingCopy::UnhandledWorkingCopy do
21
+ Piston::WorkingCopy.guess(@dir)
22
+ end
23
+ end
24
+
25
+ def test_guess_returns_first_handler_that_understands_the_url
26
+ handler = mock("handler")
27
+ handler.stubs(:name).returns("aname")
28
+ handler.expects(:understands_dir?).with(@dir).returns(true)
29
+ handler_instance = mock("handler_instance")
30
+ handler.expects(:new).with(File.expand_path(@dir)).returns(handler_instance)
31
+
32
+ Piston::WorkingCopy.add_handler handler
33
+ assert_equal handler_instance, Piston::WorkingCopy.guess(@dir)
34
+ end
35
+ end
@@ -0,0 +1,14 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../../test_helper")
2
+
3
+ class TestWorkingCopyInfo < Piston::TestCase
4
+ def setup
5
+ super
6
+ @path = mkpath("tmp/wc")
7
+ @wc = Piston::WorkingCopy.new(@path)
8
+ end
9
+
10
+ def test_info_recalls_values
11
+ @wc.expects(:recall).returns(values = mock("recalled values"))
12
+ assert_equal values, @wc.info
13
+ end
14
+ end
@@ -0,0 +1,42 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../../test_helper")
2
+
3
+ class TestWorkingCopyRememberance < Piston::TestCase
4
+ def setup
5
+ super
6
+ @wcdir = mkpath("tmp/wc")
7
+ @wc = Piston::WorkingCopy.new(@wcdir)
8
+ end
9
+
10
+ def test_remember_generates_piston_yml_file_in_wc
11
+ @wc.remember({}, "a" => "b")
12
+ assert((@wcdir + ".piston.yml").exist?, "tmp/wc/.piston.yml file doesn't exist")
13
+ end
14
+
15
+ def test_writes_values_as_yaml_under_handler_key
16
+ expected = {"a" => "b"}
17
+ @wc.remember({}, expected)
18
+ actual = YAML.load_file(@wcdir + ".piston.yml")
19
+ assert_equal expected, actual["handler"]
20
+ end
21
+
22
+ def test_remember_calls_after_remember_with_path_to_piston_yml_file
23
+ @wc.expects(:after_remember).with(Pathname.new(@wcdir + ".piston.yml"))
24
+ @wc.remember({}, "a" => "b")
25
+ end
26
+
27
+ def test_remember_with_two_args_remembers_handler_values_separately
28
+ values = {"lock" => true}
29
+ handler_values = {"a" => "b"}
30
+
31
+ @wc.remember(values, handler_values)
32
+
33
+ actual = YAML.load_file(@wcdir + ".piston.yml")
34
+ assert_equal values.merge("format" => 1, "handler" => handler_values), actual
35
+ end
36
+
37
+ def test_recall_returns_hash_of_values
38
+ values = {"a" => "b", "handler" => {"b" => "c"}}
39
+ YAML.expects(:load_file).with(@wcdir + ".piston.yml").returns(values)
40
+ assert_equal values, @wc.recall
41
+ end
42
+ end
@@ -0,0 +1,63 @@
1
+ require File.expand_path("#{File.dirname(__FILE__)}/../../test_helper")
2
+
3
+ class TestWorkingCopyValidate < Piston::TestCase
4
+ def setup
5
+ super
6
+ @wcdir = mkpath("tmp/wc")
7
+ @wc = Piston::WorkingCopy.new(@wcdir)
8
+ end
9
+
10
+ def test_exists_when_the_directory_exists
11
+ assert @wc.exist?, "WorkingCopy should have existed, since the directory exists"
12
+ end
13
+
14
+ def test_does_not_exist_when_directory_not_present
15
+ @wcdir.rmtree
16
+ deny @wc.exist?, "WorkingCopy should NOT have existed, since the directory does not exist"
17
+ end
18
+
19
+ def test_does_not_exist_when_directory_is_a_file
20
+ @wcdir.rmtree
21
+ touch(@wcdir)
22
+ deny @wc.exist?, "WorkingCopy should NOT have existed, since the directory is a file"
23
+ end
24
+
25
+ def test_is_pistonized_if_exist_and_has_piston_dot_yaml_file
26
+ @wc.stubs(:exist?).returns(true)
27
+ touch(@wcdir + ".piston.yml")
28
+ assert @wc.pistonized?, "WorkingCopy should be Pistonized, since the .piston.yml file exists"
29
+ end
30
+
31
+ def test_is_NOT_pistonized_if_no_piston_dot_yaml_file
32
+ @wc.stubs(:exist?).returns(true)
33
+ deny @wc.pistonized?, "WorkingCopy should not be Pistonized, since the .piston.yml does not file exists"
34
+ end
35
+
36
+ def test_is_NOT_pistonized_if_exist_returns_false
37
+ @wc.stubs(:exist?).returns(false)
38
+ deny @wc.pistonized?, "WorkingCopy should not be Pistonized, since #exist? returned false"
39
+ end
40
+
41
+ def test_is_NOT_pistonized_if_piston_dot_yaml_is_a_directory
42
+ @wc.stubs(:exist?).returns(true)
43
+ (@wcdir + ".piston.yml").mkdir
44
+ deny @wc.pistonized?, "WorkingCopy should not be Pistonized, since .piston.yml is a directory"
45
+ end
46
+
47
+ def test_validate_bang_returns_self_when_ok
48
+ @wc.stubs(:pistonized?).returns(true)
49
+ assert_equal @wc, @wc.validate!
50
+ end
51
+
52
+ def test_validate_bang_raises_not_working_copy_when_not_pistonized
53
+ @wc.stubs(:pistonized?).returns(false)
54
+ assert_raise(Piston::WorkingCopy::NotWorkingCopy) do
55
+ @wc.validate!
56
+ end
57
+ end
58
+
59
+ protected
60
+ def touch(path)
61
+ File.open(path, "wb") {|f| f.write ""}
62
+ end
63
+ end
metadata ADDED
@@ -0,0 +1,178 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: francois-piston
3
+ version: !ruby/object:Gem::Version
4
+ version: 2.0.0
5
+ platform: ruby
6
+ authors:
7
+ - "Fran\xC3\xA7ois Beausoleil"
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+
12
+ date: 2009-02-23 00:00:00 -08:00
13
+ default_executable: piston
14
+ dependencies:
15
+ - !ruby/object:Gem::Dependency
16
+ name: cucumber
17
+ type: :development
18
+ version_requirement:
19
+ version_requirements: !ruby/object:Gem::Requirement
20
+ requirements:
21
+ - - ">="
22
+ - !ruby/object:Gem::Version
23
+ version: 0.1.16
24
+ version:
25
+ - !ruby/object:Gem::Dependency
26
+ name: main
27
+ type: :runtime
28
+ version_requirement:
29
+ version_requirements: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: 2.8.3
34
+ version:
35
+ - !ruby/object:Gem::Dependency
36
+ name: log4r
37
+ type: :runtime
38
+ version_requirement:
39
+ version_requirements: !ruby/object:Gem::Requirement
40
+ requirements:
41
+ - - ">="
42
+ - !ruby/object:Gem::Version
43
+ version: 1.0.5
44
+ version:
45
+ - !ruby/object:Gem::Dependency
46
+ name: activesupport
47
+ type: :runtime
48
+ version_requirement:
49
+ version_requirements: !ruby/object:Gem::Requirement
50
+ requirements:
51
+ - - ">="
52
+ - !ruby/object:Gem::Version
53
+ version: 2.0.0
54
+ version:
55
+ description: Piston makes it easy to merge vendor branches into your own repository, without worrying about which revisions were grabbed or not. Piston will also keep your local changes in addition to the remote changes.
56
+ email: francois@teksol.info
57
+ executables:
58
+ - piston
59
+ extensions: []
60
+
61
+ extra_rdoc_files: []
62
+
63
+ files:
64
+ - History.txt
65
+ - License.txt
66
+ - Manifest.txt
67
+ - README.txt
68
+ - VERSION.yml
69
+ - bin/piston
70
+ - lib/piston
71
+ - lib/piston/cli.rb
72
+ - lib/piston/commands
73
+ - lib/piston/commands/base.rb
74
+ - lib/piston/commands/convert.rb
75
+ - lib/piston/commands/diff.rb
76
+ - lib/piston/commands/import.rb
77
+ - lib/piston/commands/info.rb
78
+ - lib/piston/commands/lock_unlock.rb
79
+ - lib/piston/commands/status.rb
80
+ - lib/piston/commands/update.rb
81
+ - lib/piston/commands/upgrade.rb
82
+ - lib/piston/commands.rb
83
+ - lib/piston/git
84
+ - lib/piston/git/client.rb
85
+ - lib/piston/git/commit.rb
86
+ - lib/piston/git/repository.rb
87
+ - lib/piston/git/working_copy.rb
88
+ - lib/piston/git.rb
89
+ - lib/piston/repository.rb
90
+ - lib/piston/revision.rb
91
+ - lib/piston/svn
92
+ - lib/piston/svn/client.rb
93
+ - lib/piston/svn/repository.rb
94
+ - lib/piston/svn/revision.rb
95
+ - lib/piston/svn/working_copy.rb
96
+ - lib/piston/svn.rb
97
+ - lib/piston/version.rb
98
+ - lib/piston/working_copy.rb
99
+ - lib/piston.rb
100
+ - lib/subclass_responsibility_error.rb
101
+ - test/integration_helpers.rb
102
+ - test/spec_suite.rb
103
+ - test/test_helper.rb
104
+ - test/unit
105
+ - test/unit/git
106
+ - test/unit/git/commit
107
+ - test/unit/git/commit/test_checkout.rb
108
+ - test/unit/git/commit/test_each.rb
109
+ - test/unit/git/commit/test_rememberance.rb
110
+ - test/unit/git/commit/test_validation.rb
111
+ - test/unit/git/repository
112
+ - test/unit/git/repository/test_at.rb
113
+ - test/unit/git/repository/test_basename.rb
114
+ - test/unit/git/repository/test_branchanme.rb
115
+ - test/unit/git/repository/test_guessing.rb
116
+ - test/unit/git/working_copy
117
+ - test/unit/git/working_copy/test_copying.rb
118
+ - test/unit/git/working_copy/test_creation.rb
119
+ - test/unit/git/working_copy/test_existence.rb
120
+ - test/unit/git/working_copy/test_finalization.rb
121
+ - test/unit/git/working_copy/test_guessing.rb
122
+ - test/unit/git/working_copy/test_rememberance.rb
123
+ - test/unit/svn
124
+ - test/unit/svn/repository
125
+ - test/unit/svn/repository/test_at.rb
126
+ - test/unit/svn/repository/test_basename.rb
127
+ - test/unit/svn/repository/test_guessing.rb
128
+ - test/unit/svn/revision
129
+ - test/unit/svn/revision/test_checkout.rb
130
+ - test/unit/svn/revision/test_each.rb
131
+ - test/unit/svn/revision/test_rememberance.rb
132
+ - test/unit/svn/revision/test_validation.rb
133
+ - test/unit/svn/working_copy
134
+ - test/unit/svn/working_copy/test_copying.rb
135
+ - test/unit/svn/working_copy/test_creation.rb
136
+ - test/unit/svn/working_copy/test_existence.rb
137
+ - test/unit/svn/working_copy/test_externals.rb
138
+ - test/unit/svn/working_copy/test_finalization.rb
139
+ - test/unit/svn/working_copy/test_guessing.rb
140
+ - test/unit/svn/working_copy/test_rememberance.rb
141
+ - test/unit/test_info.rb
142
+ - test/unit/test_lock_unlock.rb
143
+ - test/unit/test_repository.rb
144
+ - test/unit/test_revision.rb
145
+ - test/unit/working_copy
146
+ - test/unit/working_copy/test_guessing.rb
147
+ - test/unit/working_copy/test_info.rb
148
+ - test/unit/working_copy/test_rememberance.rb
149
+ - test/unit/working_copy/test_validate.rb
150
+ has_rdoc: true
151
+ homepage: http://francois.github.com/piston
152
+ post_install_message:
153
+ rdoc_options:
154
+ - --inline-source
155
+ - --charset=UTF-8
156
+ require_paths:
157
+ - lib
158
+ required_ruby_version: !ruby/object:Gem::Requirement
159
+ requirements:
160
+ - - ">="
161
+ - !ruby/object:Gem::Version
162
+ version: "0"
163
+ version:
164
+ required_rubygems_version: !ruby/object:Gem::Requirement
165
+ requirements:
166
+ - - ">="
167
+ - !ruby/object:Gem::Version
168
+ version: "0"
169
+ version:
170
+ requirements: []
171
+
172
+ rubyforge_project: piston
173
+ rubygems_version: 1.2.0
174
+ signing_key:
175
+ specification_version: 2
176
+ summary: Easy your vendor branch management worries
177
+ test_files: []
178
+