module PipeRpc class DeepMapper def self.map(object, &mapper) if object.respond_to? :__rpc_server_id__ # first filter out clients so for them subsequent checks are not sent # through the pipe yield object elsif Hash === object object.map{ |k, v| [k, map(v, &mapper)] }.to_h elsif object.respond_to? :map object.map{ |item| map(item, &mapper) } else yield object end end def self.to_transport(object, hub) map(object) do |value| if value.respond_to? :__rpc_server_id__ # first filter out clients so for them subsequent checks are not sent # through the pipe "__rpc_client__#{value.__rpc_server_id__}" elsif value.respond_to? :to_rpc_server_id hub.servers.add(value) unless hub.servers.registered? value.to_rpc_server_id "__rpc_server__#{value.to_rpc_server_id}" else value end end end def self.from_transport(object, hub) map(object) do |value| if value.is_a?(String) and value.start_with?('__rpc_client__') hub.servers[value.sub('__rpc_client__', '').to_sym] elsif value.is_a?(String) and value.start_with?('__rpc_server__') Client.new(value.sub('__rpc_server__', '').to_sym, hub) else value end end end end end