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
#![deny(missing_docs, rust_2018_idioms, unused, unused_crate_dependencies, unused_import_braces, unused_lifetimes, unused_qualifications, warnings)]
#![forbid(unsafe_code)]
use {
std::{
fmt,
future::Future,
io,
num::TryFromIntError,
pin::Pin,
string::FromUtf8Error,
sync::Arc,
},
derive_more::From,
tokio::io::{
AsyncRead,
AsyncWrite,
},
};
#[cfg(feature = "blocking")] use std::io::prelude::*;
pub use async_proto_derive::Protocol;
#[doc(hidden)] pub use {
derive_more,
tokio,
};
mod impls;
#[derive(Debug, From, Clone)]
#[allow(missing_docs)]
pub enum ReadError {
BufSize(TryFromIntError),
Custom(String),
FloatNotFinite,
Io(Arc<io::Error>),
ReadNever,
#[from(ignore)]
UnknownVariant(u8),
Utf8(FromUtf8Error),
}
impl From<io::Error> for ReadError {
fn from(e: io::Error) -> ReadError {
ReadError::Io(Arc::new(e))
}
}
impl fmt::Display for ReadError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
ReadError::BufSize(e) => write!(f, "received a buffer with more than usize::MAX elements: {}", e),
ReadError::Custom(msg) => msg.fmt(f),
ReadError::FloatNotFinite => write!(f, "received an infinite or NaN JSON number"),
ReadError::Io(e) => write!(f, "I/O error: {}", e),
ReadError::ReadNever => write!(f, "attempted to read an empty type"),
ReadError::UnknownVariant(n) => write!(f, "unknown enum variant: {}", n),
ReadError::Utf8(e) => e.fmt(f),
}
}
}
#[derive(Debug, From, Clone)]
#[allow(missing_docs)]
pub enum WriteError {
BufSize(TryFromIntError),
Custom(String),
Io(Arc<io::Error>),
}
impl From<io::Error> for WriteError {
fn from(e: io::Error) -> WriteError {
WriteError::Io(Arc::new(e))
}
}
impl fmt::Display for WriteError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
WriteError::BufSize(e) => write!(f, "tried to send a buffer with more than u64::MAX elements: {}", e),
WriteError::Custom(msg) => msg.fmt(f),
WriteError::Io(e) => write!(f, "I/O error: {}", e),
}
}
}
pub trait Protocol: Sized {
fn read<'a, R: AsyncRead + Unpin + Send + 'a>(stream: &'a mut R) -> Pin<Box<dyn Future<Output = Result<Self, ReadError>> + Send + 'a>>;
fn write<'a, W: AsyncWrite + Unpin + Send + 'a>(&'a self, sink: &'a mut W) -> Pin<Box<dyn Future<Output = Result<(), WriteError>> + Send + 'a>>;
#[cfg(feature = "blocking")]
fn read_sync(stream: &mut impl Read) -> Result<Self, ReadError>;
#[cfg(feature = "blocking")]
fn write_sync(&self, sink: &mut impl Write) -> Result<(), WriteError>;
}