2023-08-18 23:49:33 +00:00
|
|
|
package model
|
|
|
|
|
|
|
|
import (
|
2024-04-29 13:11:42 +00:00
|
|
|
"errors"
|
2023-08-18 23:49:33 +00:00
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
2024-08-23 22:27:14 +00:00
|
|
|
"path/filepath"
|
2023-08-18 23:49:33 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/hpcloud/tail"
|
|
|
|
process "github.com/mudler/go-processmanager"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
)
|
|
|
|
|
2024-04-29 13:11:42 +00:00
|
|
|
func (ml *ModelLoader) StopAllExcept(s string) error {
|
|
|
|
return ml.StopGRPC(func(id string, p *process.Process) bool {
|
2023-08-18 23:49:33 +00:00
|
|
|
if id != s {
|
2023-11-26 17:36:23 +00:00
|
|
|
for ml.models[id].GRPC(false, ml.wd).IsBusy() {
|
2023-08-18 23:49:33 +00:00
|
|
|
log.Debug().Msgf("%s busy. Waiting.", id)
|
|
|
|
time.Sleep(2 * time.Second)
|
|
|
|
}
|
|
|
|
log.Debug().Msgf("[single-backend] Stopping %s", id)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ml *ModelLoader) deleteProcess(s string) error {
|
2024-05-21 12:33:47 +00:00
|
|
|
if _, exists := ml.grpcProcesses[s]; exists {
|
|
|
|
if err := ml.grpcProcesses[s].Stop(); err != nil {
|
2024-08-25 12:36:09 +00:00
|
|
|
log.Error().Err(err).Msgf("(deleteProcess) error while deleting grpc process %s", s)
|
2024-05-21 12:33:47 +00:00
|
|
|
}
|
2023-08-18 23:49:33 +00:00
|
|
|
}
|
|
|
|
delete(ml.grpcProcesses, s)
|
|
|
|
delete(ml.models, s)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type GRPCProcessFilter = func(id string, p *process.Process) bool
|
|
|
|
|
|
|
|
func includeAllProcesses(_ string, _ *process.Process) bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2024-04-29 13:11:42 +00:00
|
|
|
func (ml *ModelLoader) StopGRPC(filter GRPCProcessFilter) error {
|
|
|
|
var err error = nil
|
2023-08-18 23:49:33 +00:00
|
|
|
for k, p := range ml.grpcProcesses {
|
|
|
|
if filter(k, p) {
|
2024-04-29 13:11:42 +00:00
|
|
|
e := ml.deleteProcess(k)
|
|
|
|
err = errors.Join(err, e)
|
2023-08-18 23:49:33 +00:00
|
|
|
}
|
|
|
|
}
|
2024-04-29 13:11:42 +00:00
|
|
|
return err
|
2023-08-18 23:49:33 +00:00
|
|
|
}
|
|
|
|
|
2024-04-29 13:11:42 +00:00
|
|
|
func (ml *ModelLoader) StopAllGRPC() error {
|
|
|
|
return ml.StopGRPC(includeAllProcesses)
|
2023-08-18 23:49:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ml *ModelLoader) GetGRPCPID(id string) (int, error) {
|
|
|
|
p, exists := ml.grpcProcesses[id]
|
|
|
|
if !exists {
|
|
|
|
return -1, fmt.Errorf("no grpc backend found for %s", id)
|
|
|
|
}
|
|
|
|
return strconv.Atoi(p.PID)
|
|
|
|
}
|
|
|
|
|
2024-06-18 20:43:43 +00:00
|
|
|
func (ml *ModelLoader) startProcess(grpcProcess, id string, serverAddress string, args ...string) error {
|
2023-08-18 23:49:33 +00:00
|
|
|
// Make sure the process is executable
|
2024-04-25 22:47:06 +00:00
|
|
|
if err := os.Chmod(grpcProcess, 0700); err != nil {
|
2023-08-18 23:49:33 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug().Msgf("Loading GRPC Process: %s", grpcProcess)
|
|
|
|
|
|
|
|
log.Debug().Msgf("GRPC Service for %s will be running at: '%s'", id, serverAddress)
|
|
|
|
|
2024-08-23 22:27:14 +00:00
|
|
|
workDir, err := filepath.Abs(filepath.Dir(grpcProcess))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-08-18 23:49:33 +00:00
|
|
|
grpcControlProcess := process.New(
|
|
|
|
process.WithTemporaryStateDir(),
|
2024-08-23 22:27:14 +00:00
|
|
|
process.WithName(filepath.Base(grpcProcess)),
|
2024-06-18 20:43:43 +00:00
|
|
|
process.WithArgs(append(args, []string{"--addr", serverAddress}...)...),
|
2023-08-18 23:49:33 +00:00
|
|
|
process.WithEnvironment(os.Environ()...),
|
2024-08-23 22:27:14 +00:00
|
|
|
process.WithWorkDir(workDir),
|
2023-08-18 23:49:33 +00:00
|
|
|
)
|
|
|
|
|
2023-11-26 17:36:23 +00:00
|
|
|
if ml.wd != nil {
|
|
|
|
ml.wd.Add(serverAddress, grpcControlProcess)
|
|
|
|
ml.wd.AddAddressModelMap(serverAddress, id)
|
|
|
|
}
|
|
|
|
|
2023-08-18 23:49:33 +00:00
|
|
|
ml.grpcProcesses[id] = grpcControlProcess
|
|
|
|
|
|
|
|
if err := grpcControlProcess.Run(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug().Msgf("GRPC Service state dir: %s", grpcControlProcess.StateDir())
|
|
|
|
// clean up process
|
|
|
|
go func() {
|
|
|
|
c := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
|
|
|
|
<-c
|
2024-06-24 06:34:36 +00:00
|
|
|
err := grpcControlProcess.Stop()
|
|
|
|
if err != nil {
|
|
|
|
log.Error().Err(err).Msg("error while shutting down grpc process")
|
|
|
|
}
|
2023-08-18 23:49:33 +00:00
|
|
|
}()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
t, err := tail.TailFile(grpcControlProcess.StderrPath(), tail.Config{Follow: true})
|
|
|
|
if err != nil {
|
|
|
|
log.Debug().Msgf("Could not tail stderr")
|
|
|
|
}
|
|
|
|
for line := range t.Lines {
|
|
|
|
log.Debug().Msgf("GRPC(%s): stderr %s", strings.Join([]string{id, serverAddress}, "-"), line.Text)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
|
|
|
t, err := tail.TailFile(grpcControlProcess.StdoutPath(), tail.Config{Follow: true})
|
|
|
|
if err != nil {
|
|
|
|
log.Debug().Msgf("Could not tail stdout")
|
|
|
|
}
|
|
|
|
for line := range t.Lines {
|
|
|
|
log.Debug().Msgf("GRPC(%s): stdout %s", strings.Join([]string{id, serverAddress}, "-"), line.Text)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|