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.
Files changed (37) hide show
  1. checksums.yaml +4 -4
  2. data/.rubocop.yml +12 -1
  3. data/CHANGELOG.md +34 -0
  4. data/README.md +44 -24
  5. data/Rakefile +7 -0
  6. data/exe/rbs-trace +8 -0
  7. data/lib/rbs/trace/builder.rb +136 -0
  8. data/lib/rbs/trace/cli/inline.rb +62 -0
  9. data/lib/rbs/trace/cli/merge.rb +102 -0
  10. data/lib/rbs/trace/cli.rb +40 -0
  11. data/lib/rbs/trace/file.rb +66 -21
  12. data/lib/rbs/trace/helpers.rb +67 -0
  13. data/lib/rbs/trace/inline_comment_visitor.rb +63 -0
  14. data/lib/rbs/trace/overload_compact.rb +93 -0
  15. data/lib/rbs/trace/return_value_visitor.rb +66 -0
  16. data/lib/rbs/trace/version.rb +2 -2
  17. data/lib/rbs/trace.rb +156 -4
  18. data/rbs_collection.lock.yaml +0 -4
  19. data/sig/generated/rbs/trace/builder.rbs +40 -0
  20. data/sig/generated/rbs/trace/cli/inline.rbs +19 -0
  21. data/sig/generated/rbs/trace/cli/merge.rbs +24 -0
  22. data/sig/generated/rbs/trace/cli.rbs +17 -0
  23. data/sig/generated/rbs/trace/file.rbs +20 -9
  24. data/sig/generated/rbs/trace/helpers.rbs +28 -0
  25. data/sig/generated/rbs/trace/inline_comment_visitor.rbs +27 -0
  26. data/sig/generated/rbs/trace/overload_compact.rbs +24 -0
  27. data/sig/generated/rbs/trace/return_value_visitor.rbs +27 -0
  28. data/sig/generated/rbs/trace/version.rbs +1 -1
  29. data/sig/generated/rbs/trace.rbs +63 -1
  30. data/tmp/.keep +0 -0
  31. metadata +38 -11
  32. data/lib/rbs/trace/declaration.rb +0 -107
  33. data/lib/rbs/trace/definition.rb +0 -33
  34. data/lib/rbs/trace/method_tracing.rb +0 -182
  35. data/sig/generated/rbs/trace/declaration.rbs +0 -36
  36. data/sig/generated/rbs/trace/definition.rbs +0 -25
  37. 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: e587e127bc2689a2d6e3c9f25dad32a5b75b49164b8666d0c0ff891d9c6e35f4
4
- data.tar.gz: 47cb65ce7cef30a14cc68984e7a76614ca593a1581a439b34341386937c65591
3
+ metadata.gz: 4040f1116ab68a5b072e7a847480080c778b1ff87c37a0a47d1ee3b923f61ffb
4
+ data.tar.gz: b262f219610e463cdccfa5d7e3442526963f118d2ea1386ff442919920b9053f
5
5
  SHA512:
6
- metadata.gz: 3ccd45b222189e14609f8ef30f207341dbf6a4ed863a4c94b5cf051d28504f84cd8d101a68905de787ac5661f9bc7b9432162b81abbda85868e40e14cc741ebd
7
- data.tar.gz: 158f74515cad425793edcc6f7d68c2ee8140afb36d9922ee80b6d1c21c020db748c9842c9b3bc6c15d330c2a12834681c6082b3e373c9ae669ccc379337ed626
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: 15
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
  [![Gem Version](https://badge.fury.io/rb/rbs-trace.svg)](https://badge.fury.io/rb/rbs-trace)
2
2
  [![Test](https://github.com/sinsoku/rbs-trace/actions/workflows/test.yml/badge.svg)](https://github.com/sinsoku/rbs-trace/actions/workflows/test.yml)
3
+ [![codecov](https://codecov.io/gh/sinsoku/rbs-trace/graph/badge.svg?token=rEsPe8Quyu)](https://codecov.io/gh/sinsoku/rbs-trace)
3
4
 
4
5
  # RBS::Trace
5
6
 
6
- RBS::Trace collects argument types and return value types using TracePoint, and inserts inline RBS type declarations into files.
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 `insert_rbs` method.
40
+ Call target methods within the `enable` method block, and call the `save_comments` method.
40
41
 
41
42
  ```ruby
42
- tracing = RBS::Trace::MethodTracing.new
43
+ trace = RBS::Trace.new
43
44
 
44
45
  # Collects the types of methods called in the block.
45
- tracing.enable do
46
+ trace.enable do
46
47
  user = User.new("Nanoha", "Takamachi")
47
48
  user.say_hello
48
49
  end
49
50
 
50
- tracing.insert_rbs
51
+ # Save RBS type declarations as embedded comments
52
+ trace.save_comments
51
53
  ```
52
54
 
53
- Automatically inserts inline RBS definitions into the file.
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
- tracing = RBS::Trace::MethodTracing.new
85
+ trace = RBS::Trace.new
86
86
 
87
- config.before(:suite) { tracing.enable }
87
+ config.before(:suite) { trace.enable }
88
88
  config.after(:suite) do
89
- tracing.disable
90
- tracing.insert_rbs
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
- tracing = RBS::Trace::MethodTracing.new
107
- tracing.enable
100
+ trace = RBS::Trace.new
101
+ trace.enable
108
102
 
109
103
  Minitest.after_run do
110
- tracing.disable
111
- tracing.insert_rbs
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
- tracing.files.each do |path, file|
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 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).
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,8 @@
1
+ #!/usr/bin/env ruby
2
+ # frozen_string_literal: true
3
+
4
+ lib = File.expand_path("../lib", __dir__)
5
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
6
+
7
+ require "rbs-trace"
8
+ RBS::Trace::CLI.new.run
@@ -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