285 lines
6.0 KiB
Ruby
285 lines
6.0 KiB
Ruby
|
#
|
|||
|
# irb/slex.rb - simple lex analyzer
|
|||
|
# $Release Version: 0.9.6$
|
|||
|
# $Revision: 38545 $
|
|||
|
# by Keiju ISHITSUKA(keiju@ruby-lang.org)
|
|||
|
#
|
|||
|
# --
|
|||
|
#
|
|||
|
#
|
|||
|
#
|
|||
|
|
|||
|
require "e2mmap"
|
|||
|
require "irb/notifier"
|
|||
|
|
|||
|
# :stopdoc:
|
|||
|
module IRB
|
|||
|
class SLex
|
|||
|
@RCS_ID='-$Id: slex.rb 38545 2012-12-21 17:36:14Z zzak $-'
|
|||
|
|
|||
|
extend Exception2MessageMapper
|
|||
|
def_exception :ErrNodeNothing, "node nothing"
|
|||
|
def_exception :ErrNodeAlreadyExists, "node already exists"
|
|||
|
|
|||
|
DOUT = Notifier::def_notifier("SLex::")
|
|||
|
D_WARN = DOUT::def_notifier(1, "Warn: ")
|
|||
|
D_DEBUG = DOUT::def_notifier(2, "Debug: ")
|
|||
|
D_DETAIL = DOUT::def_notifier(4, "Detail: ")
|
|||
|
|
|||
|
DOUT.level = Notifier::D_NOMSG
|
|||
|
|
|||
|
def initialize
|
|||
|
@head = Node.new("")
|
|||
|
end
|
|||
|
|
|||
|
def def_rule(token, preproc = nil, postproc = nil, &block)
|
|||
|
D_DETAIL.pp token
|
|||
|
|
|||
|
postproc = block if block_given?
|
|||
|
create(token, preproc, postproc)
|
|||
|
end
|
|||
|
|
|||
|
def def_rules(*tokens, &block)
|
|||
|
if block_given?
|
|||
|
p = block
|
|||
|
end
|
|||
|
for token in tokens
|
|||
|
def_rule(token, nil, p)
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
def preproc(token, proc)
|
|||
|
node = search(token)
|
|||
|
node.preproc=proc
|
|||
|
end
|
|||
|
|
|||
|
#$BMW%A%'%C%/(B?
|
|||
|
def postproc(token)
|
|||
|
node = search(token, proc)
|
|||
|
node.postproc=proc
|
|||
|
end
|
|||
|
|
|||
|
def search(token)
|
|||
|
@head.search(token.split(//))
|
|||
|
end
|
|||
|
|
|||
|
def create(token, preproc = nil, postproc = nil)
|
|||
|
@head.create_subnode(token.split(//), preproc, postproc)
|
|||
|
end
|
|||
|
|
|||
|
def match(token)
|
|||
|
case token
|
|||
|
when Array
|
|||
|
when String
|
|||
|
return match(token.split(//))
|
|||
|
else
|
|||
|
return @head.match_io(token)
|
|||
|
end
|
|||
|
ret = @head.match(token)
|
|||
|
D_DETAIL.exec_if{D_DETAIL.printf "match end: %s:%s\n", ret, token.inspect}
|
|||
|
ret
|
|||
|
end
|
|||
|
|
|||
|
def inspect
|
|||
|
format("<SLex: @head = %s>", @head.inspect)
|
|||
|
end
|
|||
|
|
|||
|
#----------------------------------------------------------------------
|
|||
|
#
|
|||
|
# class Node -
|
|||
|
#
|
|||
|
#----------------------------------------------------------------------
|
|||
|
class Node
|
|||
|
# if postproc is nil, this node is an abstract node.
|
|||
|
# if postproc is non-nil, this node is a real node.
|
|||
|
def initialize(preproc = nil, postproc = nil)
|
|||
|
@Tree = {}
|
|||
|
@preproc = preproc
|
|||
|
@postproc = postproc
|
|||
|
end
|
|||
|
|
|||
|
attr_accessor :preproc
|
|||
|
attr_accessor :postproc
|
|||
|
|
|||
|
def search(chrs, opt = nil)
|
|||
|
return self if chrs.empty?
|
|||
|
ch = chrs.shift
|
|||
|
if node = @Tree[ch]
|
|||
|
node.search(chrs, opt)
|
|||
|
else
|
|||
|
if opt
|
|||
|
chrs.unshift ch
|
|||
|
self.create_subnode(chrs)
|
|||
|
else
|
|||
|
SLex.fail ErrNodeNothing
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
def create_subnode(chrs, preproc = nil, postproc = nil)
|
|||
|
if chrs.empty?
|
|||
|
if @postproc
|
|||
|
D_DETAIL.pp node
|
|||
|
SLex.fail ErrNodeAlreadyExists
|
|||
|
else
|
|||
|
D_DEBUG.puts "change abstract node to real node."
|
|||
|
@preproc = preproc
|
|||
|
@postproc = postproc
|
|||
|
end
|
|||
|
return self
|
|||
|
end
|
|||
|
|
|||
|
ch = chrs.shift
|
|||
|
if node = @Tree[ch]
|
|||
|
if chrs.empty?
|
|||
|
if node.postproc
|
|||
|
DebugLogger.pp node
|
|||
|
DebugLogger.pp self
|
|||
|
DebugLogger.pp ch
|
|||
|
DebugLogger.pp chrs
|
|||
|
SLex.fail ErrNodeAlreadyExists
|
|||
|
else
|
|||
|
D_WARN.puts "change abstract node to real node"
|
|||
|
node.preproc = preproc
|
|||
|
node.postproc = postproc
|
|||
|
end
|
|||
|
else
|
|||
|
node.create_subnode(chrs, preproc, postproc)
|
|||
|
end
|
|||
|
else
|
|||
|
if chrs.empty?
|
|||
|
node = Node.new(preproc, postproc)
|
|||
|
else
|
|||
|
node = Node.new
|
|||
|
node.create_subnode(chrs, preproc, postproc)
|
|||
|
end
|
|||
|
@Tree[ch] = node
|
|||
|
end
|
|||
|
node
|
|||
|
end
|
|||
|
|
|||
|
#
|
|||
|
# chrs: String
|
|||
|
# character array
|
|||
|
# io must have getc()/ungetc(); and ungetc() must be
|
|||
|
# able to be called arbitrary number of times.
|
|||
|
#
|
|||
|
def match(chrs, op = "")
|
|||
|
D_DETAIL.print "match>: ", chrs, "op:", op, "\n"
|
|||
|
if chrs.empty?
|
|||
|
if @preproc.nil? || @preproc.call(op, chrs)
|
|||
|
DOUT.printf(D_DETAIL, "op1: %s\n", op)
|
|||
|
@postproc.call(op, chrs)
|
|||
|
else
|
|||
|
nil
|
|||
|
end
|
|||
|
else
|
|||
|
ch = chrs.shift
|
|||
|
if node = @Tree[ch]
|
|||
|
if ret = node.match(chrs, op+ch)
|
|||
|
return ret
|
|||
|
else
|
|||
|
chrs.unshift ch
|
|||
|
if @postproc and @preproc.nil? || @preproc.call(op, chrs)
|
|||
|
DOUT.printf(D_DETAIL, "op2: %s\n", op.inspect)
|
|||
|
ret = @postproc.call(op, chrs)
|
|||
|
return ret
|
|||
|
else
|
|||
|
return nil
|
|||
|
end
|
|||
|
end
|
|||
|
else
|
|||
|
chrs.unshift ch
|
|||
|
if @postproc and @preproc.nil? || @preproc.call(op, chrs)
|
|||
|
DOUT.printf(D_DETAIL, "op3: %s\n", op)
|
|||
|
@postproc.call(op, chrs)
|
|||
|
return ""
|
|||
|
else
|
|||
|
return nil
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
|
|||
|
def match_io(io, op = "")
|
|||
|
if op == ""
|
|||
|
ch = io.getc
|
|||
|
if ch == nil
|
|||
|
return nil
|
|||
|
end
|
|||
|
else
|
|||
|
ch = io.getc_of_rests
|
|||
|
end
|
|||
|
if ch.nil?
|
|||
|
if @preproc.nil? || @preproc.call(op, io)
|
|||
|
D_DETAIL.printf("op1: %s\n", op)
|
|||
|
@postproc.call(op, io)
|
|||
|
else
|
|||
|
nil
|
|||
|
end
|
|||
|
else
|
|||
|
if node = @Tree[ch]
|
|||
|
if ret = node.match_io(io, op+ch)
|
|||
|
ret
|
|||
|
else
|
|||
|
io.ungetc ch
|
|||
|
if @postproc and @preproc.nil? || @preproc.call(op, io)
|
|||
|
DOUT.exec_if{D_DETAIL.printf "op2: %s\n", op.inspect}
|
|||
|
@postproc.call(op, io)
|
|||
|
else
|
|||
|
nil
|
|||
|
end
|
|||
|
end
|
|||
|
else
|
|||
|
io.ungetc ch
|
|||
|
if @postproc and @preproc.nil? || @preproc.call(op, io)
|
|||
|
D_DETAIL.printf("op3: %s\n", op)
|
|||
|
@postproc.call(op, io)
|
|||
|
else
|
|||
|
nil
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
end
|
|||
|
# :startdoc:
|
|||
|
|
|||
|
if $0 == __FILE__
|
|||
|
# Tracer.on
|
|||
|
case $1
|
|||
|
when "1"
|
|||
|
tr = SLex.new
|
|||
|
print "0: ", tr.inspect, "\n"
|
|||
|
tr.def_rule("=") {print "=\n"}
|
|||
|
print "1: ", tr.inspect, "\n"
|
|||
|
tr.def_rule("==") {print "==\n"}
|
|||
|
print "2: ", tr.inspect, "\n"
|
|||
|
|
|||
|
print "case 1:\n"
|
|||
|
print tr.match("="), "\n"
|
|||
|
print "case 2:\n"
|
|||
|
print tr.match("=="), "\n"
|
|||
|
print "case 3:\n"
|
|||
|
print tr.match("=>"), "\n"
|
|||
|
|
|||
|
when "2"
|
|||
|
tr = SLex.new
|
|||
|
print "0: ", tr.inspect, "\n"
|
|||
|
tr.def_rule("=") {print "=\n"}
|
|||
|
print "1: ", tr.inspect, "\n"
|
|||
|
tr.def_rule("==", proc{false}) {print "==\n"}
|
|||
|
print "2: ", tr.inspect, "\n"
|
|||
|
|
|||
|
print "case 1:\n"
|
|||
|
print tr.match("="), "\n"
|
|||
|
print "case 2:\n"
|
|||
|
print tr.match("=="), "\n"
|
|||
|
print "case 3:\n"
|
|||
|
print tr.match("=>"), "\n"
|
|||
|
end
|
|||
|
exit
|
|||
|
end
|
|||
|
|