JOJ3/internal/parser/log/parser.go
张泊明518370910136 91a452e23d
All checks were successful
submodules sync / sync (push) Successful in 48s
build / build (push) Successful in 3m0s
build / trigger-build-image (push) Successful in 14s
feat(parser/log): log string line by line
2025-11-27 22:47:52 -08:00

69 lines
1.6 KiB
Go

package log
import (
"context"
"encoding/json"
"fmt"
"log/slog"
"strings"
"github.com/joint-online-judge/JOJ3/internal/stage"
)
func (*Log) parse(executorResult stage.ExecutorResult, conf Conf) stage.ParserResult {
content, ok := executorResult.Files[conf.Filename]
if !ok {
slog.Error("file not found for log parser", "filename", conf.Filename)
return stage.ParserResult{
Score: 0,
Comment: fmt.Sprintf("log parser: file %s not found", conf.Filename),
}
}
contentBytes := []byte(content)
var data map[string]any
if err := json.Unmarshal(contentBytes, &data); err != nil {
// Not a valid json or failed to unmarshal, log as raw string line by line.
for line := range strings.SplitSeq(content, "\n") {
if strings.TrimSpace(line) != "" {
slog.Default().Log(
context.Background(),
slog.Level(conf.Level),
conf.Msg,
"line",
line,
)
}
}
} else {
// Valid json, log as key-value pairs.
args := make([]any, 0, len(data)*2)
for key, value := range data {
args = append(args, key, value)
}
slog.Default().Log(
context.Background(),
slog.Level(conf.Level),
conf.Msg,
args...,
)
}
return stage.ParserResult{
Score: 0,
Comment: "",
}
}
func (p *Log) Run(results []stage.ExecutorResult, confAny any) (
[]stage.ParserResult, bool, error,
) {
conf, err := stage.DecodeConf[Conf](confAny)
if err != nil {
return nil, true, err
}
res := make([]stage.ParserResult, 0, len(results))
for _, result := range results {
res = append(res, p.parse(result, *conf))
}
return res, false, nil
}