| use crate::misc::remaining_capacity_as_slice_mut; |
| use crate::misc::remove_lifetime_mut; |
| use crate::varint; |
| use crate::wire_format; |
| use crate::zigzag::encode_zig_zag_32; |
| use crate::zigzag::encode_zig_zag_64; |
| use crate::Message; |
| use crate::ProtobufEnum; |
| use crate::ProtobufResult; |
| use crate::UnknownFields; |
| use crate::UnknownValueRef; |
| use std::io; |
| use std::io::Write; |
| use std::mem; |
| |
| /// Equal to the default buffer size of `BufWriter`, so when |
| /// `CodedOutputStream` wraps `BufWriter`, it often skips double buffering. |
| const OUTPUT_STREAM_BUFFER_SIZE: usize = 8 * 1024; |
| |
| #[doc(hidden)] |
| pub trait WithCodedOutputStream { |
| fn with_coded_output_stream<T, F>(self, cb: F) -> ProtobufResult<T> |
| where |
| F: FnOnce(&mut CodedOutputStream) -> ProtobufResult<T>; |
| } |
| |
| impl<'a> WithCodedOutputStream for &'a mut (dyn Write + 'a) { |
| fn with_coded_output_stream<T, F>(self, cb: F) -> ProtobufResult<T> |
| where |
| F: FnOnce(&mut CodedOutputStream) -> ProtobufResult<T>, |
| { |
| let mut os = CodedOutputStream::new(self); |
| let r = cb(&mut os)?; |
| os.flush()?; |
| Ok(r) |
| } |
| } |
| |
| impl<'a> WithCodedOutputStream for &'a mut Vec<u8> { |
| fn with_coded_output_stream<T, F>(mut self, cb: F) -> ProtobufResult<T> |
| where |
| F: FnOnce(&mut CodedOutputStream) -> ProtobufResult<T>, |
| { |
| let mut os = CodedOutputStream::vec(&mut self); |
| let r = cb(&mut os)?; |
| os.flush()?; |
| Ok(r) |
| } |
| } |
| |
| #[doc(hidden)] |
| pub fn with_coded_output_stream_to_bytes<F>(cb: F) -> ProtobufResult<Vec<u8>> |
| where |
| F: FnOnce(&mut CodedOutputStream) -> ProtobufResult<()>, |
| { |
| let mut v = Vec::new(); |
| v.with_coded_output_stream(cb)?; |
| Ok(v) |
| } |
| |
| enum OutputTarget<'a> { |
| Write(&'a mut dyn Write, Vec<u8>), |
| Vec(&'a mut Vec<u8>), |
| Bytes, |
| } |
| |
| /// Buffered write with handy utilities |
| pub struct CodedOutputStream<'a> { |
| target: OutputTarget<'a>, |
| // alias to buf from target |
| buffer: &'a mut [u8], |
| // within buffer |
| position: usize, |
| } |
| |
| impl<'a> CodedOutputStream<'a> { |
| /// Construct from given `Write`. |
| /// |
| /// `CodedOutputStream` is buffered even if `Write` is not |
| pub fn new(writer: &'a mut dyn Write) -> CodedOutputStream<'a> { |
| let buffer_len = OUTPUT_STREAM_BUFFER_SIZE; |
| |
| let mut buffer_storage = Vec::with_capacity(buffer_len); |
| unsafe { |
| buffer_storage.set_len(buffer_len); |
| } |
| |
| let buffer = unsafe { remove_lifetime_mut(&mut buffer_storage as &mut [u8]) }; |
| |
| CodedOutputStream { |
| target: OutputTarget::Write(writer, buffer_storage), |
| buffer: buffer, |
| position: 0, |
| } |
| } |
| |
| /// `CodedOutputStream` which writes directly to bytes. |
| /// |
| /// Attempt to write more than bytes capacity results in error. |
| pub fn bytes(bytes: &'a mut [u8]) -> CodedOutputStream<'a> { |
| CodedOutputStream { |
| target: OutputTarget::Bytes, |
| buffer: bytes, |
| position: 0, |
| } |
| } |
| |
| /// `CodedOutputStream` which writes directly to `Vec<u8>`. |
| /// |
| /// Caller should call `flush` at the end to guarantee vec contains |
| /// all written data. |
| pub fn vec(vec: &'a mut Vec<u8>) -> CodedOutputStream<'a> { |
| CodedOutputStream { |
| target: OutputTarget::Vec(vec), |
| buffer: &mut [], |
| position: 0, |
| } |
| } |
| |
| /// Check if EOF is reached. |
| /// |
| /// # Panics |
| /// |
| /// If underlying write has no EOF |
| pub fn check_eof(&self) { |
| match self.target { |
| OutputTarget::Bytes => { |
| assert_eq!(self.buffer.len() as u64, self.position as u64); |
| } |
| OutputTarget::Write(..) | OutputTarget::Vec(..) => { |
| panic!("must not be called with Writer or Vec"); |
| } |
| } |
| } |
| |
| fn refresh_buffer(&mut self) -> ProtobufResult<()> { |
| match self.target { |
| OutputTarget::Write(ref mut write, _) => { |
| write.write_all(&self.buffer[0..self.position as usize])?; |
| self.position = 0; |
| } |
| OutputTarget::Vec(ref mut vec) => unsafe { |
| let vec_len = vec.len(); |
| assert!(vec_len + self.position <= vec.capacity()); |
| vec.set_len(vec_len + self.position); |
| vec.reserve(1); |
| self.buffer = remove_lifetime_mut(remaining_capacity_as_slice_mut(vec)); |
| self.position = 0; |
| }, |
| OutputTarget::Bytes => { |
| panic!("refresh_buffer must not be called on CodedOutputStream create from slice"); |
| } |
| } |
| Ok(()) |
| } |
| |
| /// Flush the buffer to underlying write |
| pub fn flush(&mut self) -> ProtobufResult<()> { |
| match self.target { |
| OutputTarget::Bytes => Ok(()), |
| OutputTarget::Write(..) | OutputTarget::Vec(..) => { |
| // TODO: must not reserve additional in Vec |
| self.refresh_buffer() |
| } |
| } |
| } |
| |
| /// Write a byte |
| pub fn write_raw_byte(&mut self, byte: u8) -> ProtobufResult<()> { |
| if self.position as usize == self.buffer.len() { |
| self.refresh_buffer()?; |
| } |
| self.buffer[self.position as usize] = byte; |
| self.position += 1; |
| Ok(()) |
| } |
| |
| /// Write bytes |
| pub fn write_raw_bytes(&mut self, bytes: &[u8]) -> ProtobufResult<()> { |
| if bytes.len() <= self.buffer.len() - self.position { |
| let bottom = self.position as usize; |
| let top = bottom + (bytes.len() as usize); |
| self.buffer[bottom..top].copy_from_slice(bytes); |
| self.position += bytes.len(); |
| return Ok(()); |
| } |
| |
| self.refresh_buffer()?; |
| |
| assert!(self.position == 0); |
| |
| if self.position + bytes.len() < self.buffer.len() { |
| self.buffer[self.position..self.position + bytes.len()].copy_from_slice(bytes); |
| self.position += bytes.len(); |
| return Ok(()); |
| } |
| |
| match self.target { |
| OutputTarget::Bytes => { |
| unreachable!(); |
| } |
| OutputTarget::Write(ref mut write, _) => { |
| write.write_all(bytes)?; |
| } |
| OutputTarget::Vec(ref mut vec) => { |
| vec.extend(bytes); |
| unsafe { |
| self.buffer = remove_lifetime_mut(remaining_capacity_as_slice_mut(vec)); |
| } |
| } |
| } |
| Ok(()) |
| } |
| |
| /// Write a tag |
| pub fn write_tag( |
| &mut self, |
| field_number: u32, |
| wire_type: wire_format::WireType, |
| ) -> ProtobufResult<()> { |
| self.write_raw_varint32(wire_format::Tag::make(field_number, wire_type).value()) |
| } |
| |
| /// Write varint |
| pub fn write_raw_varint32(&mut self, value: u32) -> ProtobufResult<()> { |
| if self.buffer.len() - self.position >= 5 { |
| // fast path |
| let len = varint::encode_varint32(value, &mut self.buffer[self.position..]); |
| self.position += len; |
| Ok(()) |
| } else { |
| // slow path |
| let buf = &mut [0u8; 5]; |
| let len = varint::encode_varint32(value, buf); |
| self.write_raw_bytes(&buf[..len]) |
| } |
| } |
| |
| /// Write varint |
| pub fn write_raw_varint64(&mut self, value: u64) -> ProtobufResult<()> { |
| if self.buffer.len() - self.position >= 10 { |
| // fast path |
| let len = varint::encode_varint64(value, &mut self.buffer[self.position..]); |
| self.position += len; |
| Ok(()) |
| } else { |
| // slow path |
| let buf = &mut [0u8; 10]; |
| let len = varint::encode_varint64(value, buf); |
| self.write_raw_bytes(&buf[..len]) |
| } |
| } |
| |
| /// Write 32-bit integer little endian |
| pub fn write_raw_little_endian32(&mut self, value: u32) -> ProtobufResult<()> { |
| let bytes = unsafe { mem::transmute::<_, [u8; 4]>(value.to_le()) }; |
| self.write_raw_bytes(&bytes) |
| } |
| |
| /// Write 64-bit integer little endian |
| pub fn write_raw_little_endian64(&mut self, value: u64) -> ProtobufResult<()> { |
| let bytes = unsafe { mem::transmute::<_, [u8; 8]>(value.to_le()) }; |
| self.write_raw_bytes(&bytes) |
| } |
| |
| /// Write `float` |
| pub fn write_float_no_tag(&mut self, value: f32) -> ProtobufResult<()> { |
| let bits = unsafe { mem::transmute::<f32, u32>(value) }; |
| self.write_raw_little_endian32(bits) |
| } |
| |
| /// Write `double` |
| pub fn write_double_no_tag(&mut self, value: f64) -> ProtobufResult<()> { |
| let bits = unsafe { mem::transmute::<f64, u64>(value) }; |
| self.write_raw_little_endian64(bits) |
| } |
| |
| /// Write `float` field |
| pub fn write_float(&mut self, field_number: u32, value: f32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeFixed32)?; |
| self.write_float_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `double` field |
| pub fn write_double(&mut self, field_number: u32, value: f64) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeFixed64)?; |
| self.write_double_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write varint |
| pub fn write_uint64_no_tag(&mut self, value: u64) -> ProtobufResult<()> { |
| self.write_raw_varint64(value) |
| } |
| |
| /// Write varint |
| pub fn write_uint32_no_tag(&mut self, value: u32) -> ProtobufResult<()> { |
| self.write_raw_varint32(value) |
| } |
| |
| /// Write varint |
| pub fn write_int64_no_tag(&mut self, value: i64) -> ProtobufResult<()> { |
| self.write_raw_varint64(value as u64) |
| } |
| |
| /// Write varint |
| pub fn write_int32_no_tag(&mut self, value: i32) -> ProtobufResult<()> { |
| self.write_raw_varint64(value as u64) |
| } |
| |
| /// Write zigzag varint |
| pub fn write_sint64_no_tag(&mut self, value: i64) -> ProtobufResult<()> { |
| self.write_uint64_no_tag(encode_zig_zag_64(value)) |
| } |
| |
| /// Write zigzag varint |
| pub fn write_sint32_no_tag(&mut self, value: i32) -> ProtobufResult<()> { |
| self.write_uint32_no_tag(encode_zig_zag_32(value)) |
| } |
| |
| /// Write `fixed64` |
| pub fn write_fixed64_no_tag(&mut self, value: u64) -> ProtobufResult<()> { |
| self.write_raw_little_endian64(value) |
| } |
| |
| /// Write `fixed32` |
| pub fn write_fixed32_no_tag(&mut self, value: u32) -> ProtobufResult<()> { |
| self.write_raw_little_endian32(value) |
| } |
| |
| /// Write `sfixed64` |
| pub fn write_sfixed64_no_tag(&mut self, value: i64) -> ProtobufResult<()> { |
| self.write_raw_little_endian64(value as u64) |
| } |
| |
| /// Write `sfixed32` |
| pub fn write_sfixed32_no_tag(&mut self, value: i32) -> ProtobufResult<()> { |
| self.write_raw_little_endian32(value as u32) |
| } |
| |
| /// Write `bool` |
| pub fn write_bool_no_tag(&mut self, value: bool) -> ProtobufResult<()> { |
| self.write_raw_varint32(if value { 1 } else { 0 }) |
| } |
| |
| /// Write `enum` |
| pub fn write_enum_no_tag(&mut self, value: i32) -> ProtobufResult<()> { |
| self.write_int32_no_tag(value) |
| } |
| |
| /// Write `enum` |
| pub fn write_enum_obj_no_tag<E>(&mut self, value: E) -> ProtobufResult<()> |
| where |
| E: ProtobufEnum, |
| { |
| self.write_enum_no_tag(value.value()) |
| } |
| |
| /// Write unknown value |
| pub fn write_unknown_no_tag(&mut self, unknown: UnknownValueRef) -> ProtobufResult<()> { |
| match unknown { |
| UnknownValueRef::Fixed64(fixed64) => self.write_raw_little_endian64(fixed64), |
| UnknownValueRef::Fixed32(fixed32) => self.write_raw_little_endian32(fixed32), |
| UnknownValueRef::Varint(varint) => self.write_raw_varint64(varint), |
| UnknownValueRef::LengthDelimited(bytes) => self.write_bytes_no_tag(bytes), |
| } |
| } |
| |
| /// Write `uint64` field |
| pub fn write_uint64(&mut self, field_number: u32, value: u64) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_uint64_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `uint32` field |
| pub fn write_uint32(&mut self, field_number: u32, value: u32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_uint32_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `int64` field |
| pub fn write_int64(&mut self, field_number: u32, value: i64) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_int64_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `int32` field |
| pub fn write_int32(&mut self, field_number: u32, value: i32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_int32_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `sint64` field |
| pub fn write_sint64(&mut self, field_number: u32, value: i64) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_sint64_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `sint32` field |
| pub fn write_sint32(&mut self, field_number: u32, value: i32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_sint32_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `fixed64` field |
| pub fn write_fixed64(&mut self, field_number: u32, value: u64) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeFixed64)?; |
| self.write_fixed64_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `fixed32` field |
| pub fn write_fixed32(&mut self, field_number: u32, value: u32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeFixed32)?; |
| self.write_fixed32_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `sfixed64` field |
| pub fn write_sfixed64(&mut self, field_number: u32, value: i64) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeFixed64)?; |
| self.write_sfixed64_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `sfixed32` field |
| pub fn write_sfixed32(&mut self, field_number: u32, value: i32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeFixed32)?; |
| self.write_sfixed32_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `bool` field |
| pub fn write_bool(&mut self, field_number: u32, value: bool) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_bool_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `enum` field |
| pub fn write_enum(&mut self, field_number: u32, value: i32) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeVarint)?; |
| self.write_enum_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write `enum` field |
| pub fn write_enum_obj<E>(&mut self, field_number: u32, value: E) -> ProtobufResult<()> |
| where |
| E: ProtobufEnum, |
| { |
| self.write_enum(field_number, value.value()) |
| } |
| |
| /// Write unknown field |
| pub fn write_unknown( |
| &mut self, |
| field_number: u32, |
| value: UnknownValueRef, |
| ) -> ProtobufResult<()> { |
| self.write_tag(field_number, value.wire_type())?; |
| self.write_unknown_no_tag(value)?; |
| Ok(()) |
| } |
| |
| /// Write unknown fields |
| pub fn write_unknown_fields(&mut self, fields: &UnknownFields) -> ProtobufResult<()> { |
| for (number, values) in fields { |
| for value in values { |
| self.write_unknown(number, value)?; |
| } |
| } |
| Ok(()) |
| } |
| |
| /// Write bytes |
| pub fn write_bytes_no_tag(&mut self, bytes: &[u8]) -> ProtobufResult<()> { |
| self.write_raw_varint32(bytes.len() as u32)?; |
| self.write_raw_bytes(bytes)?; |
| Ok(()) |
| } |
| |
| /// Write string |
| pub fn write_string_no_tag(&mut self, s: &str) -> ProtobufResult<()> { |
| self.write_bytes_no_tag(s.as_bytes()) |
| } |
| |
| /// Write message |
| pub fn write_message_no_tag<M: Message>(&mut self, msg: &M) -> ProtobufResult<()> { |
| msg.write_length_delimited_to(self) |
| } |
| |
| /// Write `bytes` field |
| pub fn write_bytes(&mut self, field_number: u32, bytes: &[u8]) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeLengthDelimited)?; |
| self.write_bytes_no_tag(bytes)?; |
| Ok(()) |
| } |
| |
| /// Write `string` field |
| pub fn write_string(&mut self, field_number: u32, s: &str) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeLengthDelimited)?; |
| self.write_string_no_tag(s)?; |
| Ok(()) |
| } |
| |
| /// Write `message` field |
| pub fn write_message<M: Message>(&mut self, field_number: u32, msg: &M) -> ProtobufResult<()> { |
| self.write_tag(field_number, wire_format::WireTypeLengthDelimited)?; |
| self.write_message_no_tag(msg)?; |
| Ok(()) |
| } |
| } |
| |
| impl<'a> Write for CodedOutputStream<'a> { |
| fn write(&mut self, buf: &[u8]) -> io::Result<usize> { |
| self.write_raw_bytes(buf)?; |
| Ok(buf.len()) |
| } |
| |
| fn flush(&mut self) -> io::Result<()> { |
| CodedOutputStream::flush(self).map_err(Into::into) |
| } |
| } |
| |
| #[cfg(test)] |
| mod test { |
| use crate::coded_output_stream::CodedOutputStream; |
| use crate::hex::decode_hex; |
| use crate::hex::encode_hex; |
| use crate::wire_format; |
| use crate::ProtobufResult; |
| use std::io::Write; |
| use std::iter; |
| |
| fn test_write<F>(expected: &str, mut gen: F) |
| where |
| F: FnMut(&mut CodedOutputStream) -> ProtobufResult<()>, |
| { |
| let expected_bytes = decode_hex(expected); |
| |
| // write to Write |
| { |
| let mut v = Vec::new(); |
| { |
| let mut os = CodedOutputStream::new(&mut v as &mut dyn Write); |
| gen(&mut os).unwrap(); |
| os.flush().unwrap(); |
| } |
| assert_eq!(encode_hex(&expected_bytes), encode_hex(&v)); |
| } |
| |
| // write to &[u8] |
| { |
| let mut r = Vec::with_capacity(expected_bytes.len()); |
| r.resize(expected_bytes.len(), 0); |
| { |
| let mut os = CodedOutputStream::bytes(&mut r); |
| gen(&mut os).unwrap(); |
| os.check_eof(); |
| } |
| assert_eq!(encode_hex(&expected_bytes), encode_hex(&r)); |
| } |
| |
| // write to Vec<u8> |
| { |
| let mut r = Vec::new(); |
| r.extend(&[11, 22, 33, 44, 55, 66, 77]); |
| { |
| let mut os = CodedOutputStream::vec(&mut r); |
| gen(&mut os).unwrap(); |
| os.flush().unwrap(); |
| } |
| |
| r.drain(..7); |
| assert_eq!(encode_hex(&expected_bytes), encode_hex(&r)); |
| } |
| } |
| |
| #[test] |
| fn test_output_stream_write_raw_byte() { |
| test_write("a1", |os| os.write_raw_byte(0xa1)); |
| } |
| |
| #[test] |
| fn test_output_stream_write_tag() { |
| test_write("08", |os| os.write_tag(1, wire_format::WireTypeVarint)); |
| } |
| |
| #[test] |
| fn test_output_stream_write_raw_bytes() { |
| test_write("00 ab", |os| os.write_raw_bytes(&[0x00, 0xab])); |
| |
| let expected = iter::repeat("01 02 03 04") |
| .take(2048) |
| .collect::<Vec<_>>() |
| .join(" "); |
| test_write(&expected, |os| { |
| for _ in 0..2048 { |
| os.write_raw_bytes(&[0x01, 0x02, 0x03, 0x04])?; |
| } |
| |
| Ok(()) |
| }); |
| } |
| |
| #[test] |
| fn test_output_stream_write_raw_varint32() { |
| test_write("96 01", |os| os.write_raw_varint32(150)); |
| test_write("ff ff ff ff 0f", |os| os.write_raw_varint32(0xffffffff)); |
| } |
| |
| #[test] |
| fn test_output_stream_write_raw_varint64() { |
| test_write("96 01", |os| os.write_raw_varint64(150)); |
| test_write("ff ff ff ff ff ff ff ff ff 01", |os| { |
| os.write_raw_varint64(0xffffffffffffffff) |
| }); |
| } |
| |
| #[test] |
| fn test_output_stream_write_int32_no_tag() { |
| test_write("ff ff ff ff ff ff ff ff ff 01", |os| { |
| os.write_int32_no_tag(-1) |
| }); |
| } |
| |
| #[test] |
| fn test_output_stream_write_int64_no_tag() { |
| test_write("ff ff ff ff ff ff ff ff ff 01", |os| { |
| os.write_int64_no_tag(-1) |
| }); |
| } |
| |
| #[test] |
| fn test_output_stream_write_raw_little_endian32() { |
| test_write("f1 e2 d3 c4", |os| os.write_raw_little_endian32(0xc4d3e2f1)); |
| } |
| |
| #[test] |
| fn test_output_stream_write_float_no_tag() { |
| test_write("95 73 13 61", |os| os.write_float_no_tag(17e19)); |
| } |
| |
| #[test] |
| fn test_output_stream_write_double_no_tag() { |
| test_write("40 d5 ab 68 b3 07 3d 46", |os| { |
| os.write_double_no_tag(23e29) |
| }); |
| } |
| |
| #[test] |
| fn test_output_stream_write_raw_little_endian64() { |
| test_write("f1 e2 d3 c4 b5 a6 07 f8", |os| { |
| os.write_raw_little_endian64(0xf807a6b5c4d3e2f1) |
| }); |
| } |
| |
| #[test] |
| fn test_output_stream_io_write() { |
| let expected = [0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77]; |
| |
| // write to Write |
| { |
| let mut v = Vec::new(); |
| { |
| let mut os = CodedOutputStream::new(&mut v as &mut dyn Write); |
| Write::write(&mut os, &expected).expect("io::Write::write"); |
| Write::flush(&mut os).expect("io::Write::flush"); |
| } |
| assert_eq!(expected, *v); |
| } |
| |
| // write to &[u8] |
| { |
| let mut v = Vec::with_capacity(expected.len()); |
| v.resize(expected.len(), 0); |
| { |
| let mut os = CodedOutputStream::bytes(&mut v); |
| Write::write(&mut os, &expected).expect("io::Write::write"); |
| Write::flush(&mut os).expect("io::Write::flush"); |
| os.check_eof(); |
| } |
| assert_eq!(expected, *v); |
| } |
| |
| // write to Vec<u8> |
| { |
| let mut v = Vec::new(); |
| { |
| let mut os = CodedOutputStream::vec(&mut v); |
| Write::write(&mut os, &expected).expect("io::Write::write"); |
| Write::flush(&mut os).expect("io::Write::flush"); |
| } |
| assert_eq!(expected, *v); |
| } |
| } |
| } |