69 lines
1.8 KiB
Go
69 lines
1.8 KiB
Go
package sandbox
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"maps"
|
|
|
|
"github.com/criyle/go-judge/pb"
|
|
"github.com/joint-online-judge/JOJ3/internal/stage"
|
|
"google.golang.org/protobuf/proto"
|
|
)
|
|
|
|
func (e *Sandbox) Run(cmds []stage.Cmd) ([]stage.ExecutorResult, error) {
|
|
var err error
|
|
if e.execClient == nil {
|
|
slog.Debug("create exec client", "server", e.execServer)
|
|
e.execClient, err = createExecClient(e.execServer, e.token)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
// cannot use range loop since we need to change the value
|
|
results := make([]stage.ExecutorResult, len(cmds))
|
|
for i := 0; i < len(cmds); i += 1 {
|
|
cmd := &cmds[i]
|
|
if cmd.CopyIn == nil {
|
|
cmd.CopyIn = make(map[string]stage.CmdFile)
|
|
}
|
|
for k, v := range cmd.CopyInCached {
|
|
if fileID, ok := e.cachedMap[v]; ok {
|
|
cmd.CopyIn[k] = stage.CmdFile{FileID: &fileID}
|
|
}
|
|
}
|
|
// NOTE: send all cmds at once may cause oom, no idea on why
|
|
pbCmd := convertPBCmd([]stage.Cmd{*cmd})
|
|
pbReq := &pb.Request{Cmd: pbCmd}
|
|
slog.Debug("sandbox execute", "i", i, "pbReq size", proto.Size(pbReq))
|
|
pbRet, err := e.execClient.Exec(context.TODO(), pbReq)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if pbRet.Error != "" {
|
|
return nil, fmt.Errorf("sandbox execute error: %s", pbRet.Error)
|
|
}
|
|
if len(pbRet.Results) == 0 {
|
|
return nil, fmt.Errorf("sandbox execute error: no result")
|
|
}
|
|
result := convertPBResult(pbRet.Results)[0]
|
|
slog.Debug("sandbox execute", "i", i, "result", result)
|
|
maps.Copy(e.cachedMap, result.FileIDs)
|
|
results[i] = result
|
|
}
|
|
return results, nil
|
|
}
|
|
|
|
func (e *Sandbox) Cleanup() error {
|
|
for k, fileID := range e.cachedMap {
|
|
_, err := e.execClient.FileDelete(context.TODO(), &pb.FileID{
|
|
FileID: fileID,
|
|
})
|
|
if err != nil {
|
|
slog.Error("sandbox cleanup", "error", err)
|
|
}
|
|
delete(e.cachedMap, k)
|
|
}
|
|
return nil
|
|
}
|