2026-02-19 21:08:07 +00:00
|
|
|
//go:build linux && amd64
|
|
|
|
|
|
|
|
|
|
package rocm
|
|
|
|
|
|
|
|
|
|
import (
|
2026-02-19 21:34:46 +00:00
|
|
|
"context"
|
|
|
|
|
"fmt"
|
2026-02-19 21:12:02 +00:00
|
|
|
"os"
|
2026-02-19 21:08:07 +00:00
|
|
|
"strings"
|
|
|
|
|
"testing"
|
|
|
|
|
|
2026-02-19 21:34:46 +00:00
|
|
|
"forge.lthn.ai/core/go-inference"
|
2026-02-19 21:08:07 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func TestFindLlamaServer_InPATH(t *testing.T) {
|
|
|
|
|
// llama-server is at /usr/local/bin/llama-server on this machine.
|
|
|
|
|
path, err := findLlamaServer()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
assert.Contains(t, path, "llama-server")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestFindLlamaServer_EnvOverride(t *testing.T) {
|
|
|
|
|
t.Setenv("ROCM_LLAMA_SERVER_PATH", "/usr/local/bin/llama-server")
|
|
|
|
|
path, err := findLlamaServer()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
assert.Equal(t, "/usr/local/bin/llama-server", path)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestFindLlamaServer_EnvNotFound(t *testing.T) {
|
|
|
|
|
t.Setenv("ROCM_LLAMA_SERVER_PATH", "/nonexistent/llama-server")
|
|
|
|
|
_, err := findLlamaServer()
|
|
|
|
|
assert.ErrorContains(t, err, "not found")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestFreePort(t *testing.T) {
|
|
|
|
|
port, err := freePort()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
assert.Greater(t, port, 0)
|
|
|
|
|
assert.Less(t, port, 65536)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestFreePort_UniquePerCall(t *testing.T) {
|
|
|
|
|
p1, err := freePort()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
p2, err := freePort()
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
_ = p1
|
|
|
|
|
_ = p2
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestServerEnv_HIPVisibleDevices(t *testing.T) {
|
|
|
|
|
env := serverEnv()
|
|
|
|
|
var hipVals []string
|
|
|
|
|
for _, e := range env {
|
|
|
|
|
if strings.HasPrefix(e, "HIP_VISIBLE_DEVICES=") {
|
|
|
|
|
hipVals = append(hipVals, e)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert.Equal(t, []string{"HIP_VISIBLE_DEVICES=0"}, hipVals)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestServerEnv_FiltersExistingHIP(t *testing.T) {
|
|
|
|
|
t.Setenv("HIP_VISIBLE_DEVICES", "1")
|
|
|
|
|
env := serverEnv()
|
|
|
|
|
var hipVals []string
|
|
|
|
|
for _, e := range env {
|
|
|
|
|
if strings.HasPrefix(e, "HIP_VISIBLE_DEVICES=") {
|
|
|
|
|
hipVals = append(hipVals, e)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
assert.Equal(t, []string{"HIP_VISIBLE_DEVICES=0"}, hipVals)
|
|
|
|
|
}
|
2026-02-19 21:12:02 +00:00
|
|
|
|
|
|
|
|
func TestAvailable(t *testing.T) {
|
|
|
|
|
b := &rocmBackend{}
|
|
|
|
|
if _, err := os.Stat("/dev/kfd"); err != nil {
|
|
|
|
|
t.Skip("no ROCm hardware")
|
|
|
|
|
}
|
|
|
|
|
assert.True(t, b.Available())
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-19 21:34:46 +00:00
|
|
|
|
|
|
|
|
func TestServerAlive_Running(t *testing.T) {
|
|
|
|
|
s := &server{exited: make(chan struct{})}
|
|
|
|
|
assert.True(t, s.alive())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestServerAlive_Exited(t *testing.T) {
|
|
|
|
|
exited := make(chan struct{})
|
|
|
|
|
close(exited)
|
|
|
|
|
s := &server{exited: exited, exitErr: fmt.Errorf("process killed")}
|
|
|
|
|
assert.False(t, s.alive())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestGenerate_ServerDead(t *testing.T) {
|
|
|
|
|
exited := make(chan struct{})
|
|
|
|
|
close(exited)
|
|
|
|
|
s := &server{
|
|
|
|
|
exited: exited,
|
|
|
|
|
exitErr: fmt.Errorf("process killed"),
|
|
|
|
|
}
|
|
|
|
|
m := &rocmModel{srv: s}
|
|
|
|
|
|
|
|
|
|
var count int
|
|
|
|
|
for range m.Generate(context.Background(), "hello") {
|
|
|
|
|
count++
|
|
|
|
|
}
|
|
|
|
|
assert.Equal(t, 0, count)
|
|
|
|
|
assert.ErrorContains(t, m.Err(), "server has exited")
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-19 21:40:05 +00:00
|
|
|
func TestStartServer_RetriesOnProcessExit(t *testing.T) {
|
|
|
|
|
// /bin/false starts successfully but exits immediately with code 1.
|
|
|
|
|
// startServer should retry up to 3 times, then fail.
|
2026-02-19 23:13:19 +00:00
|
|
|
_, err := startServer("/bin/false", "/nonexistent/model.gguf", 999, 0, 0)
|
2026-02-19 21:40:05 +00:00
|
|
|
require.Error(t, err)
|
|
|
|
|
assert.Contains(t, err.Error(), "failed after 3 attempts")
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-19 21:34:46 +00:00
|
|
|
func TestChat_ServerDead(t *testing.T) {
|
|
|
|
|
exited := make(chan struct{})
|
|
|
|
|
close(exited)
|
|
|
|
|
s := &server{
|
|
|
|
|
exited: exited,
|
|
|
|
|
exitErr: fmt.Errorf("process killed"),
|
|
|
|
|
}
|
|
|
|
|
m := &rocmModel{srv: s}
|
|
|
|
|
|
|
|
|
|
msgs := []inference.Message{{Role: "user", Content: "hello"}}
|
|
|
|
|
var count int
|
|
|
|
|
for range m.Chat(context.Background(), msgs) {
|
|
|
|
|
count++
|
|
|
|
|
}
|
|
|
|
|
assert.Equal(t, 0, count)
|
|
|
|
|
assert.ErrorContains(t, m.Err(), "server has exited")
|
|
|
|
|
}
|