2017-08-14 13:08:09 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-05-28 16:04:13 -04:00
|
|
|
require "thor"
|
|
|
|
require "erb"
|
|
|
|
|
|
|
|
require "active_support/core_ext/string/filters"
|
|
|
|
require "active_support/core_ext/string/inflections"
|
|
|
|
|
2017-10-21 09:08:33 -04:00
|
|
|
require "rails/command/actions"
|
2016-05-28 16:04:13 -04:00
|
|
|
|
|
|
|
module Rails
|
|
|
|
module Command
|
|
|
|
class Base < Thor
|
|
|
|
class Error < Thor::Error # :nodoc:
|
|
|
|
end
|
|
|
|
|
|
|
|
include Actions
|
|
|
|
|
|
|
|
class << self
|
2016-09-05 16:14:55 -04:00
|
|
|
# Returns true when the app is a Rails engine.
|
|
|
|
def engine?
|
|
|
|
defined?(ENGINE_ROOT)
|
|
|
|
end
|
|
|
|
|
2016-05-28 16:04:13 -04:00
|
|
|
# Tries to get the description from a USAGE file one folder above the command
|
|
|
|
# root.
|
2016-12-31 13:17:12 -05:00
|
|
|
def desc(usage = nil, description = nil, options = {})
|
2016-05-28 16:04:13 -04:00
|
|
|
if usage
|
|
|
|
super
|
|
|
|
else
|
|
|
|
@desc ||= ERB.new(File.read(usage_path)).result(binding) if usage_path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Convenience method to get the namespace from the class name. It's the
|
|
|
|
# same as Thor default except that the Command at the end of the class
|
|
|
|
# is removed.
|
|
|
|
def namespace(name = nil)
|
|
|
|
if name
|
|
|
|
super
|
|
|
|
else
|
|
|
|
@namespace ||= super.chomp("_command").sub(/:command:/, ":")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Convenience method to hide this command from the available ones when
|
|
|
|
# running rails command.
|
|
|
|
def hide_command!
|
|
|
|
Rails::Command.hidden_commands << self
|
|
|
|
end
|
|
|
|
|
|
|
|
def inherited(base) #:nodoc:
|
|
|
|
super
|
|
|
|
|
2019-11-13 13:20:29 -05:00
|
|
|
if base.name && !base.name.end_with?("Base")
|
2016-05-28 16:04:13 -04:00
|
|
|
Rails::Command.subclasses << base
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def perform(command, args, config) # :nodoc:
|
2017-02-23 12:15:28 -05:00
|
|
|
if Rails::Command::HELP_MAPPINGS.include?(args.first)
|
|
|
|
command, args = "help", []
|
|
|
|
end
|
2016-05-28 16:04:13 -04:00
|
|
|
|
|
|
|
dispatch(command, args.dup, nil, config)
|
|
|
|
end
|
|
|
|
|
|
|
|
def printing_commands
|
2017-02-26 07:05:13 -05:00
|
|
|
namespaced_commands
|
2016-05-28 16:04:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def executable
|
2018-06-29 11:51:57 -04:00
|
|
|
"rails #{command_name}"
|
2016-05-28 16:04:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Use Rails' default banner.
|
|
|
|
def banner(*)
|
2017-07-22 08:45:53 -04:00
|
|
|
"#{executable} #{arguments.map(&:usage).join(' ')} [options]".squish
|
2016-05-28 16:04:13 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Sets the base_name taking into account the current class namespace.
|
|
|
|
#
|
|
|
|
# Rails::Command::TestCommand.base_name # => 'rails'
|
|
|
|
def base_name
|
|
|
|
@base_name ||= begin
|
|
|
|
if base = name.to_s.split("::").first
|
|
|
|
base.underscore
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Return command name without namespaces.
|
|
|
|
#
|
|
|
|
# Rails::Command::TestCommand.command_name # => 'test'
|
|
|
|
def command_name
|
|
|
|
@command_name ||= begin
|
|
|
|
if command = name.to_s.split("::").last
|
|
|
|
command.chomp!("Command")
|
|
|
|
command.underscore
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Path to lookup a USAGE description in a file.
|
|
|
|
def usage_path
|
|
|
|
if default_command_root
|
|
|
|
path = File.join(default_command_root, "USAGE")
|
|
|
|
path if File.exist?(path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Default file root to place extra files a command might need, placed
|
|
|
|
# one folder above the command file.
|
|
|
|
#
|
2017-08-26 04:23:12 -04:00
|
|
|
# For a Rails::Command::TestCommand placed in <tt>rails/command/test_command.rb</tt>
|
|
|
|
# would return <tt>rails/test</tt>.
|
2016-05-28 16:04:13 -04:00
|
|
|
def default_command_root
|
2019-01-23 17:24:52 -05:00
|
|
|
path = File.expand_path(relative_command_path, __dir__)
|
2016-05-28 16:04:13 -04:00
|
|
|
path if File.exist?(path)
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
# Allow the command method to be called perform.
|
|
|
|
def create_command(meth)
|
|
|
|
if meth == "perform"
|
|
|
|
alias_method command_name, meth
|
|
|
|
else
|
|
|
|
# Prevent exception about command without usage.
|
|
|
|
# Some commands define their documentation differently.
|
|
|
|
@usage ||= ""
|
|
|
|
@desc ||= ""
|
|
|
|
|
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
2017-02-23 12:15:28 -05:00
|
|
|
|
|
|
|
def command_root_namespace
|
2019-01-23 17:24:52 -05:00
|
|
|
(namespace.split(":") - %w(rails)).join(":")
|
|
|
|
end
|
|
|
|
|
|
|
|
def relative_command_path
|
|
|
|
File.join("../commands", *command_root_namespace.split(":"))
|
2017-02-23 12:15:28 -05:00
|
|
|
end
|
2017-02-26 07:05:13 -05:00
|
|
|
|
|
|
|
def namespaced_commands
|
|
|
|
commands.keys.map do |key|
|
2019-01-23 17:24:52 -05:00
|
|
|
if command_root_namespace.match?(/(\A|\:)#{key}\z/)
|
|
|
|
command_root_namespace
|
|
|
|
else
|
|
|
|
"#{command_root_namespace}:#{key}"
|
|
|
|
end
|
2017-02-26 07:05:13 -05:00
|
|
|
end
|
|
|
|
end
|
2016-05-28 16:04:13 -04:00
|
|
|
end
|
2016-12-31 13:14:08 -05:00
|
|
|
|
|
|
|
def help
|
|
|
|
if command_name = self.class.command_name
|
|
|
|
self.class.command_help(shell, command_name)
|
|
|
|
else
|
|
|
|
super
|
|
|
|
end
|
|
|
|
end
|
2016-05-28 16:04:13 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|