Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 1 | // Licensed to the Apache Software Foundation (ASF) under one |
| 2 | // or more contributor license agreements. See the NOTICE file |
| 3 | // distributed with this work for additional information |
| 4 | // regarding copyright ownership. The ASF licenses this file |
| 5 | // to you under the Apache License, Version 2.0 (the |
| 6 | // "License"); you may not use this file except in compliance |
| 7 | // with the License. You may obtain a copy of the License at |
| 8 | // |
| 9 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | // |
| 11 | // Unless required by applicable law or agreed to in writing, |
| 12 | // software distributed under the License is distributed on an |
| 13 | // "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| 14 | // KIND, either express or implied. See the License for the |
| 15 | // specific language governing permissions and limitations |
| 16 | // under the License. |
| 17 | |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 18 | //! Types used to send and receive bytes over an I/O channel. |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 19 | //! |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 20 | //! The core types are the `TReadTransport`, `TWriteTransport` and the |
| 21 | //! `TIoChannel` traits, through which `TInputProtocol` or |
| 22 | //! `TOutputProtocol` can receive and send primitives over the wire. While |
| 23 | //! `TInputProtocol` and `TOutputProtocol` instances deal with language primitives |
| 24 | //! the types in this module understand only bytes. |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 25 | |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 26 | use std::io; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 27 | use std::io::{Read, Write}; |
| 28 | use std::ops::{Deref, DerefMut}; |
| 29 | |
| 30 | #[cfg(test)] |
| 31 | macro_rules! assert_eq_transport_num_written_bytes { |
Allen George | ef7a189 | 2018-12-16 18:01:37 -0500 | [diff] [blame] | 32 | ($transport:ident, $num_written_bytes:expr) => {{ |
| 33 | assert_eq!($transport.channel.write_bytes().len(), $num_written_bytes); |
| 34 | }}; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 35 | } |
| 36 | |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 37 | #[cfg(test)] |
| 38 | macro_rules! assert_eq_transport_written_bytes { |
Allen George | ef7a189 | 2018-12-16 18:01:37 -0500 | [diff] [blame] | 39 | ($transport:ident, $expected_bytes:ident) => {{ |
| 40 | assert_eq!($transport.channel.write_bytes(), &$expected_bytes); |
| 41 | }}; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 42 | } |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 43 | |
| 44 | mod buffered; |
| 45 | mod framed; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 46 | mod mem; |
Allen George | ef7a189 | 2018-12-16 18:01:37 -0500 | [diff] [blame] | 47 | mod socket; |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 48 | |
Allen George | ef7a189 | 2018-12-16 18:01:37 -0500 | [diff] [blame] | 49 | pub use self::buffered::{ |
| 50 | TBufferedReadTransport, TBufferedReadTransportFactory, TBufferedWriteTransport, |
| 51 | TBufferedWriteTransportFactory, |
| 52 | }; |
| 53 | pub use self::framed::{ |
| 54 | TFramedReadTransport, TFramedReadTransportFactory, TFramedWriteTransport, |
| 55 | TFramedWriteTransportFactory, |
| 56 | }; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 57 | pub use self::mem::TBufferChannel; |
| 58 | pub use self::socket::TTcpChannel; |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 59 | |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 60 | /// Identifies a transport used by a `TInputProtocol` to receive bytes. |
| 61 | pub trait TReadTransport: Read {} |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 62 | |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 63 | /// Helper type used by a server to create `TReadTransport` instances for |
| 64 | /// accepted client connections. |
| 65 | pub trait TReadTransportFactory { |
| 66 | /// Create a `TTransport` that wraps a channel over which bytes are to be read. |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 67 | fn create(&self, channel: Box<dyn Read + Send>) -> Box<dyn TReadTransport + Send>; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 68 | } |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 69 | |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 70 | /// Identifies a transport used by `TOutputProtocol` to send bytes. |
| 71 | pub trait TWriteTransport: Write {} |
| 72 | |
| 73 | /// Helper type used by a server to create `TWriteTransport` instances for |
| 74 | /// accepted client connections. |
| 75 | pub trait TWriteTransportFactory { |
| 76 | /// Create a `TTransport` that wraps a channel over which bytes are to be sent. |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 77 | fn create(&self, channel: Box<dyn Write + Send>) -> Box<dyn TWriteTransport + Send>; |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 78 | } |
| 79 | |
Allen George | ef7a189 | 2018-12-16 18:01:37 -0500 | [diff] [blame] | 80 | impl<T> TReadTransport for T where T: Read {} |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 81 | |
Allen George | ef7a189 | 2018-12-16 18:01:37 -0500 | [diff] [blame] | 82 | impl<T> TWriteTransport for T where T: Write {} |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 83 | |
| 84 | // FIXME: implement the Debug trait for boxed transports |
| 85 | |
| 86 | impl<T> TReadTransportFactory for Box<T> |
| 87 | where |
| 88 | T: TReadTransportFactory + ?Sized, |
| 89 | { |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 90 | fn create(&self, channel: Box<dyn Read + Send>) -> Box<dyn TReadTransport + Send> { |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 91 | (**self).create(channel) |
| 92 | } |
| 93 | } |
| 94 | |
| 95 | impl<T> TWriteTransportFactory for Box<T> |
| 96 | where |
| 97 | T: TWriteTransportFactory + ?Sized, |
| 98 | { |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 99 | fn create(&self, channel: Box<dyn Write + Send>) -> Box<dyn TWriteTransport + Send> { |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 100 | (**self).create(channel) |
| 101 | } |
| 102 | } |
| 103 | |
| 104 | /// Identifies a splittable bidirectional I/O channel used to send and receive bytes. |
| 105 | pub trait TIoChannel: Read + Write { |
| 106 | /// Split the channel into a readable half and a writable half, where the |
| 107 | /// readable half implements `io::Read` and the writable half implements |
| 108 | /// `io::Write`. Returns `None` if the channel was not initialized, or if it |
| 109 | /// cannot be split safely. |
| 110 | /// |
| 111 | /// Returned halves may share the underlying OS channel or buffer resources. |
| 112 | /// Implementations **should ensure** that these two halves can be safely |
| 113 | /// used independently by concurrent threads. |
Allen George | b0d1413 | 2020-03-29 11:48:55 -0400 | [diff] [blame] | 114 | fn split(self) -> crate::Result<(crate::transport::ReadHalf<Self>, crate::transport::WriteHalf<Self>)> |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 115 | where |
| 116 | Self: Sized; |
| 117 | } |
| 118 | |
| 119 | /// The readable half of an object returned from `TIoChannel::split`. |
| 120 | #[derive(Debug)] |
| 121 | pub struct ReadHalf<C> |
| 122 | where |
| 123 | C: Read, |
| 124 | { |
| 125 | handle: C, |
| 126 | } |
| 127 | |
| 128 | /// The writable half of an object returned from `TIoChannel::split`. |
| 129 | #[derive(Debug)] |
| 130 | pub struct WriteHalf<C> |
| 131 | where |
| 132 | C: Write, |
| 133 | { |
| 134 | handle: C, |
| 135 | } |
| 136 | |
Jake W | 24918ab | 2018-11-12 12:43:04 +0800 | [diff] [blame] | 137 | impl<C> ReadHalf<C> |
| 138 | where |
| 139 | C: Read, |
| 140 | { |
| 141 | /// Create a `ReadHalf` associated with readable `handle` |
| 142 | pub fn new(handle: C) -> ReadHalf<C> { |
| 143 | ReadHalf { handle } |
| 144 | } |
| 145 | } |
| 146 | |
| 147 | impl<C> WriteHalf<C> |
| 148 | where |
| 149 | C: Write, |
| 150 | { |
| 151 | /// Create a `WriteHalf` associated with writable `handle` |
| 152 | pub fn new(handle: C) -> WriteHalf<C> { |
| 153 | WriteHalf { handle } |
| 154 | } |
| 155 | } |
| 156 | |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 157 | impl<C> Read for ReadHalf<C> |
| 158 | where |
| 159 | C: Read, |
| 160 | { |
| 161 | fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> { |
| 162 | self.handle.read(buf) |
| 163 | } |
| 164 | } |
| 165 | |
| 166 | impl<C> Write for WriteHalf<C> |
| 167 | where |
| 168 | C: Write, |
| 169 | { |
| 170 | fn write(&mut self, buf: &[u8]) -> io::Result<usize> { |
| 171 | self.handle.write(buf) |
| 172 | } |
| 173 | |
| 174 | fn flush(&mut self) -> io::Result<()> { |
| 175 | self.handle.flush() |
| 176 | } |
| 177 | } |
| 178 | |
| 179 | impl<C> Deref for ReadHalf<C> |
| 180 | where |
| 181 | C: Read, |
| 182 | { |
| 183 | type Target = C; |
| 184 | |
| 185 | fn deref(&self) -> &Self::Target { |
| 186 | &self.handle |
| 187 | } |
| 188 | } |
| 189 | |
| 190 | impl<C> DerefMut for ReadHalf<C> |
| 191 | where |
| 192 | C: Read, |
| 193 | { |
| 194 | fn deref_mut(&mut self) -> &mut C { |
| 195 | &mut self.handle |
| 196 | } |
| 197 | } |
| 198 | |
| 199 | impl<C> Deref for WriteHalf<C> |
| 200 | where |
| 201 | C: Write, |
| 202 | { |
| 203 | type Target = C; |
| 204 | |
| 205 | fn deref(&self) -> &Self::Target { |
| 206 | &self.handle |
| 207 | } |
| 208 | } |
| 209 | |
| 210 | impl<C> DerefMut for WriteHalf<C> |
| 211 | where |
| 212 | C: Write, |
| 213 | { |
| 214 | fn deref_mut(&mut self) -> &mut C { |
| 215 | &mut self.handle |
| 216 | } |
| 217 | } |
| 218 | |
| 219 | #[cfg(test)] |
| 220 | mod tests { |
| 221 | |
| 222 | use std::io::Cursor; |
| 223 | |
| 224 | use super::*; |
| 225 | |
| 226 | #[test] |
| 227 | fn must_create_usable_read_channel_from_concrete_read_type() { |
| 228 | let r = Cursor::new([0, 1, 2]); |
| 229 | let _ = TBufferedReadTransport::new(r); |
| 230 | } |
| 231 | |
| 232 | #[test] |
| 233 | fn must_create_usable_read_channel_from_boxed_read() { |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 234 | let r: Box<dyn Read> = Box::new(Cursor::new([0, 1, 2])); |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 235 | let _ = TBufferedReadTransport::new(r); |
| 236 | } |
| 237 | |
| 238 | #[test] |
| 239 | fn must_create_usable_write_channel_from_concrete_write_type() { |
| 240 | let w = vec![0u8; 10]; |
| 241 | let _ = TBufferedWriteTransport::new(w); |
| 242 | } |
| 243 | |
| 244 | #[test] |
| 245 | fn must_create_usable_write_channel_from_boxed_write() { |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 246 | let w: Box<dyn Write> = Box::new(vec![0u8; 10]); |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 247 | let _ = TBufferedWriteTransport::new(w); |
| 248 | } |
| 249 | |
| 250 | #[test] |
| 251 | fn must_create_usable_read_transport_from_concrete_read_transport() { |
| 252 | let r = Cursor::new([0, 1, 2]); |
| 253 | let mut t = TBufferedReadTransport::new(r); |
| 254 | takes_read_transport(&mut t) |
| 255 | } |
| 256 | |
| 257 | #[test] |
| 258 | fn must_create_usable_read_transport_from_boxed_read() { |
| 259 | let r = Cursor::new([0, 1, 2]); |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 260 | let mut t: Box<dyn TReadTransport> = Box::new(TBufferedReadTransport::new(r)); |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 261 | takes_read_transport(&mut t) |
| 262 | } |
| 263 | |
| 264 | #[test] |
| 265 | fn must_create_usable_write_transport_from_concrete_write_transport() { |
| 266 | let w = vec![0u8; 10]; |
| 267 | let mut t = TBufferedWriteTransport::new(w); |
| 268 | takes_write_transport(&mut t) |
| 269 | } |
| 270 | |
| 271 | #[test] |
| 272 | fn must_create_usable_write_transport_from_boxed_write() { |
| 273 | let w = vec![0u8; 10]; |
Danny Browning | 77d96c1 | 2019-08-21 13:41:07 -0600 | [diff] [blame] | 274 | let mut t: Box<dyn TWriteTransport> = Box::new(TBufferedWriteTransport::new(w)); |
Allen George | 0e22c36 | 2017-01-30 07:15:00 -0500 | [diff] [blame] | 275 | takes_write_transport(&mut t) |
| 276 | } |
| 277 | |
| 278 | fn takes_read_transport<R>(t: &mut R) |
| 279 | where |
| 280 | R: TReadTransport, |
| 281 | { |
| 282 | t.bytes(); |
| 283 | } |
| 284 | |
| 285 | fn takes_write_transport<W>(t: &mut W) |
| 286 | where |
| 287 | W: TWriteTransport, |
| 288 | { |
| 289 | t.flush().unwrap(); |
| 290 | } |
Allen George | 8b96bfb | 2016-11-02 08:01:08 -0400 | [diff] [blame] | 291 | } |