This library provides net.Dial and net.Listen functions for
GOOS=wasip1.
Applications built with this library are compatible with WasmEdge and wasi-go such as Timecraft.
Note: GOOS=wasip1 requires Go 1.21.
The WASI preview 1 specification has partial support for socket networking,
preventing a large class of Go applications from running when compiled to
WebAssembly with GOOS=wasip1. Extensions to the base specifications have been
implemented by runtimes to enable a wider range of programs to be run as
WebAssembly modules.
This package aims to offset Go applications built with GOOS=wasip1 the
opportunity to leverage those WASI extensions, by providing high level functions
similar to those found in the standard net package to create network clients
and servers.
Where possible, the package offers the ability to automatically configure the
network stack via init functions called on package imports. This model is
currently supported for http and mysql with those imports:
import _ "github.com/stealthrocket/net/http"import _ "github.com/stealthrocket/net/mysql"When imported, those packages alter the default configuration to install a dialer function implemented on top of the WASI socket extensions. When compiled to other targets, the import of those packages does nothing.
Packages implementing network clients for various protocols usually support configuration through the installation of an alternative dial function allowing the application to customize how network connections are established.
The wasip1 sub-package provides dial functions matching the signature of those
implemented in the standard net package to integrate with those configuration
mechanisms.
The sub-modules contain examples of how to configure popular Go libraries to
leverage the dial functions of wasip1. Here is an example for a Redis client:
client := redis.NewClient(&redis.Options{
Addr: "localhost:6379",
Dialer: wasip1.DialContext, // change the dial function to use socket extensions
})Network servers can be created using the wasip1.Listen function, which mimics
the signature of net.Listen but uses WASI socket extensions to create the
net.Listener.
For example, a program compiled to GOOS=wasip1 can create a http server by
first constructing a listener and passing it to the server's Serve method:
import (
"net/http"
"github.com/stealthrocket/net/wasip1"
)
func main() {
listener, err := wasip1.Listen("tcp", "127.0.0.1:3000")
if err != nil {
...
}
server := &http.Server{
...
}
if err := server.Serve(listener); err != nil {
...
}
}Note that using convenience functions like http.ListenAndServe will not
work since they are hardcoded to depend on the standard net package.
There are two methods available for resolving a set of IP addresses for a hostname.
The pure Go name resolver is the default for GOOS=wasip1.
All you need is the following import somewhere in your application:
import _ "github.com/stealthrocket/net/wasip1"The library will then automatically configure the net.DefaultResolver.
You'll then be able to use the lookup functions from the standard
library (e.g. net.LookupIP(host)).
The sock_getaddrinfo host function is used to implement name resolution.
To use this method, compile the library with the getaddrinfo build tag.
When using this method, the standard library resolver will not work; you
cannot use net.DefaultResolver, net.LookupIP, etc.
Note that sock_getaddrinfo may block.