blob: b230d636388b15da98060d7ccaac91127acaf366 [file] [log] [blame]
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
//! Types used to send and receive primitives between a Thrift client and server.
//!
//! # Examples
//!
//! Create and use a `TOutputProtocol`.
//!
//! ```no_run
//! use std::cell::RefCell;
//! use std::rc::Rc;
//! use thrift::protocol::{TBinaryOutputProtocol, TFieldIdentifier, TOutputProtocol, TType};
//! use thrift::transport::{TTcpTransport, TTransport};
//!
//! // create the I/O channel
//! let mut transport = TTcpTransport::new();
//! transport.open("127.0.0.1:9090").unwrap();
//! let transport = Rc::new(RefCell::new(Box::new(transport) as Box<TTransport>));
//!
//! // create the protocol to encode types into bytes
//! let mut o_prot = TBinaryOutputProtocol::new(transport.clone(), true);
//!
//! // write types
//! o_prot.write_field_begin(&TFieldIdentifier::new("string_thing", TType::String, 1)).unwrap();
//! o_prot.write_string("foo").unwrap();
//! o_prot.write_field_end().unwrap();
//! ```
//!
//! Create and use a `TInputProtocol`.
//!
//! ```no_run
//! use std::cell::RefCell;
//! use std::rc::Rc;
//! use thrift::protocol::{TBinaryInputProtocol, TInputProtocol};
//! use thrift::transport::{TTcpTransport, TTransport};
//!
//! // create the I/O channel
//! let mut transport = TTcpTransport::new();
//! transport.open("127.0.0.1:9090").unwrap();
//! let transport = Rc::new(RefCell::new(Box::new(transport) as Box<TTransport>));
//!
//! // create the protocol to decode bytes into types
//! let mut i_prot = TBinaryInputProtocol::new(transport.clone(), true);
//!
//! // read types from the wire
//! let field_identifier = i_prot.read_field_begin().unwrap();
//! let field_contents = i_prot.read_string().unwrap();
//! let field_end = i_prot.read_field_end().unwrap();
//! ```
use std::cell::RefCell;
use std::fmt;
use std::fmt::{Display, Formatter};
use std::convert::From;
use std::rc::Rc;
use try_from::TryFrom;
use ::{ProtocolError, ProtocolErrorKind};
use ::transport::TTransport;
mod binary;
mod compact;
mod multiplexed;
mod stored;
pub use self::binary::{TBinaryInputProtocol, TBinaryInputProtocolFactory, TBinaryOutputProtocol,
TBinaryOutputProtocolFactory};
pub use self::compact::{TCompactInputProtocol, TCompactInputProtocolFactory,
TCompactOutputProtocol, TCompactOutputProtocolFactory};
pub use self::multiplexed::TMultiplexedOutputProtocol;
pub use self::stored::TStoredInputProtocol;
// Default maximum depth to which `TInputProtocol::skip` will skip a Thrift
// field. A default is necessary because Thrift structs or collections may
// contain nested structs and collections, which could result in indefinite
// recursion.
const MAXIMUM_SKIP_DEPTH: i8 = 64;
/// Converts a stream of bytes into Thrift identifiers, primitives,
/// containers, or structs.
///
/// This trait does not deal with higher-level Thrift concepts like structs or
/// exceptions - only with primitives and message or container boundaries. Once
/// bytes are read they are deserialized and an identifier (for example
/// `TMessageIdentifier`) or a primitive is returned.
///
/// All methods return a `thrift::Result`. If an `Err` is returned the protocol
/// instance and its underlying transport should be terminated.
///
/// # Examples
///
/// Create and use a `TInputProtocol`
///
/// ```no_run
/// use std::cell::RefCell;
/// use std::rc::Rc;
/// use thrift::protocol::{TBinaryInputProtocol, TInputProtocol};
/// use thrift::transport::{TTcpTransport, TTransport};
///
/// let mut transport = TTcpTransport::new();
/// transport.open("127.0.0.1:9090").unwrap();
/// let transport = Rc::new(RefCell::new(Box::new(transport) as Box<TTransport>));
///
/// let mut i_prot = TBinaryInputProtocol::new(transport.clone(), true);
///
/// let field_identifier = i_prot.read_field_begin().unwrap();
/// let field_contents = i_prot.read_string().unwrap();
/// let field_end = i_prot.read_field_end().unwrap();
/// ```
pub trait TInputProtocol {
/// Read the beginning of a Thrift message.
fn read_message_begin(&mut self) -> ::Result<TMessageIdentifier>;
/// Read the end of a Thrift message.
fn read_message_end(&mut self) -> ::Result<()>;
/// Read the beginning of a Thrift struct.
fn read_struct_begin(&mut self) -> ::Result<Option<TStructIdentifier>>;
/// Read the end of a Thrift struct.
fn read_struct_end(&mut self) -> ::Result<()>;
/// Read the beginning of a Thrift struct field.
fn read_field_begin(&mut self) -> ::Result<TFieldIdentifier>;
/// Read the end of a Thrift struct field.
fn read_field_end(&mut self) -> ::Result<()>;
/// Read a bool.
fn read_bool(&mut self) -> ::Result<bool>;
/// Read a fixed-length byte array.
fn read_bytes(&mut self) -> ::Result<Vec<u8>>;
/// Read a word.
fn read_i8(&mut self) -> ::Result<i8>;
/// Read a 16-bit signed integer.
fn read_i16(&mut self) -> ::Result<i16>;
/// Read a 32-bit signed integer.
fn read_i32(&mut self) -> ::Result<i32>;
/// Read a 64-bit signed integer.
fn read_i64(&mut self) -> ::Result<i64>;
/// Read a 64-bit float.
fn read_double(&mut self) -> ::Result<f64>;
/// Read a fixed-length string (not null terminated).
fn read_string(&mut self) -> ::Result<String>;
/// Read the beginning of a list.
fn read_list_begin(&mut self) -> ::Result<TListIdentifier>;
/// Read the end of a list.
fn read_list_end(&mut self) -> ::Result<()>;
/// Read the beginning of a set.
fn read_set_begin(&mut self) -> ::Result<TSetIdentifier>;
/// Read the end of a set.
fn read_set_end(&mut self) -> ::Result<()>;
/// Read the beginning of a map.
fn read_map_begin(&mut self) -> ::Result<TMapIdentifier>;
/// Read the end of a map.
fn read_map_end(&mut self) -> ::Result<()>;
/// Skip a field with type `field_type` recursively until the default
/// maximum skip depth is reached.
fn skip(&mut self, field_type: TType) -> ::Result<()> {
self.skip_till_depth(field_type, MAXIMUM_SKIP_DEPTH)
}
/// Skip a field with type `field_type` recursively up to `depth` levels.
fn skip_till_depth(&mut self, field_type: TType, depth: i8) -> ::Result<()> {
if depth == 0 {
return Err(::Error::Protocol(ProtocolError {
kind: ProtocolErrorKind::DepthLimit,
message: format!("cannot parse past {:?}", field_type),
}));
}
match field_type {
TType::Bool => self.read_bool().map(|_| ()),
TType::I08 => self.read_i8().map(|_| ()),
TType::I16 => self.read_i16().map(|_| ()),
TType::I32 => self.read_i32().map(|_| ()),
TType::I64 => self.read_i64().map(|_| ()),
TType::Double => self.read_double().map(|_| ()),
TType::String => self.read_string().map(|_| ()),
TType::Struct => {
self.read_struct_begin()?;
loop {
let field_ident = self.read_field_begin()?;
if field_ident.field_type == TType::Stop {
break;
}
self.skip_till_depth(field_ident.field_type, depth - 1)?;
}
self.read_struct_end()
}
TType::List => {
let list_ident = self.read_list_begin()?;
for _ in 0..list_ident.size {
self.skip_till_depth(list_ident.element_type, depth - 1)?;
}
self.read_list_end()
}
TType::Set => {
let set_ident = self.read_set_begin()?;
for _ in 0..set_ident.size {
self.skip_till_depth(set_ident.element_type, depth - 1)?;
}
self.read_set_end()
}
TType::Map => {
let map_ident = self.read_map_begin()?;
for _ in 0..map_ident.size {
let key_type = map_ident.key_type
.expect("non-zero sized map should contain key type");
let val_type = map_ident.value_type
.expect("non-zero sized map should contain value type");
self.skip_till_depth(key_type, depth - 1)?;
self.skip_till_depth(val_type, depth - 1)?;
}
self.read_map_end()
}
u => {
Err(::Error::Protocol(ProtocolError {
kind: ProtocolErrorKind::Unknown,
message: format!("cannot skip field type {:?}", &u),
}))
}
}
}
// utility (DO NOT USE IN GENERATED CODE!!!!)
//
/// Read an unsigned byte.
///
/// This method should **never** be used in generated code.
fn read_byte(&mut self) -> ::Result<u8>;
}
/// Converts Thrift identifiers, primitives, containers or structs into a
/// stream of bytes.
///
/// This trait does not deal with higher-level Thrift concepts like structs or
/// exceptions - only with primitives and message or container boundaries.
/// Write methods take an identifier (for example, `TMessageIdentifier`) or a
/// primitive. Any or all of the fields in an identifier may be omitted when
/// writing to the transport. Write methods may even be noops. All of this is
/// transparent to the caller; as long as a matching `TInputProtocol`
/// implementation is used, received messages will be decoded correctly.
///
/// All methods return a `thrift::Result`. If an `Err` is returned the protocol
/// instance and its underlying transport should be terminated.
///
/// # Examples
///
/// Create and use a `TOutputProtocol`
///
/// ```no_run
/// use std::cell::RefCell;
/// use std::rc::Rc;
/// use thrift::protocol::{TBinaryOutputProtocol, TFieldIdentifier, TOutputProtocol, TType};
/// use thrift::transport::{TTcpTransport, TTransport};
///
/// let mut transport = TTcpTransport::new();
/// transport.open("127.0.0.1:9090").unwrap();
/// let transport = Rc::new(RefCell::new(Box::new(transport) as Box<TTransport>));
///
/// let mut o_prot = TBinaryOutputProtocol::new(transport.clone(), true);
///
/// o_prot.write_field_begin(&TFieldIdentifier::new("string_thing", TType::String, 1)).unwrap();
/// o_prot.write_string("foo").unwrap();
/// o_prot.write_field_end().unwrap();
/// ```
pub trait TOutputProtocol {
/// Write the beginning of a Thrift message.
fn write_message_begin(&mut self, identifier: &TMessageIdentifier) -> ::Result<()>;
/// Write the end of a Thrift message.
fn write_message_end(&mut self) -> ::Result<()>;
/// Write the beginning of a Thrift struct.
fn write_struct_begin(&mut self, identifier: &TStructIdentifier) -> ::Result<()>;
/// Write the end of a Thrift struct.
fn write_struct_end(&mut self) -> ::Result<()>;
/// Write the beginning of a Thrift field.
fn write_field_begin(&mut self, identifier: &TFieldIdentifier) -> ::Result<()>;
/// Write the end of a Thrift field.
fn write_field_end(&mut self) -> ::Result<()>;
/// Write a STOP field indicating that all the fields in a struct have been
/// written.
fn write_field_stop(&mut self) -> ::Result<()>;
/// Write a bool.
fn write_bool(&mut self, b: bool) -> ::Result<()>;
/// Write a fixed-length byte array.
fn write_bytes(&mut self, b: &[u8]) -> ::Result<()>;
/// Write an 8-bit signed integer.
fn write_i8(&mut self, i: i8) -> ::Result<()>;
/// Write a 16-bit signed integer.
fn write_i16(&mut self, i: i16) -> ::Result<()>;
/// Write a 32-bit signed integer.
fn write_i32(&mut self, i: i32) -> ::Result<()>;
/// Write a 64-bit signed integer.
fn write_i64(&mut self, i: i64) -> ::Result<()>;
/// Write a 64-bit float.
fn write_double(&mut self, d: f64) -> ::Result<()>;
/// Write a fixed-length string.
fn write_string(&mut self, s: &str) -> ::Result<()>;
/// Write the beginning of a list.
fn write_list_begin(&mut self, identifier: &TListIdentifier) -> ::Result<()>;
/// Write the end of a list.
fn write_list_end(&mut self) -> ::Result<()>;
/// Write the beginning of a set.
fn write_set_begin(&mut self, identifier: &TSetIdentifier) -> ::Result<()>;
/// Write the end of a set.
fn write_set_end(&mut self) -> ::Result<()>;
/// Write the beginning of a map.
fn write_map_begin(&mut self, identifier: &TMapIdentifier) -> ::Result<()>;
/// Write the end of a map.
fn write_map_end(&mut self) -> ::Result<()>;
/// Flush buffered bytes to the underlying transport.
fn flush(&mut self) -> ::Result<()>;
// utility (DO NOT USE IN GENERATED CODE!!!!)
//
/// Write an unsigned byte.
///
/// This method should **never** be used in generated code.
fn write_byte(&mut self, b: u8) -> ::Result<()>; // FIXME: REMOVE
}
/// Helper type used by servers to create `TInputProtocol` instances for
/// accepted client connections.
///
/// # Examples
///
/// Create a `TInputProtocolFactory` and use it to create a `TInputProtocol`.
///
/// ```no_run
/// use std::cell::RefCell;
/// use std::rc::Rc;
/// use thrift::protocol::{TBinaryInputProtocolFactory, TInputProtocolFactory};
/// use thrift::transport::{TTcpTransport, TTransport};
///
/// let mut transport = TTcpTransport::new();
/// transport.open("127.0.0.1:9090").unwrap();
/// let transport = Rc::new(RefCell::new(Box::new(transport) as Box<TTransport>));
///
/// let mut i_proto_factory = TBinaryInputProtocolFactory::new();
/// let i_prot = i_proto_factory.create(transport);
/// ```
pub trait TInputProtocolFactory {
/// Create a `TInputProtocol` that reads bytes from `transport`.
fn create(&mut self, transport: Rc<RefCell<Box<TTransport>>>) -> Box<TInputProtocol>;
}
/// Helper type used by servers to create `TOutputProtocol` instances for
/// accepted client connections.
///
/// # Examples
///
/// Create a `TOutputProtocolFactory` and use it to create a `TOutputProtocol`.
///
/// ```no_run
/// use std::cell::RefCell;
/// use std::rc::Rc;
/// use thrift::protocol::{TBinaryOutputProtocolFactory, TOutputProtocolFactory};
/// use thrift::transport::{TTcpTransport, TTransport};
///
/// let mut transport = TTcpTransport::new();
/// transport.open("127.0.0.1:9090").unwrap();
/// let transport = Rc::new(RefCell::new(Box::new(transport) as Box<TTransport>));
///
/// let mut o_proto_factory = TBinaryOutputProtocolFactory::new();
/// let o_prot = o_proto_factory.create(transport);
/// ```
pub trait TOutputProtocolFactory {
/// Create a `TOutputProtocol` that writes bytes to `transport`.
fn create(&mut self, transport: Rc<RefCell<Box<TTransport>>>) -> Box<TOutputProtocol>;
}
/// Thrift message identifier.
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct TMessageIdentifier {
/// Service call the message is associated with.
pub name: String,
/// Message type.
pub message_type: TMessageType,
/// Ordered sequence number identifying the message.
pub sequence_number: i32,
}
impl TMessageIdentifier {
/// Create a `TMessageIdentifier` for a Thrift service-call named `name`
/// with message type `message_type` and sequence number `sequence_number`.
pub fn new<S: Into<String>>(name: S,
message_type: TMessageType,
sequence_number: i32)
-> TMessageIdentifier {
TMessageIdentifier {
name: name.into(),
message_type: message_type,
sequence_number: sequence_number,
}
}
}
/// Thrift struct identifier.
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct TStructIdentifier {
/// Name of the encoded Thrift struct.
pub name: String,
}
impl TStructIdentifier {
/// Create a `TStructIdentifier` for a struct named `name`.
pub fn new<S: Into<String>>(name: S) -> TStructIdentifier {
TStructIdentifier { name: name.into() }
}
}
/// Thrift field identifier.
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct TFieldIdentifier {
/// Name of the Thrift field.
///
/// `None` if it's not sent over the wire.
pub name: Option<String>,
/// Field type.
///
/// This may be a primitive, container, or a struct.
pub field_type: TType,
/// Thrift field id.
///
/// `None` only if `field_type` is `TType::Stop`.
pub id: Option<i16>,
}
impl TFieldIdentifier {
/// Create a `TFieldIdentifier` for a field named `name` with type
/// `field_type` and field id `id`.
///
/// `id` should be `None` if `field_type` is `TType::Stop`.
pub fn new<N, S, I>(name: N, field_type: TType, id: I) -> TFieldIdentifier
where N: Into<Option<S>>,
S: Into<String>,
I: Into<Option<i16>>
{
TFieldIdentifier {
name: name.into().map(|n| n.into()),
field_type: field_type,
id: id.into(),
}
}
}
/// Thrift list identifier.
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct TListIdentifier {
/// Type of the elements in the list.
pub element_type: TType,
/// Number of elements in the list.
pub size: i32,
}
impl TListIdentifier {
/// Create a `TListIdentifier` for a list with `size` elements of type
/// `element_type`.
pub fn new(element_type: TType, size: i32) -> TListIdentifier {
TListIdentifier {
element_type: element_type,
size: size,
}
}
}
/// Thrift set identifier.
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct TSetIdentifier {
/// Type of the elements in the set.
pub element_type: TType,
/// Number of elements in the set.
pub size: i32,
}
impl TSetIdentifier {
/// Create a `TSetIdentifier` for a set with `size` elements of type
/// `element_type`.
pub fn new(element_type: TType, size: i32) -> TSetIdentifier {
TSetIdentifier {
element_type: element_type,
size: size,
}
}
}
/// Thrift map identifier.
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct TMapIdentifier {
/// Map key type.
pub key_type: Option<TType>,
/// Map value type.
pub value_type: Option<TType>,
/// Number of entries in the map.
pub size: i32,
}
impl TMapIdentifier {
/// Create a `TMapIdentifier` for a map with `size` entries of type
/// `key_type -> value_type`.
pub fn new<K, V>(key_type: K, value_type: V, size: i32) -> TMapIdentifier
where K: Into<Option<TType>>,
V: Into<Option<TType>>
{
TMapIdentifier {
key_type: key_type.into(),
value_type: value_type.into(),
size: size,
}
}
}
/// Thrift message types.
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum TMessageType {
/// Service-call request.
Call,
/// Service-call response.
Reply,
/// Unexpected error in the remote service.
Exception,
/// One-way service-call request (no response is expected).
OneWay,
}
impl Display for TMessageType {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
match *self {
TMessageType::Call => write!(f, "Call"),
TMessageType::Reply => write!(f, "Reply"),
TMessageType::Exception => write!(f, "Exception"),
TMessageType::OneWay => write!(f, "OneWay"),
}
}
}
impl From<TMessageType> for u8 {
fn from(message_type: TMessageType) -> Self {
match message_type {
TMessageType::Call => 0x01,
TMessageType::Reply => 0x02,
TMessageType::Exception => 0x03,
TMessageType::OneWay => 0x04,
}
}
}
impl TryFrom<u8> for TMessageType {
type Err = ::Error;
fn try_from(b: u8) -> ::Result<Self> {
match b {
0x01 => Ok(TMessageType::Call),
0x02 => Ok(TMessageType::Reply),
0x03 => Ok(TMessageType::Exception),
0x04 => Ok(TMessageType::OneWay),
unkn => {
Err(::Error::Protocol(ProtocolError {
kind: ProtocolErrorKind::InvalidData,
message: format!("cannot convert {} to TMessageType", unkn),
}))
}
}
}
}
/// Thrift struct-field types.
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum TType {
/// Indicates that there are no more serialized fields in this Thrift struct.
Stop,
/// Void (`()`) field.
Void,
/// Boolean.
Bool,
/// Signed 8-bit int.
I08,
/// Double-precision number.
Double,
/// Signed 16-bit int.
I16,
/// Signed 32-bit int.
I32,
/// Signed 64-bit int.
I64,
/// UTF-8 string.
String,
/// UTF-7 string. *Unsupported*.
Utf7,
/// Thrift struct.
Struct,
/// Map.
Map,
/// Set.
Set,
/// List.
List,
/// UTF-8 string.
Utf8,
/// UTF-16 string. *Unsupported*.
Utf16,
}
impl Display for TType {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
match *self {
TType::Stop => write!(f, "STOP"),
TType::Void => write!(f, "void"),
TType::Bool => write!(f, "bool"),
TType::I08 => write!(f, "i08"),
TType::Double => write!(f, "double"),
TType::I16 => write!(f, "i16"),
TType::I32 => write!(f, "i32"),
TType::I64 => write!(f, "i64"),
TType::String => write!(f, "string"),
TType::Utf7 => write!(f, "UTF7"),
TType::Struct => write!(f, "struct"),
TType::Map => write!(f, "map"),
TType::Set => write!(f, "set"),
TType::List => write!(f, "list"),
TType::Utf8 => write!(f, "UTF8"),
TType::Utf16 => write!(f, "UTF16"),
}
}
}
/// Compare the expected message sequence number `expected` with the received
/// message sequence number `actual`.
///
/// Return `()` if `actual == expected`, `Err` otherwise.
pub fn verify_expected_sequence_number(expected: i32, actual: i32) -> ::Result<()> {
if expected == actual {
Ok(())
} else {
Err(::Error::Application(::ApplicationError {
kind: ::ApplicationErrorKind::BadSequenceId,
message: format!("expected {} got {}", expected, actual),
}))
}
}
/// Compare the expected service-call name `expected` with the received
/// service-call name `actual`.
///
/// Return `()` if `actual == expected`, `Err` otherwise.
pub fn verify_expected_service_call(expected: &str, actual: &str) -> ::Result<()> {
if expected == actual {
Ok(())
} else {
Err(::Error::Application(::ApplicationError {
kind: ::ApplicationErrorKind::WrongMethodName,
message: format!("expected {} got {}", expected, actual),
}))
}
}
/// Compare the expected message type `expected` with the received message type
/// `actual`.
///
/// Return `()` if `actual == expected`, `Err` otherwise.
pub fn verify_expected_message_type(expected: TMessageType, actual: TMessageType) -> ::Result<()> {
if expected == actual {
Ok(())
} else {
Err(::Error::Application(::ApplicationError {
kind: ::ApplicationErrorKind::InvalidMessageType,
message: format!("expected {} got {}", expected, actual),
}))
}
}
/// Check if a required Thrift struct field exists.
///
/// Return `()` if it does, `Err` otherwise.
pub fn verify_required_field_exists<T>(field_name: &str, field: &Option<T>) -> ::Result<()> {
match *field {
Some(_) => Ok(()),
None => {
Err(::Error::Protocol(::ProtocolError {
kind: ::ProtocolErrorKind::Unknown,
message: format!("missing required field {}", field_name),
}))
}
}
}
/// Extract the field id from a Thrift field identifier.
///
/// `field_ident` must *not* have `TFieldIdentifier.field_type` of type `TType::Stop`.
///
/// Return `TFieldIdentifier.id` if an id exists, `Err` otherwise.
pub fn field_id(field_ident: &TFieldIdentifier) -> ::Result<i16> {
field_ident.id.ok_or_else(|| {
::Error::Protocol(::ProtocolError {
kind: ::ProtocolErrorKind::Unknown,
message: format!("missing field in in {:?}", field_ident),
})
})
}