ruby_traverser 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +21 -0
- data/LICENSE +20 -0
- data/README.markdown +195 -0
- data/Rakefile +48 -0
- data/VERSION +1 -0
- data/lib/mutate/api.rb +124 -0
- data/lib/mutate/replacer.rb +129 -0
- data/lib/rails/api_wrapper.rb +23 -0
- data/lib/ruby_traverser.rb +4 -0
- data/lib/traversal/api/finders.rb +41 -0
- data/lib/traversal/api/inside.rb +36 -0
- data/lib/traversal/api/traversal.rb +20 -0
- data/lib/traversal/misc.rb +152 -0
- data/lib/traversal/mixin.rb +50 -0
- data/lib/traversal/module.rb +52 -0
- data/lib/traversal/patch.rb +83 -0
- data/ruby_traverser.gemspec +79 -0
- data/spec/ruby_traverser_spec.rb +7 -0
- data/spec/spec.opts +1 -0
- data/spec/spec_helper.rb +10 -0
- data/test/mutate/mutate_class_test.rb +29 -0
- data/test/mutate/mutate_test.rb +90 -0
- data/test/rails_api/gemfile_api.rb +27 -0
- data/test/test_helper.rb +53 -0
- data/test/traversal/gemfile_test.rb +92 -0
- data/test/traversal/ruby_api_traversal_test.rb +128 -0
- metadata +122 -0
data/.gitignore
ADDED
data/LICENSE
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2009 Kristian Mandrup
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.markdown
ADDED
@@ -0,0 +1,195 @@
|
|
1
|
+
# Ruby traverser ##
|
2
|
+
|
3
|
+
A DSL for traversing ruby code as an object model (graph), which lets you find parts of the ruby code of interest.
|
4
|
+
The traverser leverages `ripper2ruby`, which leverages `ripper`, which comes with ruby 1.9. Ruby 1.9 is thus required.
|
5
|
+
|
6
|
+
See the unit tests in the test directory for examples of use.
|
7
|
+
|
8
|
+
## Finders ##
|
9
|
+
|
10
|
+
* find_module(name)
|
11
|
+
* find_class(name, options = {})
|
12
|
+
* find_call(name, options = {})
|
13
|
+
* find_block(name, options = {})
|
14
|
+
* find_def(name, options = {})
|
15
|
+
* find_variable(name, options = {})
|
16
|
+
* find_assignment(name, options = {})
|
17
|
+
|
18
|
+
## Find module ##
|
19
|
+
|
20
|
+
<pre>
|
21
|
+
src = %q{
|
22
|
+
module Xyz::Xxx::Blip
|
23
|
+
2
|
24
|
+
end
|
25
|
+
}
|
26
|
+
|
27
|
+
code = Ripper::RubyBuilder.build(src)
|
28
|
+
module_node = code.find_module('Xyz::Xxx::Blip')
|
29
|
+
assert_equal Ruby::Module, module_node.class
|
30
|
+
</pre>
|
31
|
+
|
32
|
+
## Find class ##
|
33
|
+
|
34
|
+
<pre>
|
35
|
+
# class Monty::Python ... end
|
36
|
+
clazz_node = code.find_class('Monty::Python')
|
37
|
+
</pre>
|
38
|
+
|
39
|
+
Find class inheriting from a certain superclass
|
40
|
+
<pre>
|
41
|
+
# class Monty < Abc::Blip ... end
|
42
|
+
clazz_node = code.find_class('Monty', :superclass => 'Abc::Blip')
|
43
|
+
</pre>
|
44
|
+
|
45
|
+
## Find call ##
|
46
|
+
|
47
|
+
<pre>
|
48
|
+
# gem 'ripper', :src => 'github'
|
49
|
+
gem_call = code.find_call('gem', :args => ['ripper', {:src => 'github'}])
|
50
|
+
</pre>
|
51
|
+
|
52
|
+
|
53
|
+
## Find block ##
|
54
|
+
|
55
|
+
<pre>
|
56
|
+
# my_block do ... end
|
57
|
+
block_node = code.find_block('my_block')
|
58
|
+
</pre>
|
59
|
+
|
60
|
+
<pre>
|
61
|
+
# my_block do |v| ... end
|
62
|
+
block_node = code.find_block('my_block', :block_params => ['v'])
|
63
|
+
</pre>
|
64
|
+
|
65
|
+
<pre>
|
66
|
+
# my_block 7, 'a' do ... end
|
67
|
+
block_node = code.find_block('my_block', :args => [7, 'a'])
|
68
|
+
</pre>
|
69
|
+
|
70
|
+
<pre>
|
71
|
+
# my_block 7, 'a', :k => 32 do |v| ... end
|
72
|
+
block_node = code.find_block('my_block', :args => [7, 'a', {:k => 32}], :block_params => ['v'])
|
73
|
+
</pre>
|
74
|
+
|
75
|
+
<pre>
|
76
|
+
# my_block :a => 7, b => 3 do |v| ... end
|
77
|
+
block_node = code.find_block('my_block', :args => [{:a => 7, 'b' => 3}])
|
78
|
+
</pre>
|
79
|
+
|
80
|
+
<pre>
|
81
|
+
# my_block ['a', 'b'] do |v| ... end
|
82
|
+
block_node = code.find_block('my_block', :args => [{:array =>['a', 'b']}])
|
83
|
+
</pre>
|
84
|
+
|
85
|
+
|
86
|
+
## Find variable ##
|
87
|
+
|
88
|
+
Source code:
|
89
|
+
<pre>
|
90
|
+
def hello_world(a)
|
91
|
+
my_var
|
92
|
+
end
|
93
|
+
</pre>
|
94
|
+
|
95
|
+
Ruby code find DSL:
|
96
|
+
<pre>
|
97
|
+
code = Ripper::RubyBuilder.build(src)
|
98
|
+
code.inside_def('hello_world', :params => ['a']) do |b|
|
99
|
+
call_node = b.find_variable('my_var')
|
100
|
+
assert_equal Ruby::Variable, call_node.class
|
101
|
+
puts call_node.to_ruby
|
102
|
+
end
|
103
|
+
</pre>
|
104
|
+
|
105
|
+
|
106
|
+
## Find assignment ##
|
107
|
+
|
108
|
+
Source code:
|
109
|
+
<pre>
|
110
|
+
def hello_world(a)
|
111
|
+
my_var = 2
|
112
|
+
end
|
113
|
+
</pre>
|
114
|
+
|
115
|
+
Ruby code find DSL:
|
116
|
+
<pre>
|
117
|
+
code = Ripper::RubyBuilder.build(src)
|
118
|
+
code.inside_def('hello_world', :params => ['a']) do |b|
|
119
|
+
call_node = b.find_assignment('my_var')
|
120
|
+
end
|
121
|
+
</pre>
|
122
|
+
|
123
|
+
## Inside ##
|
124
|
+
|
125
|
+
The following finder methods have corresponding `inside_` functions, which support block DSL constructs as shown below.
|
126
|
+
|
127
|
+
* inside_module
|
128
|
+
* inside_class
|
129
|
+
* inside_def
|
130
|
+
* inside_block
|
131
|
+
|
132
|
+
<pre>
|
133
|
+
# source code
|
134
|
+
src = %q{
|
135
|
+
gem 'ripper', :src => 'github', :blip => 'blap'
|
136
|
+
group :test do
|
137
|
+
gem 'ripper', :src => 'github'
|
138
|
+
end
|
139
|
+
}
|
140
|
+
</pre>
|
141
|
+
|
142
|
+
<pre>
|
143
|
+
code = Ripper::RubyBuilder.build(src)
|
144
|
+
# chaining finders using 'inside__' DSL block constructs
|
145
|
+
code.inside_block('group', :args => [:test]) do |b|
|
146
|
+
call_node = b.find_call('gem', :args => ['ripper', {:src => 'github'}])
|
147
|
+
assert_equal Ruby::Call, call_node.class
|
148
|
+
puts call_node.to_ruby # output ruby code as string for found node
|
149
|
+
end
|
150
|
+
</pre>
|
151
|
+
|
152
|
+
<pre>
|
153
|
+
src = %q{
|
154
|
+
def hello_world(b)
|
155
|
+
3
|
156
|
+
end
|
157
|
+
|
158
|
+
def hello_world(a)
|
159
|
+
gem 'ripper', :src => 'github'
|
160
|
+
end
|
161
|
+
|
162
|
+
}
|
163
|
+
</pre>
|
164
|
+
|
165
|
+
<pre>
|
166
|
+
# chaining finders using 'inside__' DSL block constructs
|
167
|
+
code = Ripper::RubyBuilder.build(src)
|
168
|
+
code.inside_def('hello_world', :params => ['a']) do |b|
|
169
|
+
call_node = b.find_call('gem', :args => ['ripper', {:src => 'github'}])
|
170
|
+
assert_equal Ruby::Call, call_node.class
|
171
|
+
puts call_node.to_ruby # output ruby code as string for found node
|
172
|
+
end
|
173
|
+
</pre>
|
174
|
+
|
175
|
+
## Code Mutation API ##
|
176
|
+
|
177
|
+
The API now also supports a wide variety of code mutations using a DSL.
|
178
|
+
More information will soon be available here or on the github wiki.
|
179
|
+
Check the test/mutate folder for test demonstrating what is currently possible.
|
180
|
+
|
181
|
+
Note: The mutation API code was developed in a test-driven fashion, but is in need of a major refactoring overhaul sometime soon...
|
182
|
+
|
183
|
+
## Note on Patches/Pull Requests ##
|
184
|
+
|
185
|
+
* Fork the project.
|
186
|
+
* Make your feature addition or bug fix.
|
187
|
+
* Add tests for it. This is important so I don't break it in a
|
188
|
+
future version unintentionally.
|
189
|
+
* Commit, do not mess with rakefile, version, or history.
|
190
|
+
(if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull)
|
191
|
+
* Send me a pull request. Bonus points for topic branches.
|
192
|
+
|
193
|
+
## Copyright ##
|
194
|
+
|
195
|
+
Copyright (c) 2010 Kristian Mandrup. See LICENSE for details.
|
data/Rakefile
ADDED
@@ -0,0 +1,48 @@
|
|
1
|
+
# require 'rubygems'
|
2
|
+
# require 'rake'
|
3
|
+
|
4
|
+
begin
|
5
|
+
require 'jeweler'
|
6
|
+
Jeweler::Tasks.new do |gem|
|
7
|
+
gem.name = "ruby_traverser"
|
8
|
+
gem.summary = %Q{traverse and mutate ruby code using a nice DSL}
|
9
|
+
gem.description = %Q{traverse a ruby code model and optionally mutate it along the way using a nice rubyish DSL}
|
10
|
+
gem.email = "kmandrup@gmail.com"
|
11
|
+
gem.homepage = "http://github.com/kristianmandrup/ruby_trav"
|
12
|
+
gem.authors = ["Kristian Mandrup"]
|
13
|
+
gem.add_development_dependency "rspec", ">= 2.0.0"
|
14
|
+
gem.add_dependency "ripper2ruby", "> 0.0.2"
|
15
|
+
# gem is a Gem::Specification... see http://www.rubygems.org/read/chapter/20 for additional settings
|
16
|
+
|
17
|
+
# add more gem options here
|
18
|
+
end
|
19
|
+
rescue LoadError
|
20
|
+
puts "Jeweler (or a dependency) not available. Install it with: gem install jeweler"
|
21
|
+
end
|
22
|
+
|
23
|
+
# require 'rspec/rake_task'
|
24
|
+
# Rspec::Core::RakeTask.new(:spec) do |spec|
|
25
|
+
# # spec.libs << 'lib' << 'spec'
|
26
|
+
# # spec.spec_files = FileList['spec/**/*_spec.rb']
|
27
|
+
# end
|
28
|
+
#
|
29
|
+
# require 'rspec/rake_task'
|
30
|
+
# Rspec::Core::RakeTask.new(:rcov) do |spec|
|
31
|
+
# # spec.libs << 'lib' << 'spec'
|
32
|
+
# # spec.pattern = 'spec/**/*_spec.rb'
|
33
|
+
# # spec.rcov = true
|
34
|
+
# end
|
35
|
+
#
|
36
|
+
# task :spec => :check_dependencies
|
37
|
+
#
|
38
|
+
# task :default => :spec
|
39
|
+
#
|
40
|
+
# require 'rake/rdoctask'
|
41
|
+
# Rake::RDocTask.new do |rdoc|
|
42
|
+
# version = File.exist?('VERSION') ? File.read('VERSION') : ""
|
43
|
+
#
|
44
|
+
# rdoc.rdoc_dir = 'rdoc'
|
45
|
+
# rdoc.title = "ruby_trav #{version}"
|
46
|
+
# rdoc.rdoc_files.include('README*')
|
47
|
+
# rdoc.rdoc_files.include('lib/**/*.rb')
|
48
|
+
# end
|
data/VERSION
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
0.1.0
|
data/lib/mutate/api.rb
ADDED
@@ -0,0 +1,124 @@
|
|
1
|
+
require 'mutate/replacer'
|
2
|
+
|
3
|
+
class String
|
4
|
+
# Returns an indented string, all lines of string will be indented with count of chars
|
5
|
+
def indent(char, count)
|
6
|
+
(char * count) + gsub(/(\n+)/) { $1 + (char * count) }
|
7
|
+
end
|
8
|
+
end
|
9
|
+
|
10
|
+
|
11
|
+
module RubyAPI
|
12
|
+
module Mutator
|
13
|
+
include Replacer
|
14
|
+
|
15
|
+
def append_code(code)
|
16
|
+
return append_code_simple(code) if !elemental?
|
17
|
+
obj = object
|
18
|
+
indentation = obj.last_indent
|
19
|
+
code = "\n#{code}\n".indent(' ', indentation)
|
20
|
+
ruby_code = Ripper::RubyBuilder.build(code)
|
21
|
+
inject_code = ruby_code.elements[0]
|
22
|
+
obj.get_elements << inject_code
|
23
|
+
inject_code
|
24
|
+
end
|
25
|
+
|
26
|
+
def append_code_simple(code)
|
27
|
+
indentation = position.col
|
28
|
+
code = "\n#{code}\n".indent(' ', indentation)
|
29
|
+
ruby_code = Ripper::RubyBuilder.build(code)
|
30
|
+
inject_code = ruby_code.elements[0]
|
31
|
+
index = parent.find_index(self)
|
32
|
+
parent.get_elements.insert(index+1, inject_code)
|
33
|
+
inject_code
|
34
|
+
end
|
35
|
+
|
36
|
+
def find_index(obj)
|
37
|
+
get_elements.each_with_index do |elem, i|
|
38
|
+
if elem == obj
|
39
|
+
return i
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def prepend_code(code)
|
45
|
+
obj = object
|
46
|
+
indentation = obj.first_indent
|
47
|
+
code = "\n#{code}\n".indent(' ', indentation)
|
48
|
+
ruby_code = Ripper::RubyBuilder.build(code)
|
49
|
+
inject_code = ruby_code.elements[0]
|
50
|
+
obj.get_elements.insert(0, inject_code)
|
51
|
+
obj
|
52
|
+
end
|
53
|
+
|
54
|
+
|
55
|
+
def append_comment(text)
|
56
|
+
append_code("# #{text}")
|
57
|
+
end
|
58
|
+
|
59
|
+
def elemental?
|
60
|
+
respond_to?(:body) || respond_to?(:elements)
|
61
|
+
end
|
62
|
+
|
63
|
+
def get_elements
|
64
|
+
case self
|
65
|
+
when Ruby::Class
|
66
|
+
body.elements
|
67
|
+
else
|
68
|
+
elements
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
|
73
|
+
protected
|
74
|
+
|
75
|
+
def last_indent
|
76
|
+
case self
|
77
|
+
when Ruby::Block, Ruby::Class
|
78
|
+
last_position(get_elements)
|
79
|
+
else
|
80
|
+
puts "unknown: #{obj.class}"
|
81
|
+
end
|
82
|
+
end
|
83
|
+
|
84
|
+
def last_position(elements)
|
85
|
+
last_element = elements.last
|
86
|
+
return position.col
|
87
|
+
return position.col if simple_pos?
|
88
|
+
return last_element.identifier.position.col if elements && elements.size > 0
|
89
|
+
inside_indent
|
90
|
+
end
|
91
|
+
|
92
|
+
def simple_pos?
|
93
|
+
[Ruby::Token, Ruby::Variable].include?(last_element.class)
|
94
|
+
end
|
95
|
+
|
96
|
+
def first_indent
|
97
|
+
case self
|
98
|
+
when Ruby::Block, Ruby::Class
|
99
|
+
first_position(get_elements)
|
100
|
+
else
|
101
|
+
puts "unknown: #{obj.class}"
|
102
|
+
end
|
103
|
+
end
|
104
|
+
|
105
|
+
def first_position(elements)
|
106
|
+
first_element = elements.first
|
107
|
+
return position.col if simple_pos?
|
108
|
+
return first_element.identifier.position.col if elements && elements.size > 0
|
109
|
+
inside_indent
|
110
|
+
end
|
111
|
+
|
112
|
+
def object
|
113
|
+
self.respond_to?(:block) ? self.block : self
|
114
|
+
end
|
115
|
+
|
116
|
+
end
|
117
|
+
end
|
118
|
+
|
119
|
+
module Ruby
|
120
|
+
class Node
|
121
|
+
include RubyAPI::Mutator
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
@@ -0,0 +1,129 @@
|
|
1
|
+
module PositionReplacer
|
2
|
+
def replace_position_arg(options)
|
3
|
+
pos = position_arg?(options[:arg])
|
4
|
+
self.arguments.elements[pos.to_i].replace_pos_argument(options)
|
5
|
+
end
|
6
|
+
|
7
|
+
def replace_pos_argument(options)
|
8
|
+
case self.arg
|
9
|
+
when Ruby::String
|
10
|
+
replace_arg_token(options[:replace_arg])
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
def position_arg?(arg)
|
15
|
+
return arg[1] if arg && arg[0] == '#'
|
16
|
+
nil
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
module TokenReplacer
|
21
|
+
def replace_arg_token(replacement)
|
22
|
+
self.arg.elements[0].token = replacement
|
23
|
+
end
|
24
|
+
|
25
|
+
def matching_string_arg?(txt)
|
26
|
+
self.arg.elements[0].token == txt
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
module HashReplacer
|
31
|
+
def hash_arg?(arg)
|
32
|
+
case arg
|
33
|
+
when Hash
|
34
|
+
matching_hash_arg?(arg)
|
35
|
+
when Symbol
|
36
|
+
matching_symbol_arg?(arg)
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
def replace_hash_arg(options)
|
41
|
+
src = options[:replace_code]
|
42
|
+
code = Ripper::RubyBuilder.build(src)
|
43
|
+
code.set_ldelim(self.arg)
|
44
|
+
self.arg = code
|
45
|
+
end
|
46
|
+
|
47
|
+
def set_ldelim(arg)
|
48
|
+
if arg.respond_to? :elements
|
49
|
+
self.ldelim = arg.elements[0].key.ldelim
|
50
|
+
self.ldelim.token = ''
|
51
|
+
else
|
52
|
+
self.ldelim = arg.ldelim
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
def matching_hash_arg?(arg)
|
57
|
+
if self.arg.respond_to? :elements
|
58
|
+
if self.arg.elements[0].class == Ruby::Assoc
|
59
|
+
key = self.arg.elements[0].key
|
60
|
+
value = self.arg.elements[0].value
|
61
|
+
arg_key = arg.first[0]
|
62
|
+
arg_value = arg.first[1]
|
63
|
+
return key.identifier.token.to_sym == arg_key && value.elements[0].token == arg_value
|
64
|
+
end
|
65
|
+
end
|
66
|
+
false
|
67
|
+
end
|
68
|
+
|
69
|
+
def matching_symbol_arg?(arg)
|
70
|
+
if self.arg.respond_to? :elements
|
71
|
+
if self.arg.elements[0].class == Ruby::Assoc
|
72
|
+
return self.arg.elements[0].key.identifier.token.to_sym == arg
|
73
|
+
end
|
74
|
+
else
|
75
|
+
# remove ':' token from symbol
|
76
|
+
self.arg.ldelim.token = ''
|
77
|
+
return self.arg.identifier.token.to_sym == arg
|
78
|
+
end
|
79
|
+
false
|
80
|
+
end
|
81
|
+
end
|
82
|
+
|
83
|
+
module ValueReplacer
|
84
|
+
def replace_value(options)
|
85
|
+
if self.class == Ruby::Assignment
|
86
|
+
self.right.token = options[:value]
|
87
|
+
end
|
88
|
+
end
|
89
|
+
end
|
90
|
+
|
91
|
+
module RubyAPI
|
92
|
+
module Mutator
|
93
|
+
module Replacer
|
94
|
+
include PositionReplacer
|
95
|
+
include TokenReplacer
|
96
|
+
include HashReplacer
|
97
|
+
include ValueReplacer
|
98
|
+
|
99
|
+
# :arg => 'ripper', :replace_arg => 'rapper'
|
100
|
+
def replace(options)
|
101
|
+
if options[:value]
|
102
|
+
return replace_value(options)
|
103
|
+
end
|
104
|
+
|
105
|
+
if position_arg?(options[:arg])
|
106
|
+
return replace_position_arg(options)
|
107
|
+
end
|
108
|
+
|
109
|
+
self.arguments.elements.each_with_index do |elem, i|
|
110
|
+
case elem
|
111
|
+
when Ruby::Arg
|
112
|
+
if elem.hash_arg?(options[:arg])
|
113
|
+
return elem.replace_hash_arg(options)
|
114
|
+
end
|
115
|
+
elem.replace_argument(options)
|
116
|
+
end
|
117
|
+
end
|
118
|
+
end
|
119
|
+
|
120
|
+
def replace_argument(options)
|
121
|
+
case self.arg
|
122
|
+
when Ruby::String
|
123
|
+
replace_arg_token(options[:replace_arg]) if matching_string_arg?(options[:arg])
|
124
|
+
end
|
125
|
+
end
|
126
|
+
|
127
|
+
end
|
128
|
+
end
|
129
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module RubyAPI
|
2
|
+
module Rails
|
3
|
+
module Gemfile
|
4
|
+
def inside_group(name, &block)
|
5
|
+
inside_block('group', :args => [:"#{name}"], :extend => RubyAPI::Rails::Gemfile, &block)
|
6
|
+
end
|
7
|
+
|
8
|
+
def add_gem(name)
|
9
|
+
append_code("gem '#{name}'")
|
10
|
+
end
|
11
|
+
|
12
|
+
def replace_gem(name, replace_name)
|
13
|
+
found = find_gem(name)
|
14
|
+
puts "gem:#{found}"
|
15
|
+
found.replace(:arg => name, :replace_arg => replace_name) if found
|
16
|
+
end
|
17
|
+
|
18
|
+
def find_gem(name, options = nil)
|
19
|
+
find_call('gem', :args => ["#{name}", options])
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,41 @@
|
|
1
|
+
require 'yaml'
|
2
|
+
|
3
|
+
module RubyAPI
|
4
|
+
module Finders
|
5
|
+
def find_module(name)
|
6
|
+
get_obj.select(Ruby::Module, :identifier => name).first
|
7
|
+
end
|
8
|
+
|
9
|
+
def find_class(name, options = {})
|
10
|
+
options.merge!(:identifier => name)
|
11
|
+
get_obj.select(Ruby::Class, options).first
|
12
|
+
end
|
13
|
+
|
14
|
+
def find_variable(name, options = {})
|
15
|
+
options.merge!(:token => name)
|
16
|
+
get_obj.select(Ruby::Variable, options).first
|
17
|
+
end
|
18
|
+
|
19
|
+
def find_assignment(name, options = {})
|
20
|
+
options.merge!(:left_token => name)
|
21
|
+
get_obj.select(Ruby::Assignment, options).first
|
22
|
+
end
|
23
|
+
|
24
|
+
def find_call(name, options = {})
|
25
|
+
options.merge!(:identifier => name)
|
26
|
+
get_obj.select(Ruby::Call, options).first
|
27
|
+
end
|
28
|
+
|
29
|
+
def find_block(name, options = {})
|
30
|
+
options.merge!(:identifier => name)
|
31
|
+
options.merge!(:block => true) if !options.has_key?(:block_params)
|
32
|
+
get_obj.select(Ruby::Call, options).first
|
33
|
+
end
|
34
|
+
|
35
|
+
|
36
|
+
def find_def(name, options = {})
|
37
|
+
options.merge!(:identifier => name)
|
38
|
+
get_obj.select(Ruby::Method, options).first
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
@@ -0,0 +1,36 @@
|
|
1
|
+
module RubyAPI
|
2
|
+
module Inside
|
3
|
+
def inside_block(name, options = {}, &block)
|
4
|
+
call_block(find_block(name, options), options, &block)
|
5
|
+
end
|
6
|
+
|
7
|
+
def inside_module(name, &block)
|
8
|
+
call_block(find_module(name), options, &block)
|
9
|
+
end
|
10
|
+
|
11
|
+
def inside_class(name, options = {}, &block)
|
12
|
+
call_block(find_class(name, options), options, &block)
|
13
|
+
end
|
14
|
+
|
15
|
+
def inside_def(name, options = {}, &block)
|
16
|
+
call_block(find_def(name, options), options, &block)
|
17
|
+
end
|
18
|
+
|
19
|
+
protected
|
20
|
+
|
21
|
+
def call_block(s, options, &block)
|
22
|
+
# inc_inside_indent
|
23
|
+
s.extend(options[:extend]) if options[:extend]
|
24
|
+
block.arity < 1 ? s.instance_eval(&block) : block.call(s)
|
25
|
+
end
|
26
|
+
|
27
|
+
def inside_indent
|
28
|
+
if self.class == Ruby::Class
|
29
|
+
pos = ldelim.position.col
|
30
|
+
return pos
|
31
|
+
end
|
32
|
+
2
|
33
|
+
end
|
34
|
+
|
35
|
+
end
|
36
|
+
end
|
@@ -0,0 +1,20 @@
|
|
1
|
+
require 'traversal/api/finders'
|
2
|
+
require 'traversal/api/inside'
|
3
|
+
|
4
|
+
module RubyAPI
|
5
|
+
include Finders
|
6
|
+
include Inside
|
7
|
+
|
8
|
+
protected
|
9
|
+
|
10
|
+
def get_obj(options = {})
|
11
|
+
return self.block if self.class == Ruby::Method
|
12
|
+
self
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
module Ruby
|
17
|
+
class Node
|
18
|
+
include RubyAPI
|
19
|
+
end
|
20
|
+
end
|