# encoding: utf-8 require 'rouge/context' require 'rouge/builtins' require 'rouge/var' require 'rouge/atom' class Rouge::Namespace @namespaces = {} class VarNotFoundError < StandardError; end class RecursiveNamespaceError < StandardError; end def initialize(name) @name = name raise ArgumentError, "bad ns name" unless @name.is_a? Symbol @table = {} @refers = [] end def inspect "#" end def refer(ns) if ns.name == @name raise RecursiveNamespaceError, "#@name will not refer #{ns.name}" end @refers << ns if not @refers.include? ns self end def [](key) if @table.include? key return @table[key] end @refers.each do |ns| begin return ns[key] rescue VarNotFoundError # no-op end end raise VarNotFoundError, key end def set_here(key, value) @table[key] = Rouge::Var.new(@name, key, value) end def intern(key) @table[key] ||= Rouge::Var.new(@name, key) end def read(input) Rouge::Reader.new(self, input).lex end def clear @table = {} self end attr_reader :name, :refers end class << Rouge::Namespace def exists?(ns) @namespaces.include? ns end def [](ns) r = @namespaces[ns] return r if r self[ns] = new(ns) @namespaces[ns] = new(ns) end def get(ns) @namespaces[ns] end def []=(ns, value) @namespaces[ns] = value end def destroy(ns) @namespaces.delete ns end end class Rouge::Namespace::Ruby @@cache = {} def [](name) return @@cache[name] if @@cache.include? name if name =~ /^\$/ @@cache[name] = Rouge::Var.new(:ruby, name, eval(name.to_s)) else @@cache[name] = Rouge::Var.new(:ruby, name, Kernel.const_get(name)) end rescue NameError raise Rouge::Namespace::VarNotFoundError, name end def set_here(name, value) @@cache[name] = Rouge::Var.new(:ruby, name, value) Kernel.const_set name, value end def name :ruby end end ns = Rouge::Namespace[:"rouge.builtin"] Rouge::Builtins.methods(false).reject {|s| s =~ /^_compile_/}.each do |m| ns.set_here m, Rouge::Builtin[Rouge::Builtins.method(m)] end Rouge::Builtins::SYMBOLS.each do |name, val| ns.set_here name, val end Rouge::Namespace[:ruby] = Rouge::Namespace::Ruby.new # vim: set sw=2 et cc=80: