Ruby bindings for the Apache Thrift RPC system. The gem contains the runtime types, transports, protocols, and servers used by generated Ruby code for both clients and services.
gem install thriftbundle install, gem build thrift.gemspec, then install the
resulting thrift-*.gem. The native accelerator is built when the gem is
installed on supported runtimes.The Ruby library does not include the Thrift compiler. Use a compiler built from the root of this repository to generate Ruby bindings:
thrift --gen rb path/to/service.thrift
# with namespaced modules
thrift --gen rb:namespaced --recurse path/to/service.thrift
Generated files are typically written to gen-rb/ and can be required
directly from your application.
$:.push File.expand_path('gen-rb', __dir__)
require 'thrift'
require 'calculator'
socket = Thrift::Socket.new('localhost', 9090)
transport = Thrift::BufferedTransport.new(socket)
protocol = Thrift::BinaryProtocol.new(transport)
client = Calculator::Client.new(protocol)
transport.open
puts client.add(1, 1)
transport.close
$:.push File.expand_path('gen-rb', __dir__)
require 'thrift'
require 'calculator'
class CalculatorHandler
def add(a, b)
a + b
end
end
handler = CalculatorHandler.new
processor = Calculator::Processor.new(handler)
server_transport = Thrift::ServerSocket.new(9090)
transport_factory = Thrift::BufferedTransportFactory.new
protocol_factory = Thrift::BinaryProtocolFactory.new
server = Thrift::ThreadedServer.new(processor, server_transport,
transport_factory, protocol_factory)
server.serve
bundle exec rake spec runs the Ruby specs. It expects a built Thrift
compiler at ../../compiler/cpp/thrift.bundle exec rake test runs the cross-language test suite; it must be
executed from a full Thrift checkout.bundle exec rake build_ext (implicit in the tasks above) compiles the
optional native extension that accelerates protocols and buffers.tutorial/rb/RubyClient.rb and tutorial/rb/RubyServer.rblib/rb/benchmarktest/rb/benchmarks/protocol_benchmark.rblib/rb/speclib/rb/test/fuzztest/rbConnect timeout handling changed for both Thrift::Socket and
Thrift::SSLSocket.
timeout == nil and timeout == 0 now use blocking connect/open semantics.
Older releases already treated nil that way, but treated 0 differently
across operations: read and write used the blocking path, plain TCP
open used a zero-length poll, and TLS open could spin in the handshake
retry loop.Thrift::TransportException::TIMED_OUT. Older releases could report the same
condition as NOT_OPEN, and Thrift::SSLSocket could keep retrying the
handshake after the wait timed out.If your application matched NOT_OPEN for connect timeout handling, update it
to handle TIMED_OUT. If you relied on timeout = 0 meaning immediate failure
or on repeated retries extending the effective timeout during TCP fallback or
TLS handshake, update those call paths before upgrading.
The documented source-build flow now effectively requires Ruby 2.7+.
The committed development bundle no longer resolves on Ruby 2.6
(json-2.18.1 requires ruby version >= 2.7), so building and testing this
library from source should be treated as 2.7+.
Generated structs and unions now consistently raise
Thrift::ProtocolException::INVALID_DATA for invalid payloads such as unset
required fields, invalid enum values, or invalid union state. If your
application or tests matched older exception types or messages, update them.
Regenerated Ruby clients now validate replies more strictly. Mismatched reply
message types, method names, or sequence IDs raise
Thrift::ApplicationException::INVALID_MESSAGE_TYPE,
Thrift::ApplicationException::WRONG_METHOD_NAME, or
Thrift::ApplicationException::BAD_SEQUENCE_ID. If you relied on older,
looser reply handling in servers, proxies, or tests, regenerate and update
those call paths together.
Generated Ruby clients have never been safe to share across concurrent threads. A client tracks pending sequence IDs on a single reply stream, so use one client/transport pair per thread or serialize access yourself.
Treat Thrift::ApplicationException::BAD_SEQUENCE_ID as a correctness bug
that needs immediate attention. It means the client read a reply whose
sequence ID did not match the next pending request, so the connection may
already be out of sync and you may be reading a reply intended for a
different call. The most common cause is sharing one client across threads,
but a buggy proxy or server can also cause it.
Ruby development and CI moved to Ruby 2.4+, but the runtime still claimed
support for older interpreters. Treat Ruby < 2.4 on the 0.13.x line as
best-effort, not guaranteed.
T* constants or legacy require paths such as
TBinaryProtocol -> Thrift::BinaryProtocol.rb:namespaced changes the generated file layout. Flat output from
thrift --gen rb and namespaced output from thrift --gen rb:namespaced
use different require paths, so switch them atomically with regenerated code.
# --gen rb
require 'calculator'
# --gen rb:namespaced
require 'my_namespace/calculator'
0.24.0, treat timeout on Thrift::Socket and
Thrift::SSLSocket as one budget for the whole open path. For
Thrift::SSLSocket, that includes both the TCP connect and the TLS
handshake.0.24.0, handle connect/open timeout expiry as
Thrift::TransportException::TIMED_OUT instead of NOT_OPEN.Thrift::ApplicationException::BAD_SEQUENCE_ID, treat the
connection as out of sync. Close it, create a new client/transport pair, and
investigate the root cause before retrying.thrift --gen rb and thrift --gen rb:namespaced,
regenerate all Ruby output and update require paths in the same change.thrift_native extension changes which implementation you are
running. It replaces
Thrift::Struct, Thrift::Union, and Thrift::CompactProtocol methods
with C implementations in place. Thrift::BinaryProtocol remains available
in pure Ruby, and the C-backed binary protocol is opt-in through
Thrift::BinaryProtocolAcceleratedFactory or
Thrift::BinaryProtocolAccelerated when that class is available.Thrift::NonblockingServer expects framed input. Use
Thrift::FramedTransport with it on the wire.Thrift::SSLSocket
performs a hostname check against the host you pass in.