Files
wazero/internal/sock/sock.go
Edoardo Vacchi 97d0d70b73
Some checks failed
Release CLI / Pre-release build (push) Has been cancelled
Release CLI / Pre-release test (macos-12) (push) Has been cancelled
Release CLI / Pre-release test (ubuntu-22.04) (push) Has been cancelled
Release CLI / Pre-release test (windows-2022) (push) Has been cancelled
Release CLI / Release (push) Has been cancelled
wasi: add support for sockets (#1493)
Signed-off-by: Edoardo Vacchi <evacchi@users.noreply.github.com>
Signed-off-by: Adrian Cole <adrian@tetrate.io>
Co-authored-by: Crypt Keeper <64215+codefromthecrypt@users.noreply.github.com>
Co-authored-by: Achille <achille.roussel@gmail.com>
Co-authored-by: Adrian Cole <adrian@tetrate.io>
2023-06-02 20:45:42 +08:00

89 lines
2.2 KiB
Go

package sock
import (
"fmt"
"net"
"syscall"
"github.com/tetratelabs/wazero/internal/fsapi"
)
// TCPSock is a pseudo-file representing a TCP socket.
type TCPSock interface {
fsapi.File
Accept() (TCPConn, syscall.Errno)
}
// TCPConn is a pseudo-file representing a TCP connection.
type TCPConn interface {
fsapi.File
// Recvfrom only supports the flag sysfs.MSG_PEEK
Recvfrom(p []byte, flags int) (n int, errno syscall.Errno)
Shutdown(how int) syscall.Errno
}
// ConfigKey is a context.Context Value key. Its associated value should be a Config.
type ConfigKey struct{}
// Config is an internal struct meant to implement
// the interface in experimental/sock/Config.
type Config struct {
// TCPAddresses is a slice of the configured host:port pairs.
TCPAddresses []TCPAddress
}
// TCPAddress is a host:port pair to pre-open.
type TCPAddress struct {
// Host is the host name for this listener.
Host string
// Port is the port number for this listener.
Port int
}
// WithTCPListener implements the method of the same name in experimental/sock/Config.
//
// However, to avoid cyclic dependencies, this is returning the *Config in this scope.
// The interface is implemented in experimental/sock/Config via delegation.
func (c *Config) WithTCPListener(host string, port int) *Config {
ret := c.clone()
ret.TCPAddresses = append(ret.TCPAddresses, TCPAddress{host, port})
return &ret
}
// Makes a deep copy of this sockConfig.
func (c *Config) clone() Config {
ret := *c
ret.TCPAddresses = make([]TCPAddress, 0, len(c.TCPAddresses))
ret.TCPAddresses = append(ret.TCPAddresses, c.TCPAddresses...)
return ret
}
// BuildTCPListeners build listeners from the current configuration.
func (c *Config) BuildTCPListeners() (tcpListeners []*net.TCPListener, err error) {
for _, tcpAddr := range c.TCPAddresses {
var ln net.Listener
ln, err = net.Listen("tcp", tcpAddr.String())
if err != nil {
break
}
if tcpln, ok := ln.(*net.TCPListener); ok {
tcpListeners = append(tcpListeners, tcpln)
}
}
if err != nil {
// An error occurred, cleanup.
for _, l := range tcpListeners {
_ = l.Close() // Ignore errors, we are already cleaning.
}
tcpListeners = nil
}
return
}
func (t TCPAddress) String() string {
return fmt.Sprintf("%s:%d", t.Host, t.Port)
}