rbs-trace 0.3.2 → 0.4.1
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.
- checksums.yaml +4 -4
- data/.rubocop.yml +12 -1
- data/CHANGELOG.md +34 -0
- data/README.md +44 -24
- data/Rakefile +7 -0
- data/exe/rbs-trace +8 -0
- data/lib/rbs/trace/builder.rb +136 -0
- data/lib/rbs/trace/cli/inline.rb +62 -0
- data/lib/rbs/trace/cli/merge.rb +102 -0
- data/lib/rbs/trace/cli.rb +40 -0
- data/lib/rbs/trace/file.rb +66 -21
- data/lib/rbs/trace/helpers.rb +67 -0
- data/lib/rbs/trace/inline_comment_visitor.rb +63 -0
- data/lib/rbs/trace/overload_compact.rb +93 -0
- data/lib/rbs/trace/return_value_visitor.rb +66 -0
- data/lib/rbs/trace/version.rb +2 -2
- data/lib/rbs/trace.rb +156 -4
- data/rbs_collection.lock.yaml +0 -4
- data/sig/generated/rbs/trace/builder.rbs +40 -0
- data/sig/generated/rbs/trace/cli/inline.rbs +19 -0
- data/sig/generated/rbs/trace/cli/merge.rbs +24 -0
- data/sig/generated/rbs/trace/cli.rbs +17 -0
- data/sig/generated/rbs/trace/file.rbs +20 -9
- data/sig/generated/rbs/trace/helpers.rbs +28 -0
- data/sig/generated/rbs/trace/inline_comment_visitor.rbs +27 -0
- data/sig/generated/rbs/trace/overload_compact.rbs +24 -0
- data/sig/generated/rbs/trace/return_value_visitor.rbs +27 -0
- data/sig/generated/rbs/trace/version.rbs +1 -1
- data/sig/generated/rbs/trace.rbs +63 -1
- data/tmp/.keep +0 -0
- metadata +38 -11
- data/lib/rbs/trace/declaration.rb +0 -107
- data/lib/rbs/trace/definition.rb +0 -33
- data/lib/rbs/trace/method_tracing.rb +0 -182
- data/sig/generated/rbs/trace/declaration.rbs +0 -36
- data/sig/generated/rbs/trace/definition.rbs +0 -25
- data/sig/generated/rbs/trace/method_tracing.rbs +0 -63
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: 4040f1116ab68a5b072e7a847480080c778b1ff87c37a0a47d1ee3b923f61ffb
         | 
| 4 | 
            +
              data.tar.gz: b262f219610e463cdccfa5d7e3442526963f118d2ea1386ff442919920b9053f
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: e3e1eea5dc319b4eafd62e62ffc1dc10ccc3301ffa603ddd8696f9b0a9a8f16a7604037873595fd91291211264e79622e8f1eba05a22dd02e78133466ae35e7d
         | 
| 7 | 
            +
              data.tar.gz: 30f08d4582741295df623ab8ddf1f87c97723eeea0a8217e39a6ae4bcd12dd6a1ee88623256d96e29a83a410687ec8e3fa89a3ce95a72973dc5ad1002846481d
         | 
    
        data/.rubocop.yml
    CHANGED
    
    | @@ -1,6 +1,7 @@ | |
| 1 1 | 
             
            require:
         | 
| 2 2 | 
             
              - rubocop-rake
         | 
| 3 3 | 
             
              - rubocop-rspec
         | 
| 4 | 
            +
              - rubocop-performance
         | 
| 4 5 |  | 
| 5 6 | 
             
            AllCops:
         | 
| 6 7 | 
             
              TargetRubyVersion: 3.1
         | 
| @@ -16,7 +17,7 @@ Style/Documentation: | |
| 16 17 | 
             
              Enabled: false
         | 
| 17 18 |  | 
| 18 19 | 
             
            RSpec/ExampleLength:
         | 
| 19 | 
            -
              Max:  | 
| 20 | 
            +
              Max: 50
         | 
| 20 21 |  | 
| 21 22 | 
             
            Naming/MethodParameterName:
         | 
| 22 23 | 
             
              Enabled: false
         | 
| @@ -25,8 +26,18 @@ Naming/FileName: | |
| 25 26 | 
             
              Exclude:
         | 
| 26 27 | 
             
                - "lib/rbs-trace.rb"
         | 
| 27 28 |  | 
| 29 | 
            +
            Naming/VariableNumber:
         | 
| 30 | 
            +
              Enabled: false
         | 
| 31 | 
            +
             | 
| 28 32 | 
             
            Layout/LeadingCommentSpace:
         | 
| 29 33 | 
             
              Enabled: false
         | 
| 30 34 |  | 
| 31 35 | 
             
            Style/AccessorGrouping:
         | 
| 32 36 | 
             
              Enabled: false
         | 
| 37 | 
            +
             | 
| 38 | 
            +
            RSpec/DescribeClass:
         | 
| 39 | 
            +
              Exclude:
         | 
| 40 | 
            +
                - "spec/smoke_spec.rb"
         | 
| 41 | 
            +
             | 
| 42 | 
            +
            RSpec/MultipleExpectations:
         | 
| 43 | 
            +
              Enabled: false
         | 
    
        data/CHANGELOG.md
    CHANGED
    
    | @@ -1,5 +1,39 @@ | |
| 1 1 | 
             
            ## [Unreleased]
         | 
| 2 2 |  | 
| 3 | 
            +
            ## [0.4.1] - 2025-02-24
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            - fix: Ensure class names are referenced
         | 
| 6 | 
            +
            - fix: Prefer other types over void and untyped
         | 
| 7 | 
            +
            - fix: Fix process to guess return value as void
         | 
| 8 | 
            +
            - fix: Do not parse if the caller is not Ruby
         | 
| 9 | 
            +
            - chore: Auto correct with rubocop-performance
         | 
| 10 | 
            +
             | 
| 11 | 
            +
            ## [0.4.0] - 2025-02-23
         | 
| 12 | 
            +
             | 
| 13 | 
            +
            - feat: Implement the function to save as RBS files
         | 
| 14 | 
            +
            - feat: Add CLI commands to support parallel testing
         | 
| 15 | 
            +
            - fix: Ignore if path is "inline template"
         | 
| 16 | 
            +
             | 
| 17 | 
            +
            ### BREAKING CHANGE
         | 
| 18 | 
            +
             | 
| 19 | 
            +
            - chore!: Re-design API
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            ```ruby
         | 
| 22 | 
            +
            # before
         | 
| 23 | 
            +
            tracing = RBS::Trace::MethodTracing.new
         | 
| 24 | 
            +
            tracing.enable do
         | 
| 25 | 
            +
              # something
         | 
| 26 | 
            +
            end
         | 
| 27 | 
            +
            tracing.insert_rbs
         | 
| 28 | 
            +
             | 
| 29 | 
            +
            # after
         | 
| 30 | 
            +
            trace = RBS::Trace.new
         | 
| 31 | 
            +
            trace.enable do
         | 
| 32 | 
            +
              # something
         | 
| 33 | 
            +
            end
         | 
| 34 | 
            +
            trace.save_comments
         | 
| 35 | 
            +
            ```
         | 
| 36 | 
            +
             | 
| 3 37 | 
             
            ## [0.3.2] - 2024-12-29
         | 
| 4 38 |  | 
| 5 39 | 
             
            - fix: Do not insert comments if comment `#:` exists
         | 
    
        data/README.md
    CHANGED
    
    | @@ -1,9 +1,10 @@ | |
| 1 1 | 
             
            [](https://badge.fury.io/rb/rbs-trace)
         | 
| 2 2 | 
             
            [](https://github.com/sinsoku/rbs-trace/actions/workflows/test.yml)
         | 
| 3 | 
            +
            [](https://codecov.io/gh/sinsoku/rbs-trace)
         | 
| 3 4 |  | 
| 4 5 | 
             
            # RBS::Trace
         | 
| 5 6 |  | 
| 6 | 
            -
            RBS::Trace collects argument  | 
| 7 | 
            +
            RBS::Trace automatically collects argument and return types and saves RBS type declarations as RBS files or comments.
         | 
| 7 8 |  | 
| 8 9 | 
             
            ## Installation
         | 
| 9 10 |  | 
| @@ -36,21 +37,22 @@ class User | |
| 36 37 | 
             
            end
         | 
| 37 38 | 
             
            ```
         | 
| 38 39 |  | 
| 39 | 
            -
            Call target methods within the `enable` method block, and call the ` | 
| 40 | 
            +
            Call target methods within the `enable` method block, and call the `save_comments` method.
         | 
| 40 41 |  | 
| 41 42 | 
             
            ```ruby
         | 
| 42 | 
            -
             | 
| 43 | 
            +
            trace = RBS::Trace.new
         | 
| 43 44 |  | 
| 44 45 | 
             
            # Collects the types of methods called in the block.
         | 
| 45 | 
            -
             | 
| 46 | 
            +
            trace.enable do
         | 
| 46 47 | 
             
              user = User.new("Nanoha", "Takamachi")
         | 
| 47 48 | 
             
              user.say_hello
         | 
| 48 49 | 
             
            end
         | 
| 49 50 |  | 
| 50 | 
            -
             | 
| 51 | 
            +
            # Save RBS type declarations as embedded comments
         | 
| 52 | 
            +
            trace.save_comments
         | 
| 51 53 | 
             
            ```
         | 
| 52 54 |  | 
| 53 | 
            -
            Automatically  | 
| 55 | 
            +
            Automatically insert comments into the file.
         | 
| 54 56 |  | 
| 55 57 | 
             
            ```ruby
         | 
| 56 58 | 
             
            class User
         | 
| @@ -79,36 +81,28 @@ end | |
| 79 81 | 
             
            Add the following code to `spec/support/rbs_trace.rb`.
         | 
| 80 82 |  | 
| 81 83 | 
             
            ```ruby
         | 
| 82 | 
            -
            return unless ENV["RBS_TRACE"]
         | 
| 83 | 
            -
             | 
| 84 84 | 
             
            RSpec.configure do |config|
         | 
| 85 | 
            -
               | 
| 85 | 
            +
              trace = RBS::Trace.new
         | 
| 86 86 |  | 
| 87 | 
            -
              config.before(:suite) {  | 
| 87 | 
            +
              config.before(:suite) { trace.enable }
         | 
| 88 88 | 
             
              config.after(:suite) do
         | 
| 89 | 
            -
                 | 
| 90 | 
            -
                 | 
| 89 | 
            +
                trace.disable
         | 
| 90 | 
            +
                trace.save_comments
         | 
| 91 91 | 
             
              end
         | 
| 92 92 | 
             
            end
         | 
| 93 93 | 
             
            ```
         | 
| 94 94 |  | 
| 95 | 
            -
            Then run RSpec with the environment variables.
         | 
| 96 | 
            -
             | 
| 97 | 
            -
            ```console
         | 
| 98 | 
            -
            $ RBS_TRACE=1 bundle exec rspec
         | 
| 99 | 
            -
            ```
         | 
| 100 | 
            -
             | 
| 101 95 | 
             
            ### Minitest
         | 
| 102 96 |  | 
| 103 97 | 
             
            Add the following code to `test_helper.rb`.
         | 
| 104 98 |  | 
| 105 99 | 
             
            ```ruby
         | 
| 106 | 
            -
             | 
| 107 | 
            -
             | 
| 100 | 
            +
            trace = RBS::Trace.new
         | 
| 101 | 
            +
            trace.enable
         | 
| 108 102 |  | 
| 109 103 | 
             
            Minitest.after_run do
         | 
| 110 | 
            -
               | 
| 111 | 
            -
               | 
| 104 | 
            +
              trace.disable
         | 
| 105 | 
            +
              trace.save_comments
         | 
| 112 106 | 
             
            end
         | 
| 113 107 | 
             
            ```
         | 
| 114 108 |  | 
| @@ -117,11 +111,37 @@ end | |
| 117 111 | 
             
            ### Insert RBS declarations for specific files only
         | 
| 118 112 |  | 
| 119 113 | 
             
            ```ruby
         | 
| 120 | 
            -
             | 
| 114 | 
            +
            trace.files.each do |path, file|
         | 
| 121 115 | 
             
              file.rewrite if path.include?("app/models/")
         | 
| 122 116 | 
             
            end
         | 
| 123 117 | 
             
            ```
         | 
| 124 118 |  | 
| 119 | 
            +
            ### Save RBS declarations as files
         | 
| 120 | 
            +
             | 
| 121 | 
            +
            ```ruby
         | 
| 122 | 
            +
            trace.save_files(out_dir: "sig/trace/")
         | 
| 123 | 
            +
            ```
         | 
| 124 | 
            +
             | 
| 125 | 
            +
            ### Parallel testing
         | 
| 126 | 
            +
             | 
| 127 | 
            +
            If you are using a parallel testing gem such as [parallel_tests](https://github.com/grosser/parallel_tests) or [flatware](https://github.com/briandunn/flatware), first save the type definitions in RBS files.
         | 
| 128 | 
            +
             | 
| 129 | 
            +
            ```ruby
         | 
| 130 | 
            +
            trace.save_files(out_dir: "tmp/sig-#{ENV["TEST_ENV_NUMBER"]}")
         | 
| 131 | 
            +
            ```
         | 
| 132 | 
            +
             | 
| 133 | 
            +
            Then use `rbs-trace merge` to merge multiple RBS files into one.
         | 
| 134 | 
            +
             | 
| 135 | 
            +
            ```bash
         | 
| 136 | 
            +
            $ rbs-trace merge --sig-dir='tmp/sig-*' > sig/merged.rbs
         | 
| 137 | 
            +
            ```
         | 
| 138 | 
            +
             | 
| 139 | 
            +
            Finally, insert comments using the merged RBS files.
         | 
| 140 | 
            +
             | 
| 141 | 
            +
            ```bash
         | 
| 142 | 
            +
            $ rbs-trace inline --rb-dir=app --rb-dir=lib
         | 
| 143 | 
            +
            ```
         | 
| 144 | 
            +
             | 
| 125 145 | 
             
            ### Enable debug logging
         | 
| 126 146 |  | 
| 127 147 | 
             
            If you want to enable debug logging, specify the environment variable `RBS_TRACE_DEBUG`.
         | 
| @@ -142,4 +162,4 @@ The gem is available as open source under the terms of the [MIT License](https:/ | |
| 142 162 |  | 
| 143 163 | 
             
            ## Code of Conduct
         | 
| 144 164 |  | 
| 145 | 
            -
            Everyone interacting in the  | 
| 165 | 
            +
            Everyone interacting in the RBS::Trace project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](https://github.com/sinsoku/rbs-trace/blob/main/CODE_OF_CONDUCT.md).
         | 
    
        data/Rakefile
    CHANGED
    
    | @@ -12,6 +12,13 @@ RuboCop::RakeTask.new | |
| 12 12 | 
             
            desc "Generate rbs files"
         | 
| 13 13 | 
             
            task :rbs_inline do
         | 
| 14 14 | 
             
              sh "rbs-inline --output --opt-out lib"
         | 
| 15 | 
            +
             | 
| 16 | 
            +
              # If the Ruby file is deleted, delete the RBS file
         | 
| 17 | 
            +
              Dir.glob("sig/generated/**/*.rbs").each do |path|
         | 
| 18 | 
            +
                rbs_path = Pathname(path)
         | 
| 19 | 
            +
                rb_path = rbs_path.sub(%r{^sig/generated}, "lib").sub_ext(".rb")
         | 
| 20 | 
            +
                rbs_path.delete unless File.exist?(rb_path)
         | 
| 21 | 
            +
              end
         | 
| 15 22 | 
             
            end
         | 
| 16 23 |  | 
| 17 24 | 
             
            desc "Run Steep"
         | 
    
        data/exe/rbs-trace
    ADDED
    
    
| @@ -0,0 +1,136 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module RBS
         | 
| 4 | 
            +
              class Trace
         | 
| 5 | 
            +
                class Builder # rubocop:disable Metrics/ClassLength
         | 
| 6 | 
            +
                  include Helpers
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  UNBOUND_CLASS_METHOD = Object.instance_method(:class)
         | 
| 9 | 
            +
                  UNBOUND_NAME_METHOD = Class.instance_method(:name)
         | 
| 10 | 
            +
                  private_constant :UNBOUND_CLASS_METHOD, :UNBOUND_NAME_METHOD
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  GENERICS_SIZE = {
         | 
| 13 | 
            +
                    Array => 1,
         | 
| 14 | 
            +
                    Range => 1,
         | 
| 15 | 
            +
                    Hash => 2
         | 
| 16 | 
            +
                  }.freeze
         | 
| 17 | 
            +
                  private_constant :GENERICS_SIZE
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                  # @rbs (bind: Binding, parameters: Array[__todo__], void: bool) -> Array[__todo__]
         | 
| 20 | 
            +
                  def method_call(bind:, parameters:, void:)
         | 
| 21 | 
            +
                    method_type = parse_parameters(bind, parameters)
         | 
| 22 | 
            +
                    return_type = type_void if void
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                    [method_type, return_type].tap do |types|
         | 
| 25 | 
            +
                      stack_traces << types
         | 
| 26 | 
            +
                    end
         | 
| 27 | 
            +
                  end
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                  # @rbs (__todo__) -> AST::Members::MethodDefinition::Overload
         | 
| 30 | 
            +
                  def method_return(return_value)
         | 
| 31 | 
            +
                    method_type, return_type = stack_traces.pop
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                    type = return_type || parse_object(return_value)
         | 
| 34 | 
            +
                    new_type = method_type.type.with_return_type(type)
         | 
| 35 | 
            +
                    method_type = method_type.update(type: new_type) # rubocop:disable Style/RedundantSelfAssignment
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                    AST::Members::MethodDefinition::Overload.new(method_type:, annotations: [])
         | 
| 38 | 
            +
                  end
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                  private
         | 
| 41 | 
            +
             | 
| 42 | 
            +
                  def stack_traces
         | 
| 43 | 
            +
                    @stack_traces ||= {} #: Hash[Thread, Array[__todo__]]
         | 
| 44 | 
            +
                    @stack_traces[Thread.current] ||= [] # steep:ignore UnannotatedEmptyCollection
         | 
| 45 | 
            +
                  end
         | 
| 46 | 
            +
             | 
| 47 | 
            +
                  # @rbs (Binding, Array[__todo__]) -> MethodType
         | 
| 48 | 
            +
                  def parse_parameters(bind, parameters) # rubocop:disable Metrics
         | 
| 49 | 
            +
                    fn = Types::Function.empty(type_void)
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                    parameters.each do |kind, name| # rubocop:disable Metrics/BlockLength
         | 
| 52 | 
            +
                      case kind
         | 
| 53 | 
            +
                      when :req
         | 
| 54 | 
            +
                        value = bind.local_variable_get(name)
         | 
| 55 | 
            +
                        fn.required_positionals << Types::Function::Param.new(name: nil, type: parse_object(value))
         | 
| 56 | 
            +
                      when :opt
         | 
| 57 | 
            +
                        value = bind.local_variable_get(name)
         | 
| 58 | 
            +
                        fn.optional_positionals << Types::Function::Param.new(name: nil, type: parse_object(value))
         | 
| 59 | 
            +
                      when :rest
         | 
| 60 | 
            +
                        type = if name.nil? || name == :*
         | 
| 61 | 
            +
                                 type_untyped
         | 
| 62 | 
            +
                               else
         | 
| 63 | 
            +
                                 value = bind.local_variable_get(name)
         | 
| 64 | 
            +
                                 parse_classes(value.map { |v| obj_to_class(v) }.uniq)
         | 
| 65 | 
            +
                               end
         | 
| 66 | 
            +
                        fn = fn.update(rest_positionals: Types::Function::Param.new(name: nil, type:)) # rubocop:disable Style/RedundantSelfAssignment
         | 
| 67 | 
            +
                      when :keyreq
         | 
| 68 | 
            +
                        value = bind.local_variable_get(name)
         | 
| 69 | 
            +
                        fn.required_keywords[name] = Types::Function::Param.new(name: nil, type: parse_object(value))
         | 
| 70 | 
            +
                      when :key
         | 
| 71 | 
            +
                        value = bind.local_variable_get(name)
         | 
| 72 | 
            +
                        fn.optional_keywords[name] = Types::Function::Param.new(name: nil, type: parse_object(value))
         | 
| 73 | 
            +
                      when :keyrest
         | 
| 74 | 
            +
                        type = if name.nil? || name == :**
         | 
| 75 | 
            +
                                 type_untyped
         | 
| 76 | 
            +
                               else
         | 
| 77 | 
            +
                                 value = bind.local_variable_get(name)
         | 
| 78 | 
            +
                                 parse_classes(value.values.map { |v| obj_to_class(v) }.uniq)
         | 
| 79 | 
            +
                               end
         | 
| 80 | 
            +
                        fn = fn.update(rest_keywords: Types::Function::Param.new(name: nil, type:)) # rubocop:disable Style/RedundantSelfAssignment
         | 
| 81 | 
            +
                      when :block
         | 
| 82 | 
            +
                        # TODO: support block argument
         | 
| 83 | 
            +
                        next
         | 
| 84 | 
            +
                      end
         | 
| 85 | 
            +
                    end
         | 
| 86 | 
            +
             | 
| 87 | 
            +
                    MethodType.new(
         | 
| 88 | 
            +
                      type_params: [],
         | 
| 89 | 
            +
                      type: fn,
         | 
| 90 | 
            +
                      block: nil,
         | 
| 91 | 
            +
                      location: nil
         | 
| 92 | 
            +
                    )
         | 
| 93 | 
            +
                  end
         | 
| 94 | 
            +
             | 
| 95 | 
            +
                  # @rbs (Array[untyped]) -> Types::t
         | 
| 96 | 
            +
                  def parse_classes(classes)
         | 
| 97 | 
            +
                    types = classes.filter_map { |klass| parse_class(klass) unless klass == NilClass }.uniq
         | 
| 98 | 
            +
                    return type_nil if types.empty?
         | 
| 99 | 
            +
             | 
| 100 | 
            +
                    type = types.one? ? types.first : Types::Union.new(types:, location: nil) #: Types::t
         | 
| 101 | 
            +
                    if classes.include?(NilClass)
         | 
| 102 | 
            +
                      Types::Optional.new(type:, location: nil)
         | 
| 103 | 
            +
                    else
         | 
| 104 | 
            +
                      type
         | 
| 105 | 
            +
                    end
         | 
| 106 | 
            +
                  end
         | 
| 107 | 
            +
             | 
| 108 | 
            +
                  # @rbs (untyped) -> Types::t
         | 
| 109 | 
            +
                  def parse_class(klass) # rubocop:disable Metrics/MethodLength
         | 
| 110 | 
            +
                    class_name = UNBOUND_NAME_METHOD.bind_call(klass)
         | 
| 111 | 
            +
                    if [TrueClass, FalseClass].include?(klass)
         | 
| 112 | 
            +
                      type_bool
         | 
| 113 | 
            +
                    elsif klass == NilClass
         | 
| 114 | 
            +
                      type_nil
         | 
| 115 | 
            +
                    elsif klass == Object || class_name.nil?
         | 
| 116 | 
            +
                      type_untyped
         | 
| 117 | 
            +
                    else
         | 
| 118 | 
            +
                      size = GENERICS_SIZE[klass].to_i
         | 
| 119 | 
            +
                      args = Array.new(size) { type_untyped }
         | 
| 120 | 
            +
                      Types::ClassInstance.new(name: TypeName.parse(class_name), args:, location: nil)
         | 
| 121 | 
            +
                    end
         | 
| 122 | 
            +
                  end
         | 
| 123 | 
            +
             | 
| 124 | 
            +
                  # @rbs (BasicObject) -> Types::t
         | 
| 125 | 
            +
                  def parse_object(object)
         | 
| 126 | 
            +
                    klass = obj_to_class(object)
         | 
| 127 | 
            +
                    parse_class(klass)
         | 
| 128 | 
            +
                  end
         | 
| 129 | 
            +
             | 
| 130 | 
            +
                  # @rbs (BasicObject) -> Class
         | 
| 131 | 
            +
                  def obj_to_class(obj)
         | 
| 132 | 
            +
                    UNBOUND_CLASS_METHOD.bind_call(obj)
         | 
| 133 | 
            +
                  end
         | 
| 134 | 
            +
                end
         | 
| 135 | 
            +
              end
         | 
| 136 | 
            +
            end
         | 
| @@ -0,0 +1,62 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module RBS
         | 
| 4 | 
            +
              class Trace
         | 
| 5 | 
            +
                class CLI
         | 
| 6 | 
            +
                  class Inline
         | 
| 7 | 
            +
                    BANNER = <<~USAGE
         | 
| 8 | 
            +
                      Usage: rbs-trace inline --sig-dir=DIR --rb-dir=DIR
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                      Insert RBS inline comments from RBS files.
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                      Examples:
         | 
| 13 | 
            +
                        # Insert inline comments to `app/**/*.rb`.
         | 
| 14 | 
            +
                        $ rbs-trace inline --sig-dir=sig --rb-dir=app
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                        # Generate RBS files with rbs-inline.
         | 
| 17 | 
            +
                        $ rbs-inline --output --opt-out app
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                        # Remove method definitions that have been migrated to inline comments.
         | 
| 20 | 
            +
                        $ rbs subtract --write sig sig/generated/
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                      Options:
         | 
| 23 | 
            +
                    USAGE
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                    # @rbs (Array[String]) -> void
         | 
| 26 | 
            +
                    def run(args) # rubocop:disable Metrics
         | 
| 27 | 
            +
                      sig_dir = Pathname.pwd.join("sig").to_s
         | 
| 28 | 
            +
                      rb_dirs = [] #: Array[String]
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                      opts = OptionParser.new(BANNER)
         | 
| 31 | 
            +
                      opts.on("--sig-dir DIR") { |dir| sig_dir = dir }
         | 
| 32 | 
            +
                      opts.on("--rb-dir DIR") { |dir| rb_dirs << dir }
         | 
| 33 | 
            +
                      opts.parse!(args)
         | 
| 34 | 
            +
             | 
| 35 | 
            +
                      if rb_dirs.empty?
         | 
| 36 | 
            +
                        puts opts.help
         | 
| 37 | 
            +
                      else
         | 
| 38 | 
            +
                        env = load_env(sig_dir) # steep:ignore ArgumentTypeMismatch
         | 
| 39 | 
            +
                        decls = env.class_decls.transform_values { |v| v.primary.decl }
         | 
| 40 | 
            +
             | 
| 41 | 
            +
                        rb_files = rb_dirs.flat_map { |rb_dir| Dir.glob("#{rb_dir}/**/*.rb") }
         | 
| 42 | 
            +
                        rb_files.each do |path|
         | 
| 43 | 
            +
                          file = File.new(path, decls)
         | 
| 44 | 
            +
                          file.rewrite
         | 
| 45 | 
            +
                        end
         | 
| 46 | 
            +
                      end
         | 
| 47 | 
            +
                    end
         | 
| 48 | 
            +
             | 
| 49 | 
            +
                    private
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                    # @rbs (String) -> Environment
         | 
| 52 | 
            +
                    def load_env(dir)
         | 
| 53 | 
            +
                      Environment.new.tap do |env|
         | 
| 54 | 
            +
                        loader = EnvironmentLoader.new(core_root: nil)
         | 
| 55 | 
            +
                        loader.add(path: Pathname(dir))
         | 
| 56 | 
            +
                        loader.load(env:)
         | 
| 57 | 
            +
                      end
         | 
| 58 | 
            +
                    end
         | 
| 59 | 
            +
                  end
         | 
| 60 | 
            +
                end
         | 
| 61 | 
            +
              end
         | 
| 62 | 
            +
            end
         | 
| @@ -0,0 +1,102 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module RBS
         | 
| 4 | 
            +
              class Trace
         | 
| 5 | 
            +
                class CLI
         | 
| 6 | 
            +
                  class Merge
         | 
| 7 | 
            +
                    BANNER = <<~USAGE
         | 
| 8 | 
            +
                      Usage: rbs-trace merge --sig-dir=DIR
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                      Merge multiple RBS files into one.
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                      Examples:
         | 
| 13 | 
            +
                        # Merge RBS files in `tmp/sig-1/` and `tmp/sig-2/`.
         | 
| 14 | 
            +
                        $ rbs-trace merge --sig-dir=tmp/sig-1 --sig-dir=tmp/sig-2
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                        # Or you can specify a glob pattern.
         | 
| 17 | 
            +
                        $ rbs-trace merge --sig-dir=tmp/sig-*
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                      Options:
         | 
| 20 | 
            +
                    USAGE
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                    # @rbs (Array[String]) -> void
         | 
| 23 | 
            +
                    def run(args) # rubocop:disable Metrics
         | 
| 24 | 
            +
                      sig_dirs = [] #: Array[String]
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                      opts = OptionParser.new(BANNER)
         | 
| 27 | 
            +
                      opts.on("--sig-dir DIR") { |dir| sig_dirs << dir }
         | 
| 28 | 
            +
                      opts.parse!(args)
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                      if sig_dirs.empty?
         | 
| 31 | 
            +
                        puts opts.help
         | 
| 32 | 
            +
                      else
         | 
| 33 | 
            +
                        envs = sig_dirs.flat_map { |sig_dir| Dir.glob(sig_dir) }
         | 
| 34 | 
            +
                                       .map { |dir| load_env(dir) }
         | 
| 35 | 
            +
                        env = merge_envs(envs)
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                        out = StringIO.new
         | 
| 38 | 
            +
                        writer = Writer.new(out:)
         | 
| 39 | 
            +
                        writer.write(env.declarations)
         | 
| 40 | 
            +
             | 
| 41 | 
            +
                        puts out.string
         | 
| 42 | 
            +
                      end
         | 
| 43 | 
            +
                    end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                    # @rbs (Array[Environment]) -> Environment
         | 
| 46 | 
            +
                    def merge_envs(others) # rubocop:disable Metrics
         | 
| 47 | 
            +
                      Environment.new.tap do |env|
         | 
| 48 | 
            +
                        others.each do |other|
         | 
| 49 | 
            +
                          other.declarations.each do |decl|
         | 
| 50 | 
            +
                            next unless decl.is_a?(AST::Declarations::Class) || decl.is_a?(AST::Declarations::Module)
         | 
| 51 | 
            +
             | 
| 52 | 
            +
                            entry = env.module_class_entry(decl.name.absolute!)
         | 
| 53 | 
            +
             | 
| 54 | 
            +
                            if entry.is_a?(Environment::MultiEntry)
         | 
| 55 | 
            +
                              decl.members.each { |member| merge(entry.primary.decl, member) }
         | 
| 56 | 
            +
                            else
         | 
| 57 | 
            +
                              env << decl
         | 
| 58 | 
            +
                            end
         | 
| 59 | 
            +
                          end
         | 
| 60 | 
            +
                        end
         | 
| 61 | 
            +
                      end
         | 
| 62 | 
            +
                    end
         | 
| 63 | 
            +
             | 
| 64 | 
            +
                    private
         | 
| 65 | 
            +
             | 
| 66 | 
            +
                    # @rbs (String) -> Environment
         | 
| 67 | 
            +
                    def load_env(dir)
         | 
| 68 | 
            +
                      Environment.new.tap do |env|
         | 
| 69 | 
            +
                        loader = EnvironmentLoader.new(core_root: nil)
         | 
| 70 | 
            +
                        loader.add(path: Pathname(dir))
         | 
| 71 | 
            +
                        loader.load(env:)
         | 
| 72 | 
            +
                      end
         | 
| 73 | 
            +
                    end
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                    def merge(decl, member) # rubocop:disable Metrics
         | 
| 76 | 
            +
                      case member
         | 
| 77 | 
            +
                      when AST::Declarations::Class, AST::Declarations::Module
         | 
| 78 | 
            +
                        d = decl.members.find { |m| m.is_a?(member.class) && m.name == member.name }
         | 
| 79 | 
            +
             | 
| 80 | 
            +
                        if d
         | 
| 81 | 
            +
                          member.members.each { |m| merge(d, m) }
         | 
| 82 | 
            +
                        else
         | 
| 83 | 
            +
                          decl.members << member
         | 
| 84 | 
            +
                        end
         | 
| 85 | 
            +
                      when AST::Members::MethodDefinition
         | 
| 86 | 
            +
                        found = decl.members.find { |m| m.is_a?(member.class) && m.name == member.name && m.kind == member.kind }
         | 
| 87 | 
            +
             | 
| 88 | 
            +
                        if found
         | 
| 89 | 
            +
                          (member.overloads - found.overloads).each do |overload|
         | 
| 90 | 
            +
                            found.overloads << overload
         | 
| 91 | 
            +
                          end
         | 
| 92 | 
            +
                        else
         | 
| 93 | 
            +
                          decl.members << member
         | 
| 94 | 
            +
                        end
         | 
| 95 | 
            +
                      else
         | 
| 96 | 
            +
                        decl.members << member unless decl.members.include?(member)
         | 
| 97 | 
            +
                      end
         | 
| 98 | 
            +
                    end
         | 
| 99 | 
            +
                  end
         | 
| 100 | 
            +
                end
         | 
| 101 | 
            +
              end
         | 
| 102 | 
            +
            end
         | 
| @@ -0,0 +1,40 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module RBS
         | 
| 4 | 
            +
              class Trace
         | 
| 5 | 
            +
                class CLI
         | 
| 6 | 
            +
                  BANNER = <<~USAGE
         | 
| 7 | 
            +
                    Usage: rbs-trace <command> [<args>]
         | 
| 8 | 
            +
             | 
| 9 | 
            +
                    Available commands: inline, merge
         | 
| 10 | 
            +
                  USAGE
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  # @rbs (Array[String]) -> void
         | 
| 13 | 
            +
                  def run(args = ARGV)
         | 
| 14 | 
            +
                    opts = OptionParser.new(BANNER)
         | 
| 15 | 
            +
                    opts.version = RBS::Trace::VERSION
         | 
| 16 | 
            +
                    opts.order!(args)
         | 
| 17 | 
            +
                    command = args.shift&.to_sym
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                    klass = command_class(command)
         | 
| 20 | 
            +
                    if klass
         | 
| 21 | 
            +
                      klass.new.run(args)
         | 
| 22 | 
            +
                    else
         | 
| 23 | 
            +
                      puts opts.help
         | 
| 24 | 
            +
                    end
         | 
| 25 | 
            +
                  end
         | 
| 26 | 
            +
             | 
| 27 | 
            +
                  private
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                  # @rbs (Symbol?) -> (singleton(Inline) | singleton(Merge))?
         | 
| 30 | 
            +
                  def command_class(command)
         | 
| 31 | 
            +
                    case command
         | 
| 32 | 
            +
                    when :inline
         | 
| 33 | 
            +
                      Inline
         | 
| 34 | 
            +
                    when :merge
         | 
| 35 | 
            +
                      Merge
         | 
| 36 | 
            +
                    end
         | 
| 37 | 
            +
                  end
         | 
| 38 | 
            +
                end
         | 
| 39 | 
            +
              end
         | 
| 40 | 
            +
            end
         |