Skip to content

Commit 7411bf1

Browse files
committed
Add function to downcast a stream error to a network-related error. Similar to the existing filesystem-error-code and http-error-code functions.
1 parent 8f94133 commit 7411bf1

File tree

5 files changed

+63
-2
lines changed

5 files changed

+63
-2
lines changed

crates/wasi-http/src/bindings.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ mod generated {
88
wasmtime::component::bindgen!({
99
path: "wit",
1010
world: "wasi:http/proxy",
11+
features: ["network-error-code"],
1112
tracing: true,
1213
// Flag this as "possibly async" which will cause the exports to be
1314
// generated as async, but none of the imports here are async since
@@ -56,6 +57,7 @@ pub mod sync {
5657
#![allow(missing_docs)]
5758
wasmtime::component::bindgen!({
5859
world: "wasi:http/proxy",
60+
features: ["network-error-code"],
5961
tracing: true,
6062
async: false,
6163
with: {

crates/wasi-http/wit/deps/sockets/network.wit

+16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,8 @@
11
@since(version = 0.2.0)
22
interface network {
3+
@unstable(feature = network-error-code)
4+
use wasi:io/error@0.2.1.{error};
5+
36
/// An opaque resource that represents access to (a subset of) the network.
47
/// This enables context-based security for networking.
58
/// There is no need for this to map 1:1 to a physical network interface.
@@ -105,6 +108,19 @@ interface network {
105108
permanent-resolver-failure,
106109
}
107110

111+
/// Attempts to extract a network-related `error-code` from the stream
112+
/// `error` provided.
113+
///
114+
/// Stream operations which return `stream-error::last-operation-failed`
115+
/// have a payload with more information about the operation that failed.
116+
/// This payload can be passed through to this function to see if there's
117+
/// network-related information about the error to return.
118+
///
119+
/// Note that this function is fallible because not all stream-related
120+
/// errors are network-related errors.
121+
@unstable(feature = network-error-code)
122+
network-error-code: func(err: borrow<error>) -> option<error-code>;
123+
108124
@since(version = 0.2.0)
109125
enum ip-address-family {
110126
/// Similar to `AF_INET` in POSIX.

crates/wasi/src/bindings.rs

+2
Original file line numberDiff line numberDiff line change
@@ -148,6 +148,7 @@ pub mod sync {
148148
wasmtime::component::bindgen!({
149149
path: "wit",
150150
world: "wasi:cli/command",
151+
features: ["network-error-code"],
151152
tracing: true,
152153
trappable_error_type: {
153154
"wasi:io/streams/stream-error" => StreamError,
@@ -326,6 +327,7 @@ mod async_io {
326327
wasmtime::component::bindgen!({
327328
path: "wit",
328329
world: "wasi:cli/command",
330+
features: ["network-error-code"],
329331
tracing: true,
330332
trappable_imports: true,
331333
async: {

crates/wasi/src/host/network.rs

+27-2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,19 @@ where
1515
fn convert_error_code(&mut self, error: SocketError) -> anyhow::Result<ErrorCode> {
1616
error.downcast()
1717
}
18+
19+
fn network_error_code(
20+
&mut self,
21+
err: Resource<anyhow::Error>,
22+
) -> anyhow::Result<Option<ErrorCode>> {
23+
let err = self.table().get(&err)?;
24+
25+
if let Some(err) = err.downcast_ref::<std::io::Error>() {
26+
return Ok(Some(ErrorCode::from(err)));
27+
}
28+
29+
Ok(None)
30+
}
1831
}
1932

2033
impl<T> crate::bindings::sockets::network::HostNetwork for WasiImpl<T>
@@ -32,8 +45,14 @@ where
3245

3346
impl From<io::Error> for ErrorCode {
3447
fn from(value: io::Error) -> Self {
48+
(&value).into()
49+
}
50+
}
51+
52+
impl From<&io::Error> for ErrorCode {
53+
fn from(value: &io::Error) -> Self {
3554
// Attempt the more detailed native error code first:
36-
if let Some(errno) = Errno::from_io_error(&value) {
55+
if let Some(errno) = Errno::from_io_error(value) {
3756
return errno.into();
3857
}
3958

@@ -62,7 +81,13 @@ impl From<io::Error> for ErrorCode {
6281

6382
impl From<Errno> for ErrorCode {
6483
fn from(value: Errno) -> Self {
65-
match value {
84+
(&value).into()
85+
}
86+
}
87+
88+
impl From<&Errno> for ErrorCode {
89+
fn from(value: &Errno) -> Self {
90+
match *value {
6691
Errno::WOULDBLOCK => ErrorCode::WouldBlock,
6792
#[allow(unreachable_patterns)] // EWOULDBLOCK and EAGAIN can have the same value.
6893
Errno::AGAIN => ErrorCode::WouldBlock,

crates/wasi/wit/deps/sockets/network.wit

+16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,8 @@
11
@since(version = 0.2.0)
22
interface network {
3+
@unstable(feature = network-error-code)
4+
use wasi:io/error@0.2.1.{error};
5+
36
/// An opaque resource that represents access to (a subset of) the network.
47
/// This enables context-based security for networking.
58
/// There is no need for this to map 1:1 to a physical network interface.
@@ -105,6 +108,19 @@ interface network {
105108
permanent-resolver-failure,
106109
}
107110

111+
/// Attempts to extract a network-related `error-code` from the stream
112+
/// `error` provided.
113+
///
114+
/// Stream operations which return `stream-error::last-operation-failed`
115+
/// have a payload with more information about the operation that failed.
116+
/// This payload can be passed through to this function to see if there's
117+
/// network-related information about the error to return.
118+
///
119+
/// Note that this function is fallible because not all stream-related
120+
/// errors are network-related errors.
121+
@unstable(feature = network-error-code)
122+
network-error-code: func(err: borrow<error>) -> option<error-code>;
123+
108124
@since(version = 0.2.0)
109125
enum ip-address-family {
110126
/// Similar to `AF_INET` in POSIX.

0 commit comments

Comments
 (0)