gelauto 2.0.0 → 2.2.0

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a9a90485262d6867a89f54f088f6cd71cd7490b41d22622edd4376387d97555a
4
- data.tar.gz: 808a3f83fd11126ccf4d41ff65e770e28ba9e4d4912bfd9ddca50655471369b9
3
+ metadata.gz: 195e070c508f3696ca8107bf020e99571d25b9ce3b6d963f650423b4ffcab265
4
+ data.tar.gz: 8619e1c12452751ee78d247296afb0c9d7c9b3f4328fcdfba9ea913d5cb492d1
5
5
  SHA512:
6
- metadata.gz: 861d47eebcf4b9709d3cdafa5119a39a30bcecdb0a5fe201accf46464861676a23392322e962afbdd356332537a5fca1bf50e8578baafa7e728564b5d2f9ce6c
7
- data.tar.gz: 19011ece03802f5bd64991382d5cf0b7eae12cc8e431c13ff72e4d807d16a29e949e2a1df90b8505ac1c9f145f731d45af5265686fe3cf27f66699431ce803f8
6
+ metadata.gz: a50267721b8e10e49fd34d3c5f8a35ece67d0106599a927d8d8c9c53ceddcaf81be51bd41b58bf2df00d90fdf75311e37371fd6071d28100fbf4174fca34e0a1
7
+ data.tar.gz: 6f4d3179cb04eca9631315b5017061d99e726ae2721096a1de1ca5b544ce81c4955145e24f48fb5158669fe4a3f85acabb469de8ba86e831191ae58f0892b201
data/Gemfile CHANGED
@@ -2,6 +2,8 @@ source 'https://rubygems.org'
2
2
 
3
3
  gemspec
4
4
 
5
+ gem 'sorbet-runtime'
6
+
5
7
  group :development do
6
8
  gem 'pry-byebug'
7
9
  gem 'rspec'
data/README.md CHANGED
@@ -1,4 +1,4 @@
1
- ## gelauto [![Build Status](https://travis-ci.com/camertron/gelauto.svg?branch=master)](https://travis-ci.com/camertron/gelauto)
1
+ ## gelauto [![Build status](https://github.com/camertron/gelauto/actions/workflows/test.yml/badge.svg)](https://github.com/camertron/gelauto/actions/workflows/test.yml)
2
2
 
3
3
  Automatically annotate your code with Sorbet type definitions.
4
4
 
@@ -45,13 +45,13 @@ First, install the gem by running `gem install gelauto`. That will make the `gel
45
45
  Gelauto's only subcommand is `run`, which accepts a list of Ruby files to scan for methods and a command to run that will exercise your code.
46
46
 
47
47
  In this example, we're going to be running an [RSpec](https://github.com/rspec/rspec) test suite.
48
- Like most RSpec test suites, let's assume ours is stored in the `spec/` directory (that's the RSpec default too). Let's furthermore assume our code is stored in the `lib` directory. To run the test suite in `spec/` and add type definitions to our files in `lib/`, we might run the following command:
48
+ Like most RSpec test suites, let's assume ours is stored in the `spec/` directory (that's the RSpec default too). To run the test suite in `spec/` and add type definitions to our ruby files, we might run the following command:
49
49
 
50
50
  ```bash
51
51
  gelauto run --annotate $(find . -name '*.rb') -- bundle exec rspec spec/
52
52
  ```
53
53
 
54
- You can also choose to run Gelauto with the `--rbi` flag, which will cause Gelauto to print results to standard output or to a file in [RBI format](https://sorbet.org/docs/rbi):
54
+ You can also choose to run Gelauto with the `--rbi` flag, which will cause Gelauto to print results to standard output or to the given file in [RBI format](https://sorbet.org/docs/rbi):
55
55
 
56
56
  ```bash
57
57
  # print RBI output to STDOUT
@@ -126,17 +126,27 @@ require 'gelauto/rspec'
126
126
  to your spec_helper.rb, Rakefile, or wherever RSpec is configured. You'll also need to set the `GELAUTO_FILES` environment variable when running your test suite. For example:
127
127
 
128
128
  ```bash
129
- GELAUTO_FILES=$(find ./lib -name *.rb) bundle exec rspec
129
+ GELAUTO_FILES=$(find . -name *.rb) bundle exec rspec
130
130
  ```
131
131
 
132
132
  Files can be separated by spaces, newlines, or commas. If you want Gelauto to annotate them, set `GELAUTO_ANNOTATE` to `true`, eg:
133
133
 
134
134
  ```bash
135
- GELAUTO_FILES=$(find ./lib -name *.rb) GELAUTO_ANNOTATE=true bundle exec rspec
135
+ GELAUTO_FILES=$(find . -name *.rb) GELAUTO_ANNOTATE=true bundle exec rspec
136
136
  ```
137
137
 
138
138
  Finally, set `GELAUTO_RBI=/path/to/output.rbi` to have Gelauto emit an RBI file when the test suite finishes.
139
139
 
140
+ #### Minitest Helper
141
+
142
+ Gelauto also comes with a handy Minitest helper. Simply add
143
+
144
+ ```ruby
145
+ require 'gelauto/minitest'
146
+ ```
147
+
148
+ to your test_helper.rb (or wherever Minitest is `require`d and configured). The same environment variables described above for RSpec, eg. `GELAUTO_FILES`, `GELAUTO_ANNOTATE`, and `GELAUTO_RBI`, can be used in a Minitest setup.
149
+
140
150
  ## How does it Work?
141
151
 
142
152
  Gelauto makes use of Ruby's [TracePoint API](https://ruby-doc.org/core-2.6/TracePoint.html). TracePoint effectively allows Gelauto to receive a notification whenever a Ruby method is called and whenever a method returns. That info combined with method location information gathered from parsing your Ruby files ahead of time allows Gelauto to know a) where methods are located, 2) what arguments they take, 3) the types of those arguments, and 4) the type of the return value.
@@ -147,7 +157,6 @@ Gelauto makes use of Ruby's [TracePoint API](https://ruby-doc.org/core-2.6/Trace
147
157
 
148
158
  * Half-baked support for singleton (i.e. static) methods.
149
159
  * Gelauto does not annotate Ruby files with `# typed: true` comments or `extend T::Sig`.
150
- * Gelauto ignores existing type signatures and will simply add another one right above the method.
151
160
 
152
161
  ## Running Tests
153
162
 
data/gelauto.gemspec CHANGED
@@ -11,7 +11,7 @@ Gem::Specification.new do |s|
11
11
  s.description = s.summary = 'Automatically annotate your code with Sorbet type definitions.'
12
12
  s.platform = Gem::Platform::RUBY
13
13
 
14
- s.add_dependency 'parser', '~> 2.6'
14
+ s.add_dependency 'parser', '~> 3.0'
15
15
  s.add_dependency 'gli', '~> 2.0'
16
16
 
17
17
  s.executables << 'gelauto'
@@ -1,9 +1,10 @@
1
1
  module Gelauto
2
2
  class MethodIndex
3
- attr_reader :index
3
+ attr_reader :index, :sig_index
4
4
 
5
5
  def initialize
6
6
  @index = Hash.new { |h, k| h[k] = {} }
7
+ @sig_index = Hash.new { |h, k| h[k] = {} }
7
8
  end
8
9
 
9
10
  def index_methods_in(path, ast, nesting = [])
@@ -16,10 +17,16 @@ module Gelauto
16
17
 
17
18
  if ast.type == :def
18
19
  name = ast.children[0]
19
- args = ast.children[1].children.map { |c| c.children.first }
20
+ args = ast.children[1].children.each_with_object([]) do |c, memo|
21
+ # ignore anonymous args like **, etc
22
+ memo << c.children.first if c.children.first
23
+ end
20
24
  else
21
25
  name = ast.children[1]
22
- args = ast.children[2].children.map { |c| c.children.first }
26
+ args = ast.children[2].children.each_with_object([]) do |c, memo|
27
+ # ignore anonymous args like **, etc
28
+ memo << c.children.first if c.children.first
29
+ end
23
30
  end
24
31
 
25
32
  md = MethodDef.new(name, args, nesting)
@@ -31,6 +38,10 @@ module Gelauto
31
38
  when :class, :module
32
39
  const_name = ast.children.first.children.last
33
40
  return visit_children(path, ast, nesting + [Namespace.new(const_name, ast.type)])
41
+ when :block
42
+ if ast.children.first.children.last == :sig
43
+ sig_index[path][ast.location.line] = true
44
+ end
34
45
  end
35
46
 
36
47
  visit_children(path, ast, nesting)
@@ -66,13 +77,14 @@ module Gelauto
66
77
  def annotate(path, code)
67
78
  lines = code.split(/\r?\n/)
68
79
  mds = index[path]
80
+ sigs = sig_index[path]
69
81
 
70
82
  [].tap do |annotated|
71
83
  lines.each_with_index do |line, idx|
72
84
  lineno = idx + 1
73
85
  md = mds[lineno]
74
86
 
75
- if md.is_a?(MethodDef)
87
+ if md.is_a?(MethodDef) && !sigs[idx]
76
88
  indent = line[0...line.index(/[^\s]/)]
77
89
  annotated << "#{indent}#{md.to_sig}"
78
90
  end
@@ -0,0 +1,25 @@
1
+ require 'gelauto'
2
+
3
+ Gelauto.paths += ENV.fetch('GELAUTO_FILES', '').split(/[\s\n,]/).map(&:strip)
4
+ Gelauto.setup
5
+
6
+ Minitest.after_run do
7
+ Gelauto.teardown
8
+
9
+ if ENV['GELAUTO_ANNOTATE'] == 'true'
10
+ Gelauto.each_absolute_path do |path|
11
+ Gelauto.annotate_file(path)
12
+ Gelauto::Logger.info("Annotated #{path}")
13
+ end
14
+ end
15
+
16
+ if ENV['GELAUTO_RBI']
17
+ rbi_str = Gelauto::Rbi.new(Gelauto.method_index).to_s
18
+
19
+ if ENV['GELAUTO_RBI'] == '-'
20
+ puts rbi_str
21
+ else
22
+ File.write(ENV['GELAUTO_RBI'], rbi_str)
23
+ end
24
+ end
25
+ end
data/lib/gelauto/rbi.rb CHANGED
@@ -31,7 +31,7 @@ module Gelauto
31
31
  io.write(indent(h[:methods].map { |md| md.to_rbi }.join("\n\n"), indent_level))
32
32
 
33
33
  h[:children].each_with_index do |((namespace, type), next_level), idx|
34
- io.write("\n\n") if idx > 0
34
+ io.write("\n\n") if idx > 0 || h[:methods].size > 0
35
35
  io.write(indent("#{type} #{namespace}", indent_level))
36
36
  io.write("\n")
37
37
  compose(next_level, io, indent_level + 1)
@@ -43,7 +43,9 @@ module Gelauto
43
43
  if rt == ::NilClass
44
44
  nilable = true
45
45
  else
46
- ret << t.to_sig
46
+ t.to_sig.tap do |s|
47
+ ret << s if s
48
+ end
47
49
  end
48
50
  end
49
51
 
@@ -1,3 +1,3 @@
1
1
  module Gelauto
2
- VERSION = '2.0.0'
2
+ VERSION = '2.2.0'
3
3
  end
data/lib/gelauto.rb CHANGED
@@ -92,7 +92,7 @@ module Gelauto
92
92
  )
93
93
 
94
94
  Gelauto::Logger.info("Indexed #{idx + 1}/#{paths.size} paths")
95
- rescue Parser::SyntaxError => e
95
+ rescue Parser::SyntaxError
96
96
  Gelauto::Logger.error("Syntax error in #{path}, skipping")
97
97
  end
98
98
  end
data/spec/gelauto_spec.rb CHANGED
@@ -80,6 +80,27 @@ describe Gelauto do
80
80
  end
81
81
  end
82
82
 
83
+ context 'with existing signatures' do
84
+ before do
85
+ Gelauto.discover do
86
+ @request = GelautoSpecs::Request.new
87
+ @request.to_a('Hello', 'World')
88
+ @request.to_s(100)
89
+ end
90
+ end
91
+
92
+ it 'skips existing sig, adds missing sig' do
93
+ file = File.read('spec/support/annotated.rb')
94
+ expect(file.lines.count).to eq(17)
95
+ # does not add a signature to the method with an existing signature
96
+ annotated = annotate(@request, :to_a, file)
97
+ expect(annotated.lines.count).to eq(17)
98
+ # does add a signature to the method without a signature
99
+ annotated = annotate(@request, :to_s, file)
100
+ expect(annotated.lines.count).to eq(18)
101
+ end
102
+ end
103
+
83
104
  context 'with nested generic types' do
84
105
  before do
85
106
  Gelauto.discover do
data/spec/spec_helper.rb CHANGED
@@ -20,6 +20,11 @@ RSpec.configure do |config|
20
20
  path, lineno = obj.method(method_name).source_location
21
21
  Gelauto.method_index.find(path, lineno)
22
22
  end
23
+
24
+ def annotate(obj, method_name, code)
25
+ path, _lineno = obj.method(method_name).source_location
26
+ Gelauto.method_index.annotate(path, code)
27
+ end
23
28
  end
24
29
  )
25
30
  end
@@ -0,0 +1,17 @@
1
+ require 'sorbet-runtime'
2
+
3
+ module GelautoSpecs
4
+ class Request
5
+ extend T::Sig
6
+ sig { params(status: String, body: String).returns(T::Array[String]) }
7
+ def to_a(status, body)
8
+ [status, body]
9
+ end
10
+
11
+ def to_s(number)
12
+ number&.to_s
13
+ end
14
+ end
15
+ end
16
+
17
+ Gelauto.paths << __FILE__
@@ -1,3 +1,5 @@
1
+ require 'sorbet-runtime'
2
+
1
3
  module GelautoSpecs
2
4
  class Response
3
5
  attr_reader :status, :body
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: gelauto
3
3
  version: !ruby/object:Gem::Version
4
- version: 2.0.0
4
+ version: 2.2.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Cameron Dutro
8
- autorequire:
8
+ autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2020-01-25 00:00:00.000000000 Z
11
+ date: 2025-04-03 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: parser
@@ -16,14 +16,14 @@ dependencies:
16
16
  requirements:
17
17
  - - "~>"
18
18
  - !ruby/object:Gem::Version
19
- version: '2.6'
19
+ version: '3.0'
20
20
  type: :runtime
21
21
  prerelease: false
22
22
  version_requirements: !ruby/object:Gem::Requirement
23
23
  requirements:
24
24
  - - "~>"
25
25
  - !ruby/object:Gem::Version
26
- version: '2.6'
26
+ version: '3.0'
27
27
  - !ruby/object:Gem::Dependency
28
28
  name: gli
29
29
  requirement: !ruby/object:Gem::Requirement
@@ -61,6 +61,7 @@ files:
61
61
  - lib/gelauto/logger.rb
62
62
  - lib/gelauto/method_def.rb
63
63
  - lib/gelauto/method_index.rb
64
+ - lib/gelauto/minitest.rb
64
65
  - lib/gelauto/namespace.rb
65
66
  - lib/gelauto/null_logger.rb
66
67
  - lib/gelauto/rbi.rb
@@ -72,6 +73,7 @@ files:
72
73
  - lib/gelauto/version.rb
73
74
  - spec/gelauto_spec.rb
74
75
  - spec/spec_helper.rb
76
+ - spec/support/annotated.rb
75
77
  - spec/support/client.rb
76
78
  - spec/support/config.yml
77
79
  - spec/support/image.rb
@@ -80,7 +82,7 @@ files:
80
82
  homepage: http://github.com/camertron/gelauto
81
83
  licenses: []
82
84
  metadata: {}
83
- post_install_message:
85
+ post_install_message:
84
86
  rdoc_options: []
85
87
  require_paths:
86
88
  - lib
@@ -95,9 +97,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
95
97
  - !ruby/object:Gem::Version
96
98
  version: '0'
97
99
  requirements: []
98
- rubyforge_project:
99
- rubygems_version: 2.7.6.2
100
- signing_key:
100
+ rubygems_version: 3.5.16
101
+ signing_key:
101
102
  specification_version: 4
102
103
  summary: Automatically annotate your code with Sorbet type definitions.
103
104
  test_files: []