woj-server/internal/service/runner/service.go

88 lines
2.1 KiB
Go
Raw Normal View History

package runner
import (
2024-01-27 17:37:27 +08:00
"context"
2023-07-14 21:47:11 +08:00
"git.0x7f.app/WOJ/woj-server/internal/e"
2023-07-15 16:19:49 +08:00
"git.0x7f.app/WOJ/woj-server/internal/misc/config"
"git.0x7f.app/WOJ/woj-server/internal/misc/log"
2024-01-06 15:06:12 +08:00
"git.0x7f.app/WOJ/woj-server/pkg/pool"
"git.0x7f.app/WOJ/woj-server/pkg/utils"
2024-01-27 17:37:27 +08:00
"github.com/containerd/containerd"
"github.com/containerd/containerd/namespaces"
2023-07-15 16:19:49 +08:00
"github.com/samber/do"
"go.uber.org/zap"
2024-01-06 15:06:12 +08:00
"runtime"
2024-01-27 17:37:27 +08:00
"sync/atomic"
)
var _ Service = (*service)(nil)
type Service interface {
// EnsureDeps build docker images
2024-01-28 21:51:16 +08:00
EnsureDeps() e.Status
// NewProblem = Download + Parse + Prebuild
NewProblem(meta *JudgeMeta, url string, force bool) (*Config, e.Status)
// Compile compile user submission
Compile(meta *JudgeMeta) (*JudgeStatus, e.Status)
// RunAndJudge execute user program
RunAndJudge(meta *JudgeMeta) (*JudgeStatus, int32, e.Status)
// ParseConfig parse config file
ParseConfig(meta *JudgeMeta, skipCheck bool) (*Config, error)
// ProblemExists check if problem exists
ProblemExists(meta *JudgeMeta) bool
2023-07-15 16:19:49 +08:00
HealthCheck() error
2024-01-06 19:21:37 +08:00
Shutdown() error
2023-07-15 16:19:49 +08:00
}
func NewService(i *do.Injector) (Service, error) {
2024-01-06 15:06:12 +08:00
concurrency := utils.If(runtime.NumCPU() > 1, runtime.NumCPU()-1, 1)
2024-01-27 17:37:27 +08:00
cfg := do.MustInvoke[config.Service](i).GetConfig()
2024-01-06 15:06:12 +08:00
2024-01-06 19:21:37 +08:00
srv := &service{
2023-07-15 16:19:49 +08:00
log: do.MustInvoke[log.Service](i).GetLogger("runner"),
2024-01-06 15:06:12 +08:00
pool: pool.NewTaskPool(concurrency, concurrency),
2024-01-27 17:37:27 +08:00
verbose: cfg.Development,
2024-01-06 19:21:37 +08:00
}
2024-01-27 17:37:27 +08:00
var err error
srv.container.client, err = containerd.New(cfg.Runner.Address)
if err != nil {
srv.log.Error("failed to connect to containerd", zap.Error(err))
return nil, err
}
srv.container.ctx = namespaces.WithNamespace(context.Background(), "woj")
srv.container.count.Store(0)
srv.pool.Start()
2024-01-06 19:21:37 +08:00
return srv, nil
}
type service struct {
2024-01-27 17:37:27 +08:00
log *zap.Logger
pool *pool.TaskPool
container struct {
client *containerd.Client
ctx context.Context
count atomic.Uint64
}
2022-10-28 21:57:28 +08:00
verbose bool
}
2023-07-15 16:19:49 +08:00
func (s *service) HealthCheck() error {
return nil
}
2024-01-06 19:21:37 +08:00
func (s *service) Shutdown() error {
s.pool.Stop()
2024-01-27 17:37:27 +08:00
if s.container.client != nil {
// TODO: wait and kill all containers
_ = s.container.client.Close()
}
2024-01-06 19:21:37 +08:00
return nil
}