Sync IRB f9347b1 (#10611)

This commit is contained in:
Stan Lo 2024-04-24 05:00:56 +08:00 committed by GitHub
parent b9109b270d
commit e11237904c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
8 changed files with 190 additions and 7 deletions

View File

@ -24,7 +24,9 @@ module IRB
def help_message
commands_info = IRB::Command.all_commands_info
helper_methods_info = IRB::HelperMethod.all_helper_methods_info
commands_grouped_by_categories = commands_info.group_by { |cmd| cmd[:category] }
commands_grouped_by_categories["Helper methods"] = helper_methods_info
user_aliases = irb_context.instance_variable_get(:@user_aliases)

View File

@ -98,10 +98,7 @@ module IRB
end
_register_with_aliases(:irb_context, Command::Context,
[
[:context, NO_OVERRIDE],
[:conf, NO_OVERRIDE],
],
[:context, NO_OVERRIDE]
)
_register_with_aliases(:irb_exit, Command::Exit,

29
lib/irb/helper_method.rb Normal file
View File

@ -0,0 +1,29 @@
require_relative "helper_method/base"
module IRB
module HelperMethod
@helper_methods = {}
class << self
attr_reader :helper_methods
def register(name, helper_class)
@helper_methods[name] = helper_class
if defined?(HelpersContainer)
HelpersContainer.install_helper_methods
end
end
def all_helper_methods_info
@helper_methods.map do |name, helper_class|
{ display_name: name, description: helper_class.description }
end
end
end
# Default helper_methods
require_relative "helper_method/conf"
register(:conf, HelperMethod::Conf)
end
end

View File

@ -0,0 +1,12 @@
module IRB
module HelperMethod
class Base
class << self
def description(description = nil)
@description = description if description
@description
end
end
end
end
end

View File

@ -0,0 +1,11 @@
module IRB
module HelperMethod
class Conf < Base
description "Returns the current context."
def execute
IRB.CurrentContext
end
end
end
end

View File

@ -6,6 +6,8 @@
require "delegate"
require_relative "helper_method"
IRB::TOPLEVEL_BINDING = binding
module IRB # :nodoc:
class WorkSpace
@ -109,9 +111,9 @@ EOF
attr_reader :main
def load_helper_methods_to_main
if !(class<<main;ancestors;end).include?(ExtendCommandBundle)
main.extend ExtendCommandBundle
end
ancestors = class<<main;ancestors;end
main.extend ExtendCommandBundle if !ancestors.include?(ExtendCommandBundle)
main.extend HelpersContainer if !ancestors.include?(HelpersContainer)
end
# Evaluate the given +statements+ within the context of this workspace.
@ -172,4 +174,16 @@ EOF
"\nFrom: #{file} @ line #{pos + 1} :\n\n#{body}#{Color.clear}\n"
end
end
module HelpersContainer
def self.install_helper_methods
HelperMethod.helper_methods.each do |name, helper_method_class|
define_method name do |*args, **opts, &block|
helper_method_class.new.execute(*args, **opts, &block)
end unless method_defined?(name)
end
end
install_helper_methods
end
end

View File

@ -62,5 +62,14 @@ module TestIRB
assert_match(/\$\s+Alias for `show_source`/, out)
assert_match(/@\s+Alias for `whereami`/, out)
end
def test_help_lists_helper_methods
out = run_ruby_file do
type "help"
type "exit"
end
assert_match(/Helper methods\s+conf\s+Returns the current context/, out)
end
end
end

View File

@ -0,0 +1,109 @@
# frozen_string_literal: true
require "irb"
require_relative "helper"
module TestIRB
class HelperMethodTestCase < TestCase
def setup
$VERBOSE = nil
@verbosity = $VERBOSE
save_encodings
IRB.instance_variable_get(:@CONF).clear
end
def teardown
$VERBOSE = @verbosity
restore_encodings
end
def execute_lines(*lines, conf: {}, main: self, irb_path: nil)
IRB.init_config(nil)
IRB.conf[:VERBOSE] = false
IRB.conf[:PROMPT_MODE] = :SIMPLE
IRB.conf.merge!(conf)
input = TestInputMethod.new(lines)
irb = IRB::Irb.new(IRB::WorkSpace.new(main), input)
irb.context.return_format = "=> %s\n"
irb.context.irb_path = irb_path if irb_path
IRB.conf[:MAIN_CONTEXT] = irb.context
IRB.conf[:USE_PAGER] = false
capture_output do
irb.eval_input
end
end
end
module TestHelperMethod
class ConfTest < HelperMethodTestCase
def test_conf_returns_the_context_object
out, err = execute_lines("conf.ap_name")
assert_empty err
assert_include out, "=> \"irb\""
end
end
end
class HelperMethodIntegrationTest < IntegrationTestCase
def test_arguments_propogation
write_ruby <<~RUBY
require "irb/helper_method"
class MyHelper < IRB::HelperMethod::Base
description "This is a test helper"
def execute(
required_arg, optional_arg = nil, *splat_arg, required_keyword_arg:,
optional_keyword_arg: nil, **double_splat_arg, &block_arg
)
puts [required_arg, optional_arg, splat_arg, required_keyword_arg, optional_keyword_arg, double_splat_arg, block_arg.call].to_s
end
end
IRB::HelperMethod.register(:my_helper, MyHelper)
binding.irb
RUBY
output = run_ruby_file do
type <<~INPUT
my_helper(
"required", "optional", "splat", required_keyword_arg: "required",
optional_keyword_arg: "optional", a: 1, b: 2
) { "block" }
INPUT
type "exit"
end
assert_include(output, '["required", "optional", ["splat"], "required", "optional", {:a=>1, :b=>2}, "block"]')
end
def test_helper_method_injection_can_happen_after_irb_require
write_ruby <<~RUBY
require "irb"
class MyHelper < IRB::HelperMethod::Base
description "This is a test helper"
def execute
puts "Hello from MyHelper"
end
end
IRB::HelperMethod.register(:my_helper, MyHelper)
binding.irb
RUBY
output = run_ruby_file do
type <<~INPUT
my_helper
INPUT
type "exit"
end
assert_include(output, 'Hello from MyHelper')
end
end
end