2023-12-26 16:03:45 -08:00
|
|
|
package lifecycle
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"log/slog"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
2024-03-14 10:24:13 -07:00
|
|
|
"syscall"
|
2023-12-26 16:03:45 -08:00
|
|
|
"time"
|
|
|
|
|
2024-03-26 13:04:17 -07:00
|
|
|
"github.com/ollama/ollama/api"
|
2023-12-26 16:03:45 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func getCLIFullPath(command string) string {
|
|
|
|
cmdPath := ""
|
|
|
|
appExe, err := os.Executable()
|
|
|
|
if err == nil {
|
|
|
|
cmdPath = filepath.Join(filepath.Dir(appExe), command)
|
|
|
|
_, err := os.Stat(cmdPath)
|
|
|
|
if err == nil {
|
|
|
|
return cmdPath
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cmdPath, err = exec.LookPath(command)
|
|
|
|
if err == nil {
|
|
|
|
_, err := os.Stat(cmdPath)
|
|
|
|
if err == nil {
|
|
|
|
return cmdPath
|
|
|
|
}
|
|
|
|
}
|
2024-02-16 15:33:16 -08:00
|
|
|
pwd, err := os.Getwd()
|
2023-12-26 16:03:45 -08:00
|
|
|
if err == nil {
|
2024-02-16 15:33:16 -08:00
|
|
|
cmdPath = filepath.Join(pwd, command)
|
|
|
|
_, err = os.Stat(cmdPath)
|
|
|
|
if err == nil {
|
|
|
|
return cmdPath
|
|
|
|
}
|
2023-12-26 16:03:45 -08:00
|
|
|
}
|
2024-02-16 15:33:16 -08:00
|
|
|
|
2023-12-26 16:03:45 -08:00
|
|
|
return command
|
|
|
|
}
|
|
|
|
|
|
|
|
func SpawnServer(ctx context.Context, command string) (chan int, error) {
|
|
|
|
done := make(chan int)
|
|
|
|
|
|
|
|
logDir := filepath.Dir(ServerLogFile)
|
|
|
|
_, err := os.Stat(logDir)
|
|
|
|
if errors.Is(err, os.ErrNotExist) {
|
|
|
|
if err := os.MkdirAll(logDir, 0o755); err != nil {
|
|
|
|
return done, fmt.Errorf("create ollama server log dir %s: %v", logDir, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd := getCmd(ctx, getCLIFullPath(command))
|
|
|
|
// send stdout and stderr to a file
|
|
|
|
stdout, err := cmd.StdoutPipe()
|
|
|
|
if err != nil {
|
|
|
|
return done, fmt.Errorf("failed to spawn server stdout pipe %s", err)
|
|
|
|
}
|
|
|
|
stderr, err := cmd.StderrPipe()
|
|
|
|
if err != nil {
|
|
|
|
return done, fmt.Errorf("failed to spawn server stderr pipe %s", err)
|
|
|
|
}
|
|
|
|
stdin, err := cmd.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
return done, fmt.Errorf("failed to spawn server stdin pipe %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO - rotation
|
|
|
|
logFile, err := os.OpenFile(ServerLogFile, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0755)
|
|
|
|
if err != nil {
|
|
|
|
return done, fmt.Errorf("failed to create server log %w", err)
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
defer logFile.Close()
|
|
|
|
io.Copy(logFile, stdout) //nolint:errcheck
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
defer logFile.Close()
|
|
|
|
io.Copy(logFile, stderr) //nolint:errcheck
|
|
|
|
}()
|
|
|
|
|
2024-03-14 10:24:13 -07:00
|
|
|
// Re-wire context done behavior to attempt a graceful shutdown of the server
|
|
|
|
cmd.Cancel = func() error {
|
|
|
|
if cmd.Process != nil {
|
|
|
|
cmd.Process.Signal(os.Interrupt) //nolint:errcheck
|
|
|
|
tick := time.NewTicker(10 * time.Millisecond)
|
|
|
|
defer tick.Stop()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-tick.C:
|
|
|
|
// OS agnostic "is it still running"
|
|
|
|
if proc, err := os.FindProcess(int(cmd.Process.Pid)); err != nil || errors.Is(proc.Signal(syscall.Signal(0)), os.ErrProcessDone) {
|
|
|
|
return nil //nolint:nilerr
|
|
|
|
}
|
|
|
|
case <-time.After(5 * time.Second):
|
|
|
|
slog.Warn("graceful server shutdown timeout, killing", "pid", cmd.Process.Pid)
|
|
|
|
cmd.Process.Kill() //nolint:errcheck
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-12-26 16:03:45 -08:00
|
|
|
// run the command and wait for it to finish
|
|
|
|
if err := cmd.Start(); err != nil {
|
|
|
|
return done, fmt.Errorf("failed to start server %w", err)
|
|
|
|
}
|
|
|
|
if cmd.Process != nil {
|
|
|
|
slog.Info(fmt.Sprintf("started ollama server with pid %d", cmd.Process.Pid))
|
|
|
|
}
|
|
|
|
slog.Info(fmt.Sprintf("ollama server logs %s", ServerLogFile))
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
// Keep the server running unless we're shuttind down the app
|
|
|
|
crashCount := 0
|
|
|
|
for {
|
|
|
|
cmd.Wait() //nolint:errcheck
|
|
|
|
stdin.Close()
|
|
|
|
var code int
|
|
|
|
if cmd.ProcessState != nil {
|
|
|
|
code = cmd.ProcessState.ExitCode()
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
2024-03-14 10:24:13 -07:00
|
|
|
slog.Info(fmt.Sprintf("server shutdown with exit code %d", code))
|
2023-12-26 16:03:45 -08:00
|
|
|
done <- code
|
|
|
|
return
|
|
|
|
default:
|
|
|
|
crashCount++
|
|
|
|
slog.Warn(fmt.Sprintf("server crash %d - exit code %d - respawning", crashCount, code))
|
|
|
|
time.Sleep(500 * time.Millisecond)
|
|
|
|
if err := cmd.Start(); err != nil {
|
|
|
|
slog.Error(fmt.Sprintf("failed to restart server %s", err))
|
|
|
|
// Keep trying, but back off if we keep failing
|
|
|
|
time.Sleep(time.Duration(crashCount) * time.Second)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return done, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func IsServerRunning(ctx context.Context) bool {
|
|
|
|
client, err := api.ClientFromEnvironment()
|
|
|
|
if err != nil {
|
|
|
|
slog.Info("unable to connect to server")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
err = client.Heartbeat(ctx)
|
|
|
|
if err != nil {
|
|
|
|
slog.Debug(fmt.Sprintf("heartbeat from server: %s", err))
|
|
|
|
slog.Info("unable to connect to server")
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|