Files
wazero/internal/platform/select_test.go
Edoardo Vacchi ea336061c2
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: introduce platform.Select and use it for poll_oneoff (#1346)
The PR introduces the `platform.Select()` API, wrapping `select(2)` on POSIX and emulated in some cases on Windows. RATIONALE.md contains a full explanation of the approach followed in `poll_oneoff` to handle Stdin and the other types of file descriptors, and the clock subscriptions.

It also introduces an abstraction (`StdioFilePoller`) to allow the simulation of different scenarios (waiting for input, input ready, timeout expired, etc.) when unit-testing interactive input.

This closes #1317.

Signed-off-by: Edoardo Vacchi <evacchi@users.noreply.github.com>
2023-04-18 16:31:34 +02:00

93 lines
2.1 KiB
Go

package platform
import (
"os"
"runtime"
"syscall"
"testing"
"time"
"github.com/tetratelabs/wazero/internal/testing/require"
)
func TestSelect(t *testing.T) {
t.Run("should return immediately with no fds and duration 0", func(t *testing.T) {
for {
dur := time.Duration(0)
n, err := Select(0, nil, nil, nil, &dur)
if err == syscall.EINTR {
t.Logf("Select interrupted")
continue
}
require.NoError(t, err)
require.Equal(t, 0, n)
break
}
})
t.Run("should wait for the given duration", func(t *testing.T) {
dur := 250 * time.Millisecond
var took time.Duration
for {
// On some platforms (e.g. Linux), the passed-in timeval is
// updated by select(2). We are not accounting for this
// in our implementation.
start := time.Now()
n, err := Select(0, nil, nil, nil, &dur)
took = time.Since(start)
if err == syscall.EINTR {
t.Logf("Select interrupted after %v", took)
continue
}
require.NoError(t, err)
require.Equal(t, 0, n)
break
}
// On some platforms the actual timeout might be arbitrarily
// less than requested.
if took < dur {
if runtime.GOOS == "linux" {
// Linux promises to only return early if a file descriptor
// becomes ready (not applicable here), or the call
// is interrupted by a signal handler (explicitly retried in the loop above),
// or the timeout expires.
t.Errorf("Select: slept for %v, expected %v", took, dur)
} else {
t.Logf("Select: slept for %v, requested %v", took, dur)
}
}
})
t.Run("should return 1 if a given FD has data", func(t *testing.T) {
rr, ww, err := os.Pipe()
require.NoError(t, err)
defer rr.Close()
defer ww.Close()
_, err = ww.Write([]byte("TEST"))
require.NoError(t, err)
rFdSet := &FdSet{}
fd := int(rr.Fd())
rFdSet.Set(fd)
for {
n, err := Select(fd+1, rFdSet, nil, nil, nil)
if runtime.GOOS == "windows" {
// Not implemented for fds != wasiFdStdin
require.ErrorIs(t, err, syscall.ENOSYS)
require.Equal(t, -1, n)
break
}
if err == syscall.EINTR {
t.Log("Select interrupted")
continue
}
require.NoError(t, err)
require.Equal(t, 1, n)
break
}
})
}