def parse(line)
s = line.scan(/\(|\)|"(?:\\.|[^"])*"|[^()" \n]+/)
f = ->(t=nil){
case x = t || s.shift
when ?(
y = []
while x = s.shift
return y if x==?)
y << f[x]
end
fail ")?"
when nil
nil
when /^#(.*)$/
{?t=>true,?f=>false}[$~[1]]
when /\d+/
x.to_i
when /^\"(.*)\"$/m
$~[1].gsub(/\\./,'\n'=>"\n","\\\\"=>"\\","\\\""=>'"')
else
x.to_sym
end
}
f[]
end
$tests = {
"" => nil,
"1" => 1,
"(a b)" => [:a,:b],
"(1 2(2 3))" => [1,2,[2,3]],
'(a"b"c"d"(ef))' => [:a,"b",:c,"d",[:ef]],
'("a b" (c d-e))' => ["a b",[:c,:"d-e"]],
"(\"a\nb\"\nb\nc (d))" => ["a\nb",:b,:c,[:d]],
' "a\nb"' => "a\nb",
'"a\\"b"'=>'a"b',
"\"\\\\\""=> "\\",
"(#t #f)" => [true,false],
"(define (f x) (if (zero? x) 0 (+ x (f (- x 1)))))" =>
[:define,[:f,:x],[:if,[:zero?,:x],0,[:+,:x,[:f,[:-,:x,1]]]]],
}
p $tests.all?{|x,y|parse(x)==y}