From 913f756d073f2214cf711d34b4c519f401dfcafe Mon Sep 17 00:00:00 2001 From: tompng Date: Tue, 22 Oct 2024 02:34:52 +0900 Subject: [PATCH] Fix for RBS::Types::UntypedFunction Fix completion of code including `Thread.new{}` and `proc.call` --- .github/workflows/test.yml | 2 +- lib/repl_type_completor/type_analyzer.rb | 2 +- lib/repl_type_completor/types.rb | 4 +++- test/repl_type_completor/test_type_analyze.rb | 10 +++++++++- 4 files changed, 14 insertions(+), 4 deletions(-) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 84a77c0..14bf715 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -35,7 +35,7 @@ jobs: strategy: fail-fast: false matrix: - rbs: ['latest', '3.4', '3.2', '3.0', '2.7.0'] + rbs: ['latest', '3.6', '3.4', '3.2', '3.0', '2.7.0'] env: GEMFILE_RBS_VERSION: ${{ matrix.rbs }} steps: diff --git a/lib/repl_type_completor/type_analyzer.rb b/lib/repl_type_completor/type_analyzer.rb index 2d60212..ecb5506 100644 --- a/lib/repl_type_completor/type_analyzer.rb +++ b/lib/repl_type_completor/type_analyzer.rb @@ -1145,7 +1145,7 @@ def method_call(receiver, method_name, args, kwargs, block, scope, name_match: t receiver_vars = receiver.is_a?(Types::InstanceType) ? receiver.params : {} free_vars = method.type.free_variables - receiver_vars.keys.to_set vars = receiver_vars.merge Types.match_free_variables(free_vars, method_params, given_params) - if block && method.block + if block && method.block && method.block.type.respond_to?(:required_positionals) params_type = method.block.type.required_positionals.map do |func_param| Types.from_rbs_type func_param.type, receiver, vars end diff --git a/lib/repl_type_completor/types.rb b/lib/repl_type_completor/types.rb index aed36d9..73b9160 100644 --- a/lib/repl_type_completor/types.rb +++ b/lib/repl_type_completor/types.rb @@ -83,7 +83,9 @@ def self.rbs_methods(type, method_name, args_types, kwargs_type, has_block) methods_with_score = receivers.flat_map do |receiver_type, klass, singleton| method = rbs_search_method klass, method_name, singleton next [] unless method - method.method_types.map do |method_type| + method.method_types.filter_map do |method_type| + next unless method_type.type.respond_to?(:required_positionals) + score = 0 score += 2 if !!method_type.block == has_block reqs = method_type.type.required_positionals diff --git a/test/repl_type_completor/test_type_analyze.rb b/test/repl_type_completor/test_type_analyze.rb index 8742603..9c047ae 100644 --- a/test/repl_type_completor/test_type_analyze.rb +++ b/test/repl_type_completor/test_type_analyze.rb @@ -13,7 +13,7 @@ def ReplTypeCompletor.handle_exception(e) end ReplTypeCompletor::Types.load_rbs_builder unless ReplTypeCompletor::Types.rbs_builder end - + def teardown ReplTypeCompletor.singleton_class.remove_method(:handle_exception) ReplTypeCompletor.define_singleton_method(:handle_exception, &@handle_exception_method) @@ -711,5 +711,13 @@ def test_array_aref assert_call('[1].[](0).', include: Integer, exclude: [Array, NilClass]) assert_call('[1].[](0){}.', include: Integer, exclude: [Array, NilClass]) end + + def test_rbs_untyped_function + # Block of Thread#initialize `(*untyped) { (?) -> void } -> void` is RBS::Types::UntypedFunction + assert_call('Thread.new{_1.', include: NilClass) + assert_call('"a".instance_eval{Thread.new{self.', include: String) + # Proc#call `(?) -> untyped` is RBS::Types::UntypedFunction + assert_call('proc{}.call; 1.', include: Integer) + end end end