aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--railties/CHANGELOG2
-rw-r--r--railties/lib/commands/plugin.rb155
2 files changed, 96 insertions, 61 deletions
diff --git a/railties/CHANGELOG b/railties/CHANGELOG
index b355e7ad70..89aa9568d8 100644
--- a/railties/CHANGELOG
+++ b/railties/CHANGELOG
@@ -1,5 +1,7 @@
*SVN*
+* Eliminate Subversion dependencies in scripts/plugin. Correct install options. Introduce --force option to reinstall a plugin. #2842 [Chad Fowler, Rick Olson]
+
* SQLite: the clone_structure_to_test and purge_test_database Rake tasks should always use the test environment. #2846 [Rick Olson]
* Make sure that legacy db tasks also reference :database for SQLite #2830 [kazuhiko@fdiary.net]
diff --git a/railties/lib/commands/plugin.rb b/railties/lib/commands/plugin.rb
index 6946f0b969..dd25c8e60f 100644
--- a/railties/lib/commands/plugin.rb
+++ b/railties/lib/commands/plugin.rb
@@ -31,10 +31,13 @@
# look like subversion repositories with plugins:
# http://wiki.rubyonrails.org/rails/pages/Plugins
#
+# * Unless you specify that you want to use svn, script/plugin uses plain ole
+# HTTP for downloads. The following bullets are true if you specify
+# that you want to use svn.
+#
# * If `vendor/plugins` is under subversion control, the script will
# modify the svn:externals property and perform an update. You can
-# use normal subversion commands to keep the plugins up to date or
-# you can use the built in update command.
+# use normal subversion commands to keep the plugins up to date.
#
# * Or, if `vendor/plugins` is not under subversion control, the
# plugin is pulled via `svn checkout` or `svn export` but looks
@@ -93,12 +96,13 @@ class RailsEnvironment
end
def use_svn?
- `svn --version` rescue nil
+ require 'active_support/core_ext/kernel'
+ silence_stderr {`svn --version` rescue nil}
!$?.nil? && $?.success?
end
def use_externals?
- File.directory?("#{root}/vendor/plugins/.svn")
+ use_svn? && File.directory?("#{root}/vendor/plugins/.svn")
end
def use_checkout?
@@ -155,40 +159,63 @@ class Plugin
or rails_env.externals.detect{ |name, repo| self.uri == repo }
end
- def install(method=nil)
+ def install(method=nil, options = {})
method ||= rails_env.best_install_method?
+
+ uninstall if installed? and options[:force]
+
unless installed?
- send("install_using_#{method}")
+ send("install_using_#{method}", options)
else
- puts "already installed: #{name} (#{uri})"
+ puts "already installed: #{name} (#{uri}). pass --force to reinstall"
end
end
-
+
+ def uninstall
+ path = "#{rails_env.root}/vendor/plugins/#{name}"
+ if File.directory?(path)
+ rm_r path
+ else
+ puts "Plugin doesn't exist: #{path}"
+ end
+ # clean up svn:externals
+ externals = rails_env.externals
+ externals.reject!{|n,u| name == n or name == u}
+ rails_env.externals = externals
+ end
+
private
- def install_using_export
- root = rails_env.root
- mkdir_p "#{root}/vendor/plugins"
- system("svn export #{uri} #{root}/vendor/plugins/#{name}")
+ def install_using_export(options = {})
+ svn_command :export, options
end
- def install_using_checkout
- root = rails_env.root
- mkdir_p "#{root}/vendor/plugins"
- system("svn checkout #{uri} #{root}/vendor/plugins/#{name}")
+ def install_using_checkout(options = {})
+ svn_command :checkout, options
end
- def install_using_externals
+ def install_using_externals(options = {})
externals = rails_env.externals
externals.push([@name, uri])
rails_env.externals = externals
- install_using_checkout
+ install_using_checkout(options)
end
- def install_using_http
+ def install_using_http(options = {})
root = rails_env.root
mkdir_p "#{root}/vendor/plugins"
Dir.chdir "#{root}/vendor/plugins"
- RecursiveHTTPFetcher.new(uri).fetch
+ fetcher = RecursiveHTTPFetcher.new(uri)
+ fetcher.quiet = true if options[:quiet]
+ fetcher.fetch
+ end
+
+ def svn_command(cmd, options = {})
+ root = rails_env.root
+ mkdir_p "#{root}/vendor/plugins"
+ base_cmd = "svn #{cmd} #{uri} #{root}/vendor/plugins/#{name}"
+ base_cmd += ' -q' if options[:quiet] and not $verbose
+ base_cmd += " -r #{options[:revision]}" if options[:revision]
+ system(base_cmd)
end
def guess_name(url)
@@ -239,7 +266,6 @@ class Repositories
return plugin if plugin.name == name
end
end
-
return nil
end
@@ -310,7 +336,7 @@ class Repository
puts index
end
- @plugins = index.split(/\n/).reject{ |line| line !~ /\/$/ }
+ @plugins = index.reject{ |line| line !~ /\/$/ }
@plugins.map! { |name| Plugin.new(File.join(@uri, name), name) }
end
@@ -323,7 +349,7 @@ class Repository
private
def index
- @index ||= `svn ls #{@uri}`
+ @index ||= RecursiveHTTPFetcher.new(@uri).ls
end
end
@@ -435,7 +461,6 @@ module Commands
@sources = []
@local = false
@remote = true
- @details = false
end
def options
@@ -453,8 +478,6 @@ module Commands
o.on( "--remote",
"List remotely availabled plugins. This is the default behavior",
"unless --local is provided.") {|@remote|}
- o.on( "-l", "--long",
- "Long listing / details about each plugin.") {|@details|}
end
end
@@ -469,14 +492,12 @@ module Commands
@sources.map{|r| r.plugins}.flatten.each do |plugin|
if @local or !plugin.installed?
puts plugin.to_s
- system "svn info #{plugin.uri}" if @details
end
end
else
cd "#{@base_command.environment.root}/vendor/plugins"
Dir["*"].select{|p| File.directory?(p)}.each do |name|
puts name
- system "svn info #{name}" if @details
end
end
end
@@ -644,7 +665,8 @@ module Commands
class Install
def initialize(base_command)
@base_command = base_command
- @method = :export
+ @method = :http
+ @options = { :quiet => false, :revision => nil, :force => false }
end
def options
@@ -660,6 +682,14 @@ module Commands
o.on( "-o", "--checkout",
"Use svn checkout to grab the plugin.",
"Enables updating but does not add a svn:externals entry.") { |v| @method = :checkout }
+ o.on( "-q", "--quiet",
+ "Suppresses the output from installation.",
+ "Ignored if -v is passed (./script/plugin -v install ...)") { |v| @options[:quiet] = true }
+ o.on( "-r REVISION", "--revision REVISION",
+ "Checks out the given revision from subversion.",
+ "Ignored if subversion is not used.") { |v| @options[:revision] = v }
+ o.on( "-f", "--force",
+ "Reinstalls a plugin if it's already installed.") { |v| @options[:force] = true }
o.separator ""
o.separator "You can specify plugin names as given in 'plugin list' output or absolute URLs to "
o.separator "a plugin repository."
@@ -672,8 +702,8 @@ module Commands
case
when (best == :http and @method != :http)
msg = "Cannot install using subversion because `svn' cannot be found in your PATH"
- when (best == :export and (@method != :export and method != :http))
- msg = "Cannot install using #{requested} because this project is not under subversion."
+ when (best == :export and (@method != :export and @method != :http))
+ msg = "Cannot install using #{@method} because this project is not under subversion."
when (best != :externals and @method == :externals)
msg = "Cannot install using externals because vendor/plugins is not under subversion."
end
@@ -691,11 +721,11 @@ module Commands
puts "Plugins will be installed using #{install_method}" if $verbose
args.each do |name|
if name =~ /\// then
- ::Plugin.new(name).install(install_method)
+ ::Plugin.new(name).install(install_method, @options)
else
plugin = Repositories.instance.find_plugin(name)
unless plugin.nil?
- plugin.install(install_method)
+ plugin.install(install_method, @options)
else
puts "Plugin not found: #{name}"
exit 1
@@ -705,39 +735,41 @@ module Commands
end
end
-
- class Remove
+ class Update
def initialize(base_command)
@base_command = base_command
end
-
+
def options
OptionParser.new do |o|
o.set_summary_indent(' ')
- o.banner = "Usage: #{@base_command.script_name} remove name [name]..."
- o.define_head "Remove plugins."
+ o.banner = "Usage: #{@base_command.script_name} update [name [name]...]"
+ o.on( "-r REVISION", "--revision REVISION",
+ "Checks out the given revision from subversion.",
+ "Ignored if subversion is not used.") { |v| @revision = v }
+ o.define_head "Update plugins."
end
end
-
+
def parse!(args)
options.parse!(args)
root = @base_command.environment.root
+ args = Dir["#{root}/vendor/plugins/*"].map do |f|
+ File.directory?("#{f}/.svn") ? File.basename(f) : nil
+ end.compact if args.empty?
+ cd "#{root}/vendor/plugins"
args.each do |name|
- path = "#{root}/vendor/plugins/#{name}"
- if File.directory?(path)
- rm_r path
+ if File.directory?(name)
+ puts "Updating plugin: #{name}"
+ system("svn #{$verbose ? '' : '-q'} up #{name} #{@revision ? "-r #{@revision}" : ''}")
else
- puts "Plugin doesn't exist: #{path}"
+ puts "Plugin doesn't exist: #{name}"
end
- # clean up svn:externals
- externals = @base_command.environment.externals
- externals.reject!{|n,u| name == n or name == u}
- @base_command.environment.externals = externals
end
end
end
- class Update
+ class Remove
def initialize(base_command)
@base_command = base_command
end
@@ -745,25 +777,16 @@ module Commands
def options
OptionParser.new do |o|
o.set_summary_indent(' ')
- o.banner = "Usage: #{@base_command.script_name} update [name [name]...]"
- o.define_head "Update plugins."
+ o.banner = "Usage: #{@base_command.script_name} remove name [name]..."
+ o.define_head "Remove plugins."
end
end
def parse!(args)
options.parse!(args)
root = @base_command.environment.root
- args = Dir["#{root}/vendor/plugins/*"].map do |f|
- File.directory?("#{f}/.svn") ? File.basename(f) : nil
- end.compact if args.empty?
- cd "#{root}/vendor/plugins"
args.each do |name|
- if File.directory?(name)
- puts "Updating plugin: #{name}"
- system("svn #{$verbose ? '' : '-q'} up #{name}")
- else
- puts "Plugin doesn't exist: #{name}"
- end
+ ::Plugin.new(name).uninstall
end
end
end
@@ -771,9 +794,19 @@ module Commands
end
class RecursiveHTTPFetcher
+ attr_accessor :quiet
def initialize(urls_to_fetch, cwd = ".")
@cwd = cwd
@urls_to_fetch = urls_to_fetch.to_a
+ @quiet = false
+ end
+
+ def ls
+ @urls_to_fetch.collect do |url|
+ open(url) do |stream|
+ links("", stream.read)
+ end rescue nil
+ end.flatten
end
def push_d(dir)
@@ -796,7 +829,7 @@ class RecursiveHTTPFetcher
end
def download(link)
- puts "+ #{File.join(@cwd, File.basename(link))}"
+ puts "+ #{File.join(@cwd, File.basename(link))}" unless @quiet
open(link) do |stream|
File.open(File.join(@cwd, File.basename(link)), "wb") do |file|
file.write(stream.read)