shvrpc 15.0.0

Rust implementation of the SHV-RPC protocol
Documentation
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
use crate::framerw::{attach_meta_to_timeout_error, log_data_send, read_raw_data, try_chainpack_buf_to_meta, try_receive_frame_base, FrameReader, RawData};
use crate::framerw::{serialize_meta, FrameWriter, ReceiveFrameError};
use crate::rpcframe::{RpcFrame};
use crate::rpcmessage::PeerId;
use async_trait::async_trait;
use crc::{Crc, Digest, CRC_32_ISO_HDLC};
use futures::{AsyncRead, AsyncWrite, AsyncWriteExt};
use log::{log, log_enabled, Level};
use crate::streamrw::DEFAULT_FRAME_SIZE_LIMIT;

const STX: u8 = 0xA2;
const ETX: u8 = 0xA3;
const ATX: u8 = 0xA4;
const ESC: u8 = 0xAA;

const ESTX: u8 = 0x02;
const EETX: u8 = 0x03;
const EATX: u8 = 0x04;
const EESC: u8 = 0x0A;

pub struct SerialFrameReader<R: AsyncRead + Unpin + Send> {
    peer_id: PeerId,
    reader: R,
    raw_data: RawData,
    frame_size_limit: usize,
    with_crc: bool,
}

// https://reveng.sourceforge.io/crc-catalogue/all.htm#crc.cat.crc-32-iso-hdlc
// https://crccalc.com
const CRC_32: Crc<u32> = Crc::<u32>::new(&CRC_32_ISO_HDLC);

impl<R: AsyncRead + Unpin + Send> SerialFrameReader<R> {
    pub fn new(reader: R) -> Self {
        Self {
            peer_id: 0,
            reader,
            raw_data: RawData::new(),
            frame_size_limit: DEFAULT_FRAME_SIZE_LIMIT,
            with_crc: false,
        }
    }
    #[must_use]
    pub fn with_peer_id(mut self, peer_id: PeerId) -> Self {
        self.peer_id = peer_id;
        self
    }

    #[must_use]
    pub fn with_frame_size_limit(mut self, frame_size_limit: usize) -> Self {
        self.frame_size_limit = frame_size_limit;
        self
    }

    #[must_use]
    pub fn with_crc_check(mut self, on: bool) -> Self {
        self.with_crc = on;
        self
    }
    async fn get_raw_byte(&mut self, with_timeout: bool) -> Result<u8, ReceiveFrameError> {
        if self.raw_data.bytes_available() == 0 {
            read_raw_data(&mut self.reader, &mut self.raw_data, with_timeout).await?;
        }
        let b = *self.raw_data.data.get(self.raw_data.consumed).expect("Byte must be available");
        self.raw_data.consumed += 1;
        Ok(b)
    }
    fn unget_stx(&mut self) {
        self.raw_data.consumed -= 1;
        assert!(self.raw_data.data.len() > self.raw_data.consumed, "Length must be more than consumed");
        assert_eq!(self.raw_data.data.get(self.raw_data.consumed), Some(&STX), "Last byte must be STX");
    }
    fn unescape_byte(b: u8) -> Result<u8, ReceiveFrameError> {
        let b = match b {
            ESTX => STX,
            EETX => ETX,
            EATX => ATX,
            EESC => ESC,
            b => {
                return Err(ReceiveFrameError::FramingError(format!(
                    "Framing error, invalid escape byte {b:#02x}"
                )));
            }
        };
        Ok(b)
    }
    async fn get_frame_bytes_impl(&mut self) -> Result<Vec<u8>, ReceiveFrameError> {
        let mut crc_digest = if self.with_crc { Some(CRC_32.digest()) } else { None };
        let update_crc_digest = |crc_digest: &mut Option<Digest<u32>>, b: u8| {
            if let Some(crc_digest) = crc_digest {
                crc_digest.update(&[b]);
            }
        };

        let peer_id = self.peer_id;
        let frame_size_limit = self.frame_size_limit();
        let mut data = vec![];
        let push_data_byte = |b: u8, data: &mut Vec<u8>| {
            if data.len() > frame_size_limit {
                Err(ReceiveFrameError::FrameTooLarge(
                        format!("Client ID: {peer_id}, Jumbo frame threshold {frame_size_limit} bytes exceeded during read."),
                        try_chainpack_buf_to_meta(data))
                )
            } else {
                data.push(b);
                Ok(())
            }
        };
        while self.get_raw_byte(false).await? != STX {}
        loop {
            match self.get_raw_byte(true).await.map_err(|e| attach_meta_to_timeout_error(e, &data))? {
                STX => {
                    self.unget_stx();
                    return Err(ReceiveFrameError::FramingError(
                        "Incomplete frame, new STX received within data.".into(),
                    ));
                }
                ETX => {
                    if let Some(crc_digest) = crc_digest {
                        let mut crc_data = [0u8; 4];
                        for crc_b in &mut crc_data {
                            let b = match self.get_raw_byte(true).await.map_err(|e| attach_meta_to_timeout_error(e, &data))? {
                                STX => {
                                    self.unget_stx();
                                    return Err(ReceiveFrameError::FramingError(
                                        "STX received in CRC.".into(),
                                    ));
                                }
                                ESC => Self::unescape_byte(self.get_raw_byte(true).await.map_err(|e| attach_meta_to_timeout_error(e, &data))?)?,
                                ATX => {
                                    return Err(ReceiveFrameError::FramingError(
                                        "ATX received in CRC.".into(),
                                    ));
                                }
                                ETX => {
                                    return Err(ReceiveFrameError::FramingError(
                                        "ETX received in CRC.".into(),
                                    ));
                                }
                                b => b,
                            };
                            *crc_b = b;
                        }
                        fn as_u32_be(array: [u8; 4]) -> u32 {
                            (u32::from(array[0]) << 24)
                                + (u32::from(array[1]) << 16)
                                + (u32::from(array[2]) << 8)
                                + u32::from(array[3])
                        }
                        let crc1 = as_u32_be(crc_data);
                        // let digest = replace(&mut self.crc_digest, CRC_32.digest());
                        let crc2 = crc_digest.finalize();
                        //info!("CRC1 {:#04x}", crc1);
                        //info!("CRC2 {:#04x}", crc2);
                        if crc1 != crc2 {
                            log!(target: "Serial", Level::Warn, "CRC error");
                            return Err(ReceiveFrameError::FramingError("CRC check error.".into()));
                        }
                    }
                    break;
                }
                ATX => {
                    return Err(ReceiveFrameError::FramingError(
                        "ATX received, aborting current frame.".into(),
                    ))
                }
                ESC => {
                    update_crc_digest(&mut crc_digest, ESC);
                    let b = self.get_raw_byte(true).await.map_err(|e| attach_meta_to_timeout_error(e, &data))?;
                    if b == STX {
                        self.unget_stx();
                        return Err(ReceiveFrameError::FramingError(
                            "Incomplete frame, new STX received within escaped data.".into(),
                        ));
                    }
                    update_crc_digest(&mut crc_digest, b);
                    let ub = Self::unescape_byte(b)?;
                    push_data_byte(ub, &mut data)?;
                }
                b => {
                    update_crc_digest(&mut crc_digest, b);
                    push_data_byte(b, &mut data)?;
                }
            }
        }
        Ok(data)
    }
    #[cfg(test)]
    async fn read_escaped(&mut self) -> crate::Result<Vec<u8>> {
        let mut data = Vec::default();
        while let Ok(b) = self.get_raw_byte(false).await.map_err(|e| attach_meta_to_timeout_error(e, &data)) {
            let b = match b {
                ESC => Self::unescape_byte(self.get_raw_byte(false).await.map_err(|e| attach_meta_to_timeout_error(e, &data))?)?,
                b => b,
            };
            data.push(b);
        }
        Ok(data)
    }
}
#[async_trait]
impl<R: AsyncRead + Unpin + Send> FrameReader for SerialFrameReader<R> {
    fn peer_id(&self) -> PeerId {
       self.peer_id
    }

    fn frame_size_limit(&self) -> usize {
        self.frame_size_limit
    }

    async fn get_frame_bytes(&mut self) -> Result<Vec<u8>, ReceiveFrameError> {
        self.get_frame_bytes_impl().await
    }
    async fn try_receive_frame(&mut self) -> Result<RpcFrame, ReceiveFrameError> {
        loop {
            match try_receive_frame_base(self).await {
                Ok(frame) => return Ok(frame),
                Err(ReceiveFrameError::FramingError(e)) => {
                    // silently ignore ATX, and CRC erorrs
                    log!(target: "SerialFrameError", Level::Warn, "Ignoring serial framing error: {e}");
                }
                Err(e) => {
                    return Err(e)
                }
            }
        }
    }
}
pub struct SerialFrameWriter<W: AsyncWrite + Unpin + Send> {
    peer_id: PeerId,
    writer: W,
    with_crc: bool,
}
impl<W: AsyncWrite + Unpin + Send> SerialFrameWriter<W> {
    pub fn new(writer: W) -> Self {
        Self {
            peer_id: 0,
            writer,
            with_crc: false,
        }
    }
    #[must_use]
    pub fn with_peer_id(mut self, peer_id: PeerId) -> Self {
        self.peer_id = peer_id;
        self
    }
    #[must_use]
    pub fn with_crc_check(mut self, on: bool) -> Self {
        self.with_crc = on;
        self
    }
    async fn write_bytes(
        &mut self,
        digest: &mut Option<crc::Digest<'_, u32>>,
        data: &[u8],
    ) -> crate::Result<()> {
        if let Some(digest) = digest {
            digest.update(data);
        }
        if log_enabled!(target: "RpcData", Level::Debug) {
            log_data_send(data);
        }
        self.writer.write_all(data).await?;
        Ok(())
    }
    async fn write_escaped(
        &mut self,
        digest: &mut Option<crc::Digest<'_, u32>>,
        data: &[u8],
    ) -> crate::Result<()> {
        for b in data {
            match *b {
                STX => self.write_bytes(digest, &[ESC, ESTX]).await?,
                ETX => self.write_bytes(digest, &[ESC, EETX]).await?,
                ATX => self.write_bytes(digest, &[ESC, EATX]).await?,
                ESC => self.write_bytes(digest, &[ESC, EESC]).await?,
                b => self.write_bytes(digest, &[b]).await?,
            }
        }
        Ok(())
    }
}
#[async_trait]
impl<W: AsyncWrite + Unpin + Send> FrameWriter for SerialFrameWriter<W> {
    fn peer_id(&self) -> PeerId {
        self.peer_id
    }
    async fn send_frame_impl(&mut self, frame: RpcFrame) -> crate::Result<()> {
        let crc = crc::Crc::<u32>::new(&CRC_32_ISO_HDLC);
        let mut digest = if self.with_crc {
            Some(crc.digest())
        } else {
            None
        };
        let meta_data = serialize_meta(&frame)?;
        self.write_bytes(&mut None, &[STX]).await?;
        let protocol = [frame.protocol as u8];
        self.write_escaped(&mut digest, &protocol).await?;
        self.write_escaped(&mut digest, &meta_data).await?;
        self.write_escaped(&mut digest, frame.data()).await?;
        self.write_bytes(&mut None, &[ETX]).await?;
        if self.with_crc {
            fn u32_to_bytes(x: u32) -> [u8; 4] {
                let b0: u8 = ((x >> 24) & 0xff) as u8;
                let b1: u8 = ((x >> 16) & 0xff) as u8;
                let b2: u8 = ((x >> 8) & 0xff) as u8;
                let b3: u8 = (x & 0xff) as u8;
                [b0, b1, b2, b3]
            }
            let crc = digest.expect("digest should be some here").finalize();
            let crc_bytes = u32_to_bytes(crc);
            self.write_escaped(&mut None, &crc_bytes).await?;
        }
        // Ensure the encoded frame is written to the socket. The calls above
        // are to the buffered stream and writes. Calling `flush` writes the
        // remaining contents of the buffer to the socket.
        self.writer.flush().await?;
        Ok(())
    }
}

#[cfg(test)]
mod test {
    use super::*;
    use crate::framerw::test::from_hex;
    use crate::RpcMessage;
    use log::{debug, error};
    use macro_rules_attribute::apply;
    use smol_macros::test;
    use smol::io::BufWriter;
    use shvproto::util::hex_dump;
    use crate::rpcframe::Protocol;
    use crate::util::hex_string;

    fn init_log() {
        env_logger::builder()
            // .filter(None, LevelFilter::Debug)
            .is_test(true)
            .try_init()
            .inspect_err(|err| error!("Logger didn't work: {err}"))
            .ok();
    }

    #[apply(test!)]
    async fn test_write_bytes() {
        for (data, esc_data) in [
            (&b"hello"[..], &b"hello"[..]),
            (&[STX], &[ESC, ESTX]),
            (&[ETX], &[ESC, EETX]),
            (&[ATX], &[ESC, EATX]),
            (&[ESC], &[ESC, EESC]),
            (&[STX, ESC], &[ESC, ESTX, ESC, EESC]),
        ] {
            let mut buff: Vec<u8> = vec![];
            {
                let buffwr = BufWriter::new(&mut buff);
                let mut wr = SerialFrameWriter::new(buffwr);
                wr.write_escaped(&mut None, data).await.unwrap();
                //drop(wr);
                wr.writer.flush().await.unwrap();
                assert_eq!(&buff, esc_data);
            }
            {
                let buffrd = smol::io::BufReader::new(&*buff);
                let mut rd = SerialFrameReader::new(buffrd);
                let read_data = rd.read_escaped().await.unwrap();
                assert_eq!(&read_data, data);
            }
        }
    }

    #[apply(test!)]
    async fn test_write_reset_session_frame() {
        init_log();
        let reset_frame_data = [STX, Protocol::ResetSession as u8, ETX];
        {
            let frame = RpcFrame::new_reset_session();
            let mut buff: Vec<u8> = vec![];
            let buffwr = BufWriter::new(&mut buff);
            {
                let mut wr = SerialFrameWriter::new(buffwr).with_crc_check(false);
                wr.send_frame(frame.clone()).await.unwrap();
            }
            assert_eq!(&buff[..], reset_frame_data);
        }
        {
            let mut crc_digest = CRC_32.digest();
            crc_digest.update(&reset_frame_data[1 .. 2]);
            let crc = crc_digest.finalize();
            assert_eq!(crc, 0xd202_ef8d);
            let frame = RpcFrame::new_reset_session();
            let mut buff: Vec<u8> = vec![];
            let buffwr = BufWriter::new(&mut buff);
            {
                let mut wr = SerialFrameWriter::new(buffwr).with_crc_check(true);
                wr.send_frame(frame.clone()).await.unwrap();
            }
            assert_eq!(&buff[0 .. reset_frame_data.len()], reset_frame_data);
            assert_eq!(&buff[reset_frame_data.len() ..], [0xd2, 0x02, 0xef, 0x8d]);
        }
    }

    #[apply(test!)]
    async fn test_write_read_frame() {
        init_log();

        for with_crc in [false, true] {
            let msg = RpcMessage::new_request("foo/bar", "baz").with_param(with_crc);
            let frame = msg.to_frame().unwrap();
            let mut buff: Vec<u8> = vec![];
            let buffwr = BufWriter::new(&mut buff);
            {
                let mut wr = SerialFrameWriter::new(buffwr).with_crc_check(with_crc);
                wr.send_frame(frame.clone()).await.unwrap();
            }
            debug!("msg: {msg}");
            debug!("len: {}, hex: {:?}", buff.len(), hex_string(&buff, Some(" ")));
            debug!("with crc: {with_crc}");
            for prefix in [b"".to_vec(), b"1234".to_vec(), [ATX].to_vec()] {
                let mut buff2 = prefix;
                buff2.append(&mut buff.clone());
                //debug!("bytes:\n{}\n-------------", hex_dump(&buff2));
                let buffrd = smol::io::BufReader::new(&*buff2);
                let mut rd = SerialFrameReader::new(buffrd).with_crc_check(with_crc);
                let rd_frame = rd.receive_frame().await.unwrap();
                assert_eq!(&rd_frame, &frame);
            }
            for prefix in [
                [STX].to_vec(),
                [STX, ESC].to_vec(),
                [STX, ESC, 1u8].to_vec(),
                [STX, ATX].to_vec(),
                [STX, ESC, ATX].to_vec(),
            ] {
                let mut buff2 = prefix;
                buff2.append(&mut buff.clone());
                //debug!("bytes:\n{}\n-------------", hex_dump(&buff2));
                let buffrd = smol::io::BufReader::new(&*buff2);
                let mut rd = SerialFrameReader::new(buffrd).with_crc_check(with_crc);
                let rd_frame = rd.receive_frame().await.unwrap();
                assert_eq!(&rd_frame, &frame);
            }
        }
    }

    #[apply(test!)]
    async fn test_read_frame_by_chunks() {
        init_log();

        for chunks in [
            // msg: <1:1,8:1,9:"foo/bar",10:"baz">i{1:1}
            vec![
                    from_hex("STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df"),
            ],
            vec![
                from_hex("STX 01 8b 41 41"),
                from_hex("48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX"),
                from_hex("44 cc 24 df"),
            ],

            // ESC in CRC
            // msg: <1:1,8:19,9:"foo/bar",10:"baz">i{1:18}
            vec![
                from_hex("STX 01 8b 41 41 48 53 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 52 ff ETX ESC ESTX 84 e6 8c"),
            ],
            vec![
                from_hex("STX 01 8b 41 41 48 53 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 52 ff ETX ESC"),
                from_hex("ESTX 84 e6 8c"),
            ],

            // ESC in data
            // msg: <1:1,8:162,9:"foo/bar",10:"baz">i{1:161}
            vec![
                from_hex("STX 01 8b 41 41 48 82 80 ESC ESTX 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 82 80 a1 ff ETX 12 53 57 e5"),
            ],
            vec![
                from_hex("STX 01 8b 41 41 48 82 80 ESC"),
                from_hex("02 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 82 80 a1 ff"),
                from_hex("ETX"),
                from_hex("12 53 57"),
                from_hex("e5"),
            ],
            ] {
                debug!("hex: {chunks:?}");
                let mut rd = SerialFrameReader::new(crate::framerw::test::Chunks { chunks }).with_crc_check(true);
                let frame = rd.receive_frame().await;
                debug!("frame: {:?}", &frame);
                assert!(frame.is_ok());
            };
    }

    #[apply(test!)]
    async fn test_read_multiframe_chunk() {
        init_log();
        for with_crc in [false, true] {
            for data in [
                // msg: <1:1,8:1,9:"foo/bar",10:"baz">i{1:1}
                from_hex(
                    "STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    STX 01 8b 41 41 48 41 49 86 07 66 6f 6f "
                ),
                from_hex(
                    "STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    ESC 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff     44 cc 24 df
                    STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df"
                ),
            ] {
                debug!("bytes:\n{}\n-------------", hex_dump(&data));
                let buffrd = smol::io::BufReader::new(&*data);
                let mut rd = SerialFrameReader::new(buffrd).with_crc_check(with_crc);
                for _ in 0 .. 3 {
                    let frame = rd.receive_frame().await;
                    debug!("frame: {:?}", &frame);
                    assert!(frame.is_ok());
                }
            }
        }
    }

    #[apply(test!)]
    async fn test_read_over_frame_size_limit() {
        init_log();
        for with_crc in [false, true] {
            for data in [
                // msg: <1:1,8:162,9:"foo/bar",10:"baz">i{1:161} - 30 bytes
                // msg: <1:1,8:1,9:"foo/bar",10:"baz">i{1:1} - 26 bytes
                from_hex("
                    STX 01 8b 41 41 48 82 80 ESC ESTX 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 82 80 a1 ff ETX 12 53 57 e5
                    STX 01 8b 41 41 48 41 49 86 07 66 6f 6f 2f 62 61 72 4a 86 03 62 61 7a ff 8a 41 41 ff ETX 44 cc 24 df
                    "
                ),
            ] {
                debug!("bytes:\n{}\n-------------", hex_dump(&data));
                {
                    let buffrd = smol::io::BufReader::new(&*data);
                    let mut rd = SerialFrameReader::new(buffrd)
                        .with_crc_check(with_crc)
                        .with_frame_size_limit(30);
                    {
                        // should read first message with size equal to frame size limit
                        let frame = rd.receive_frame().await.unwrap();
                        debug!("frame: {:?}", &frame);
                        assert_eq!(frame.to_rpcmesage().unwrap().param().unwrap().as_int(), 161);
                    }
                }
                {
                    let buffrd = smol::io::BufReader::new(&*data);
                    let mut rd = SerialFrameReader::new(buffrd)
                        .with_crc_check(with_crc)
                        .with_frame_size_limit(29);
                    {
                        assert!(matches!(rd.receive_frame().await.unwrap_err(), ReceiveFrameError::FrameTooLarge(..)));
                        let frame = rd.receive_frame().await.unwrap();
                        debug!("frame: {:?}", &frame);
                        assert_eq!(frame.to_rpcmesage().unwrap().param().unwrap().as_int(), 1);
                    }
                }
            }
        }
    }
}