Me temo que las expresiones regulares no te llevarán muy lejos. Considere, por ejemplo, las siguientes expresiones (que también son válidos Rubí)
"(foo.bar.size.split('.')).last"
"(foo.bar.size.split '.').last"
"(foo.bar.size.split '(.) . .(). .').last"
El problema es que la lista de llamadas es en realidad un árbol de llamadas. La solución más fácil a la vista es probablemente usar un analizador Ruby y transformar el árbol de análisis sintáctico de acuerdo a sus necesidades (en este ejemplo estamos en descendente en el árbol de llamadas, reuniendo las llamadas en una lista):
# gem install ruby_parser
# gem install awesome_print
require 'ruby_parser'
require 'ap'
def calls_as_list code
tree = RubyParser.new.parse(code)
t = tree
calls = []
while t
# gather arguments if present
args = nil
if t[3][0] == :arglist
args = t[3][1..-1].to_a
end
# append all information to our list
calls << [t[2].to_s, args]
# descend to next call
t = t[1]
end
calls.reverse
end
p calls_as_list "foo.bar.size.split('.').last"
#=> [["foo", []], ["bar", []], ["size", []], ["split", [[:str, "."]]], ["last", []]]
p calls_as_list "puts 3, 4"
#=> [["puts", [[:lit, 3], [:lit, 4]]]]
Y para mostrar el árbol de análisis sintáctico de cualquier entrada:
ap RubyParser.new.parse("puts 3, 4")
No es tan fácil como crees. – sawa
@sawa: ¿ha cerrado una pregunta porque piensa que es demasiado difícil? – iconoclast
@iconoclast No lo recuerdo, pero no por la razón que piensas. – sawa