#!/usr/bin/env ruby
#
# Copyright (c) 2006- Facebook
# Distributed under the Thrift Software License
#
# See accompanying file LICENSE or visit the Thrift site at:
# http://developers.facebook.com/thrift/
#
# Author: Mark Slee <mcslee@facebook.com>
#
require 'thrift/thrift'
class TTransportException < TException
UNKNOWN = 0
NOT_OPEN = 1
ALREADY_OPEN = 2
TIMED_OUT = 3
END_OF_FILE = 4
attr_reader :type
def initialize(type=UNKNOWN, message=nil)
super(message)
@type = type
end
end
# TTransport is basically an abstract class, but isn't raising NotImplementedError
# TODO: Think about if this is the right thing - Kevin Clark - 3/27/08
class TTransport
def isOpen; end
def open; end
def close; end
def read(sz); end
def readAll(sz)
buff = ''
have = 0
while (have < sz)
chunk = read(sz - have)
have += chunk.length
buff << chunk
end
return buff
end
def write(buf); end
def flush; end
end
class TServerTransport
def listen(); nil; end
def accept(); nil; end
def close(); nil; end
end
class TTransportFactory
def getTransport(trans)
return trans
end
end
class TBufferedTransport < TTransport
def initialize(transport)
@transport = transport
@wbuf = ''
@rbuf = ''
end
def isOpen()
return @transport.isOpen()
end
def open()
@transport.open()
end
def close()
@transport.close()
end
DEFAULT_BUFFER = 4096
def read(sz)
ret = @rbuf.slice!(0...sz)
if ret.length == 0
@rbuf = @transport.read([sz, DEFAULT_BUFFER].max)
@rbuf.slice!(0...sz)
else
ret
end
end
def write(buf)
@wbuf << buf
end
def flush()
@transport.write(@wbuf)
@transport.flush()
@wbuf = ''
end
def borrow(requested_length = 0)
return @rbuf if requested_length == 0 and @rbuf.length > 0
if @rbuf.length < DEFAULT_BUFFER
@rbuf << @transport.read([requested_length, DEFAULT_BUFFER].max)
end
if @rbuf.length < requested_length
@rbuf << readAll(requested_length - @rbuf.length)
end
@rbuf
end
def consume!(size)
@rbuf.slice!(0...size)
end
end
class TBufferedTransportFactory < TTransportFactory
def getTransport(transport)
return TBufferedTransport.new(transport)
end
end
class TFramedTransport < TTransport
def initialize(transport, read=true, write=true)
@transport = transport
@rbuf = ''
@wbuf = ''
@read = read
@write = write
end
def isOpen()
return @transport.isOpen
end
def open()
@transport.open
end
def close()
@transport.close
end
def read(sz)
if !@read
return @transport.read(sz)
end
if (sz <= 0)
return ''
end
if (@rbuf.length == 0)
readFrame
end
# return full buf
if (sz > @rbuf.length)
out = @rbuf
@rbuf = ''
return out
end
# Return substr
out = @rbuf.slice(0, sz)
@rbuf = @rbuf.slice(sz, @rbuf.length)
return out
end
def write(buf,sz=nil)
if !@write
return @transport.write(buf);
end
if (sz != nil && sz < buf.length)
buf = buf.slice(0,sz)
end
@wbuf << buf
end
#
# Writes the output buffer to the stream in the format of a 4-byte length
# followed by the actual data.
#
def flush
if !@write
return @transport.flush
end
out = [@wbuf.length].pack('N')
out << @wbuf
@transport.write(out)
@transport.flush
@wbuf = ''
end
private
def readFrame
buf = @transport.readAll(4)
val = buf.unpack('N')
sz = val[0]
@rbuf = @transport.readAll(sz)
end
end
class TFramedTransportFactory < TTransportFactory
def getTransport(transport)
return TFramedTransport.new(transport)
end
end
class TMemoryBuffer < TTransport
def initialize(sz=1024)
@buf = ''
@sz = sz
@wpos = 0
@rpos = 0
end
def isOpen
return 1
end
def open
end
def close
end
def peek
return @rpos < @wpos
end
def getBuffer
return @buf
end
def resetBuffer(new_buf = '')
@buf = new_buf
@sz = new_buf.length
@wpos = new_buf.length
@rpos = 0
end
def available
return @wpos - @rpos
end
def read(len)
avail = available()
return '' if avail == 0
#how much to give
give = len
give = avail if avail < len
ret = @buf.slice(@rpos,give)
@rpos += give;
return ret;
end
def write(wbuf)
@buf << wbuf
@wpos += wbuf.length()
end
def flush
end
# For fast binary protocol access
def borrow(size = nil)
if size.nil?
@buf[@rpos..-1]
else
if size > @buf.length - @rpos
raise EOFError # Memory buffers only get one shot.
else
@buf[@rpos..(@rpos + size)]
end
end
end
def consume!(size)
@rpos += size
end
end
## Very very simple implementation of wrapping two objects, one with a #read
## method and one with a #write method, into a transport for thrift.
##
## Assumes both objects are open, remain open, don't require flushing, etc.
class TIOStreamTransport < TTransport
def initialize(input, output)
@input = input
@output = output
end
def isOpen; true end
def read(sz); @input.read(sz) end
def write(buf); @output.write(buf) end
end