require 'rubygems' require 'bert' require 'logger' class Ernie class << self attr_accessor :mods, :current_mod, :log attr_accessor :auto_start end self.mods = {} self.current_mod = nil self.log = Logger.new(STDOUT) self.log.level = Logger::FATAL self.auto_start = true # Record a module. # +name+ is the module Symbol # +block+ is the Block containing function definitions # # Returns nothing def self.mod(name, block) m = Mod.new(name) self.current_mod = m self.mods[name] = m block.call end # Record a function. # +name+ is the function Symbol # +block+ is the Block to associate # # Returns nothing def self.fun(name, block) self.current_mod.fun(name, block) end # Expose all public methods in a Ruby module: # +name+ is the ernie module Symbol # +mixin+ is the ruby module whose public methods are exposed # # Returns nothing def self.expose(name, mixin) context = Object.new context.extend mixin mod(name, lambda { mixin.public_instance_methods.each do |meth| fun(meth.to_sym, context.method(meth)) end }) context end # Set the logfile to given path. # +file+ is the String path to the logfile # # Returns nothing def self.logfile(file) self.log = Logger.new(file) end # Set the log level. # +level+ is the Logger level (Logger::WARN, etc) # # Returns nothing def self.loglevel(level) self.log.level = level end # Dispatch the request to the proper mod:fun. # +mod+ is the module Symbol # +fun+ is the function Symbol # +args+ is the Array of arguments # # Returns the Ruby object response def self.dispatch(mod, fun, args) self.mods[mod] || raise(ServerError.new("No such module '#{mod}'")) self.mods[mod].funs[fun] || raise(ServerError.new("No such function '#{mod}:#{fun}'")) self.mods[mod].funs[fun].call(*args) end # Read the length header from the wire. # +input+ is the IO from which to read # # Returns the size Integer if one was read # Returns nil otherwise def self.read_4(input) raw = input.read(4) return nil unless raw raw.unpack('N').first end # Read a BERP from the wire and decode it to a Ruby object. # +input+ is the IO from which to read # # Returns a Ruby object if one could be read # Returns nil otherwise def self.read_berp(input) packet_size = self.read_4(input) return nil unless packet_size bert = input.read(packet_size) BERT.decode(bert) end # Write the given Ruby object to the wire as a BERP. # +output+ is the IO on which to write # +ruby+ is the Ruby object to encode # # Returns nothing def self.write_berp(output, ruby) data = BERT.encode(ruby) output.write([data.length].pack("N")) output.write(data) end # Start the processing loop. # # Loops forever def self.start self.log.info("(#{Process.pid}) Starting") self.log.debug(self.mods.inspect) input = IO.new(3) output = IO.new(4) input.sync = true output.sync = true loop do iruby = self.read_berp(input) unless iruby puts "Could not read BERP length header. Ernie server may have gone away. Exiting now." self.log.info("(#{Process.pid}) Could not read BERP length header. Ernie server may have gone away. Exiting now.") exit! end if iruby.size == 4 && iruby[0] == :call mod, fun, args = iruby[1..3] self.log.info("-> " + iruby.inspect) begin res = self.dispatch(mod, fun, args) oruby = t[:reply, res] self.log.debug("<- " + oruby.inspect) write_berp(output, oruby) rescue ServerError => e oruby = t[:error, t[:server, 0, e.class.to_s, e.message, e.backtrace]] self.log.error("<- " + oruby.inspect) self.log.error(e.backtrace.join("\n")) write_berp(output, oruby) rescue Object => e oruby = t[:error, t[:user, 0, e.class.to_s, e.message, e.backtrace]] self.log.error("<- " + oruby.inspect) self.log.error(e.backtrace.join("\n")) write_berp(output, oruby) end elsif iruby.size == 4 && iruby[0] == :cast mod, fun, args = iruby[1..3] self.log.info("-> " + [:cast, mod, fun, args].inspect) begin self.dispatch(mod, fun, args) rescue Object => e # ignore end write_berp(output, t[:noreply]) else self.log.error("-> " + iruby.inspect) oruby = t[:error, t[:server, 0, "Invalid request: #{iruby.inspect}"]] self.log.error("<- " + oruby.inspect) write_berp(output, oruby) end end end end class Ernie::ServerError < StandardError; end class Ernie::Mod attr_accessor :name, :funs def initialize(name) self.name = name self.funs = {} end def fun(name, block) raise TypeError, "block required" if block.nil? self.funs[name] = block end end # Root level calls def logfile(name) Ernie.logfile(name) end def loglevel(level) Ernie.loglevel(level) end at_exit do Ernie.start if Ernie.auto_start end