blob: 0cf2f3b1f1d895202e9a0f8944d6c2b9ba7cb743 [file] [log] [blame]
#
# 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/
#
# package - thrift
# author - T Jake Luciani <jakers@gmail.com>
# author - Mark Slee <mcslee@facebook.com>
#
require 5.6.0;
use strict;
use warnings;
use Thrift;
use Thrift::Transport;
#
# Framed transport. Writes and reads data in chunks that are stamped with
# their length.
#
# @package thrift.transport
# @author Mark Slee <mcslee@facebook.com>
#
package Thrift::FramedTransport;
use base('Thrift::Transport');
sub new
{
my $classname = shift;
my $transport = shift;
my $read = shift || 1;
my $write = shift || 1;
my $self = {
transport => $transport,
read => $read,
write => $write,
wBuf => '',
rBuf => '',
};
return bless($self,$classname);
}
sub isOpen
{
my $self = shift;
return $self->{transport}->isOpen();
}
sub open
{
my $self = shift;
$self->{transport}->open();
}
sub close
{
my $self = shift;
$self->{transport}->close();
}
#
# Reads from the buffer. When more data is required reads another entire
# chunk and serves future reads out of that.
#
# @param int $len How much data
#
sub read
{
my $self = shift;
my $len = shift;
if (!$self->{read}) {
return $self->{transport}->read($len);
}
if (length($self->{rBuf}) == 0) {
$self->_readFrame();
}
# Just return full buff
if ($len > length($self->{rBuf})) {
my $out = $self->{rBuf};
$self->{rBuf} = '';
return $out;
}
# Return substr
my $out = substr($self->{rBuf}, 0, $len);
$self->{rBuf} = substr($self->{rBuf}, $len);
return $out;
}
#
# Reads a chunk of data into the internal read buffer.
# (private)
sub _readFrame
{
my $self = shift;
my $buf = $self->{transport}->readAll(4);
my @val = unpack('N', $buf);
my $sz = $val[0];
$self->{rBuf} = $self->{transport}->readAll($sz);
}
#
# Writes some data to the pending output buffer.
#
# @param string $buf The data
# @param int $len Limit of bytes to write
#
sub write
{
my $self = shift;
my $buf = shift;
my $len = shift;
unless($self->{write}) {
return $self->{transport}->write($buf, $len);
}
if ( defined $len && $len < length($buf)) {
$buf = substr($buf, 0, $len);
}
$self->{wBuf} .= $buf;
}
#
# Writes the output buffer to the stream in the format of a 4-byte length
# followed by the actual data.
#
sub flush
{
my $self = shift;
unless ($self->{write}) {
return $self->{transport}->flush();
}
my $out = pack('N', length($self->{wBuf}));
$out .= $self->{wBuf};
$self->{transport}->write($out);
$self->{transport}->flush();
$self->{wBuf} = '';
}
1;