mirror of
https://github.com/sipeed/picoclaw.git
synced 2026-06-12 18:08:54 +00:00
363 lines
10 KiB
Go
363 lines
10 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/sipeed/picoclaw/pkg/netbind"
|
|
"github.com/sipeed/picoclaw/web/backend/launcherconfig"
|
|
)
|
|
|
|
func TestShouldEnableLauncherFileLogging(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
enableConsole bool
|
|
debug bool
|
|
want bool
|
|
}{
|
|
{name: "gui mode", enableConsole: false, debug: false, want: true},
|
|
{name: "console mode", enableConsole: true, debug: false, want: false},
|
|
{name: "debug gui mode", enableConsole: false, debug: true, want: true},
|
|
{name: "debug console mode", enableConsole: true, debug: true, want: true},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
if got := shouldEnableLauncherFileLogging(tt.enableConsole, tt.debug); got != tt.want {
|
|
t.Fatalf(
|
|
"shouldEnableLauncherFileLogging(%t, %t) = %t, want %t",
|
|
tt.enableConsole,
|
|
tt.debug,
|
|
got,
|
|
tt.want,
|
|
)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestDashboardTokenConfigHelpPath(t *testing.T) {
|
|
const launcherPath = "/tmp/launcher-config.json"
|
|
|
|
tests := []struct {
|
|
name string
|
|
source launcherconfig.DashboardTokenSource
|
|
want string
|
|
}{
|
|
{name: "env token does not expose config path", source: launcherconfig.DashboardTokenSourceEnv, want: ""},
|
|
{name: "config token exposes config path", source: launcherconfig.DashboardTokenSourceConfig, want: launcherPath},
|
|
{name: "random token does not expose config path", source: launcherconfig.DashboardTokenSourceRandom, want: ""},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
if got := dashboardTokenConfigHelpPath(tt.source, launcherPath); got != tt.want {
|
|
t.Fatalf("dashboardTokenConfigHelpPath(%q, %q) = %q, want %q", tt.source, launcherPath, got, tt.want)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestMaskSecret(t *testing.T) {
|
|
tests := []struct {
|
|
input string
|
|
want string
|
|
}{
|
|
{"sdhjflsjdflksdf", "sdh**********ksdf"},
|
|
{"abcdefghijklmnopqrstuvwxyz", "abc**********wxyz"},
|
|
{"abcdefghijkl", "abc**********ijkl"},
|
|
{"abcdefgh", "abc**********"},
|
|
{"abcdefghijk", "abc**********"},
|
|
{"abcdefg", "abc**********"},
|
|
{"abcd", "abc**********"},
|
|
{"abc", "**********"},
|
|
{"", "**********"},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
if got := maskSecret(tt.input); got != tt.want {
|
|
t.Errorf("maskSecret(%q) = %q, want %q", tt.input, got, tt.want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestResolveLauncherHostInput(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
flagHost string
|
|
explicitFlag bool
|
|
envHost string
|
|
wantHost string
|
|
wantActive bool
|
|
wantErr bool
|
|
}{
|
|
{name: "flag host wins", flagHost: "127.0.0.1", explicitFlag: true, envHost: "::", wantHost: "127.0.0.1", wantActive: true},
|
|
{name: "env host used when flag absent", envHost: "127.0.0.1,::1", wantHost: "127.0.0.1,::1", wantActive: true},
|
|
{name: "blank env ignored", envHost: " ", wantHost: "", wantActive: false},
|
|
{name: "invalid flag rejected", flagHost: "127.0.0.1, ", explicitFlag: true, wantErr: true},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
gotHost, gotActive, err := resolveLauncherHostInput(tt.flagHost, tt.explicitFlag, tt.envHost)
|
|
if (err != nil) != tt.wantErr {
|
|
t.Fatalf("resolveLauncherHostInput() err = %v, wantErr %t", err, tt.wantErr)
|
|
}
|
|
if tt.wantErr {
|
|
return
|
|
}
|
|
if gotHost != tt.wantHost {
|
|
t.Fatalf("resolveLauncherHostInput() host = %q, want %q", gotHost, tt.wantHost)
|
|
}
|
|
if gotActive != tt.wantActive {
|
|
t.Fatalf("resolveLauncherHostInput() active = %t, want %t", gotActive, tt.wantActive)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestLauncherConsoleHosts(t *testing.T) {
|
|
t.Run("default loopback shows localhost only", func(t *testing.T) {
|
|
hosts := launcherConsoleHostsWithLocalAddrs(
|
|
"",
|
|
false,
|
|
[]string{"192.168.1.2", "10.0.0.8"},
|
|
[]string{"2001:db8::1", "2001:db8::2"},
|
|
)
|
|
want := []string{"localhost"}
|
|
if strings.Join(hosts, ",") != strings.Join(want, ",") {
|
|
t.Fatalf("hosts = %#v, want %#v", hosts, want)
|
|
}
|
|
})
|
|
|
|
t.Run("explicit loopback hosts collapse to localhost", func(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
hostInput string
|
|
}{
|
|
{name: "ipv6 loopback", hostInput: "::1"},
|
|
{name: "ipv4 loopback", hostInput: "127.0.0.1"},
|
|
{name: "localhost", hostInput: "localhost"},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
hosts := launcherConsoleHostsWithLocalAddrs(
|
|
tt.hostInput,
|
|
false,
|
|
[]string{"192.168.1.2", "10.0.0.8"},
|
|
[]string{"2001:db8::1", "2001:db8::2"},
|
|
)
|
|
want := []string{"localhost"}
|
|
if strings.Join(hosts, ",") != strings.Join(want, ",") {
|
|
t.Fatalf("hosts = %#v, want %#v", hosts, want)
|
|
}
|
|
})
|
|
}
|
|
})
|
|
|
|
t.Run("public wildcard shows localhost then ipv6 and ipv4", func(t *testing.T) {
|
|
hosts := launcherConsoleHostsWithLocalAddrs(
|
|
"",
|
|
true,
|
|
[]string{"192.168.1.2", "10.0.0.8"},
|
|
[]string{"2001:db8::1", "2001:db8::2"},
|
|
)
|
|
want := []string{"localhost", "2001:db8::1", "2001:db8::2", "192.168.1.2", "10.0.0.8"}
|
|
if strings.Join(hosts, ",") != strings.Join(want, ",") {
|
|
t.Fatalf("hosts = %#v, want %#v", hosts, want)
|
|
}
|
|
})
|
|
|
|
t.Run("explicit ipv6 any shows localhost then ipv6 variants", func(t *testing.T) {
|
|
hosts := launcherConsoleHostsWithLocalAddrs(
|
|
"::",
|
|
false,
|
|
[]string{"192.168.1.2", "10.0.0.8"},
|
|
[]string{"2001:db8::1", "2001:db8::2"},
|
|
)
|
|
want := []string{"localhost", "2001:db8::1", "2001:db8::2"}
|
|
if strings.Join(hosts, ",") != strings.Join(want, ",") {
|
|
t.Fatalf("hosts = %#v, want %#v", hosts, want)
|
|
}
|
|
|
|
for _, host := range hosts {
|
|
if host == "::1" || host == "127.0.0.1" || strings.HasPrefix(strings.ToLower(host), "fe80:") {
|
|
t.Fatalf("hosts = %#v, loopback IPs must not be displayed", hosts)
|
|
}
|
|
}
|
|
})
|
|
|
|
t.Run("explicit ipv4 any shows localhost then lan ipv4", func(t *testing.T) {
|
|
hosts := launcherConsoleHostsWithLocalAddrs(
|
|
"0.0.0.0",
|
|
false,
|
|
[]string{"192.168.1.2", "10.0.0.8"},
|
|
[]string{"2001:db8::1", "2001:db8::2"},
|
|
)
|
|
want := []string{"localhost", "192.168.1.2", "10.0.0.8"}
|
|
if strings.Join(hosts, ",") != strings.Join(want, ",") {
|
|
t.Fatalf("hosts = %#v, want %#v", hosts, want)
|
|
}
|
|
})
|
|
|
|
t.Run("explicit multi-address binding shows all exact ipv4 and global ipv6 addresses", func(t *testing.T) {
|
|
hosts := launcherConsoleHostsWithLocalAddrs(
|
|
"192.168.1.2,10.0.0.8,2001:db8::1,2001:db8::2,fe80::1",
|
|
false,
|
|
[]string{"192.168.1.2", "10.0.0.8"},
|
|
[]string{"2001:db8::1", "2001:db8::2"},
|
|
)
|
|
want := []string{"localhost", "192.168.1.2", "10.0.0.8", "2001:db8::1", "2001:db8::2"}
|
|
if strings.Join(hosts, ",") != strings.Join(want, ",") {
|
|
t.Fatalf("hosts = %#v, want %#v", hosts, want)
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestWildcardAdvertiseIP(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
bindHosts []string
|
|
ipv4 string
|
|
ipv6 string
|
|
want string
|
|
}{
|
|
{name: "ipv4 wildcard prefers ipv6 when available", bindHosts: []string{"0.0.0.0"}, ipv4: "192.168.1.2", ipv6: "2001:db8::1", want: "2001:db8::1"},
|
|
{name: "ipv6 wildcard uses ipv6", bindHosts: []string{"::"}, ipv4: "192.168.1.2", ipv6: "2001:db8::1", want: "2001:db8::1"},
|
|
{name: "ipv6 wildcard falls back to ipv4", bindHosts: []string{"::"}, ipv4: "192.168.1.2", ipv6: "", want: "192.168.1.2"},
|
|
{name: "non wildcard does not advertise", bindHosts: []string{"127.0.0.1"}, ipv4: "192.168.1.2", ipv6: "2001:db8::1", want: ""},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
if got := wildcardAdvertiseIP(tt.bindHosts, tt.ipv4, tt.ipv6); got != tt.want {
|
|
t.Fatalf("wildcardAdvertiseIP(%#v, %q, %q) = %q, want %q", tt.bindHosts, tt.ipv4, tt.ipv6, got, tt.want)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestOpenLauncherListeners_HonorsIPv6OnlyHost(t *testing.T) {
|
|
hasIPv4, hasIPv6 := netbind.DetectIPFamilies()
|
|
if !hasIPv6 {
|
|
t.Skip("IPv6 is unavailable in this environment")
|
|
}
|
|
|
|
result, err := openLauncherListeners("::", false, "0")
|
|
if err != nil {
|
|
t.Fatalf("openLauncherListeners() error = %v", err)
|
|
}
|
|
startLauncherTestHTTPServer(t, result.Listeners)
|
|
port := mustAtoi(t, result.Port)
|
|
|
|
requireLauncherHTTPReachable(t, "::1", port)
|
|
if hasIPv4 {
|
|
requireLauncherHTTPUnreachable(t, "127.0.0.1", port)
|
|
}
|
|
}
|
|
|
|
func TestOpenLauncherListeners_SupportsExplicitMultiHost(t *testing.T) {
|
|
hasIPv4, hasIPv6 := netbind.DetectIPFamilies()
|
|
if !hasIPv4 || !hasIPv6 {
|
|
t.Skip("dual-stack loopback is unavailable in this environment")
|
|
}
|
|
|
|
result, err := openLauncherListeners("127.0.0.1,::1", false, "0")
|
|
if err != nil {
|
|
t.Fatalf("openLauncherListeners() error = %v", err)
|
|
}
|
|
startLauncherTestHTTPServer(t, result.Listeners)
|
|
port := mustAtoi(t, result.Port)
|
|
|
|
requireLauncherHTTPReachable(t, "127.0.0.1", port)
|
|
requireLauncherHTTPReachable(t, "::1", port)
|
|
}
|
|
|
|
func startLauncherTestHTTPServer(t *testing.T, listeners []net.Listener) {
|
|
t.Helper()
|
|
|
|
server := &http.Server{
|
|
Handler: http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) {
|
|
_, _ = io.WriteString(w, "ok")
|
|
}),
|
|
}
|
|
|
|
errCh := make(chan error, len(listeners))
|
|
for _, listener := range listeners {
|
|
ln := listener
|
|
go func() {
|
|
errCh <- server.Serve(ln)
|
|
}()
|
|
}
|
|
|
|
t.Cleanup(func() {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
|
|
defer cancel()
|
|
_ = server.Shutdown(ctx)
|
|
for range listeners {
|
|
err := <-errCh
|
|
if err != nil && !errors.Is(err, http.ErrServerClosed) {
|
|
t.Fatalf("server.Serve() error = %v", err)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
|
|
func requireLauncherHTTPReachable(t *testing.T, host string, port int) {
|
|
t.Helper()
|
|
deadline := time.Now().Add(2 * time.Second)
|
|
for {
|
|
err := launcherHTTPGet(host, port)
|
|
if err == nil {
|
|
return
|
|
}
|
|
if time.Now().After(deadline) {
|
|
t.Fatalf("expected %s:%d to be reachable: %v", host, port, err)
|
|
}
|
|
time.Sleep(50 * time.Millisecond)
|
|
}
|
|
}
|
|
|
|
func requireLauncherHTTPUnreachable(t *testing.T, host string, port int) {
|
|
t.Helper()
|
|
if err := launcherHTTPGet(host, port); err == nil {
|
|
t.Fatalf("expected %s:%d to be unreachable", host, port)
|
|
}
|
|
}
|
|
|
|
func launcherHTTPGet(host string, port int) error {
|
|
client := &http.Client{
|
|
Timeout: 300 * time.Millisecond,
|
|
Transport: &http.Transport{
|
|
Proxy: nil,
|
|
},
|
|
}
|
|
|
|
resp, err := client.Get("http://" + net.JoinHostPort(host, strconv.Itoa(port)))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer resp.Body.Close()
|
|
if resp.StatusCode != http.StatusOK {
|
|
return errors.New(resp.Status)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func mustAtoi(t *testing.T, value string) int {
|
|
t.Helper()
|
|
n, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
t.Fatalf("Atoi(%q) error = %v", value, err)
|
|
}
|
|
return n
|
|
}
|