cmptest: add support for comparison with llgo.expect files

Fixes #671
This commit is contained in:
Aofei Sheng
2024-08-12 13:43:44 +08:00
parent 321766fd46
commit 200fe07473
3 changed files with 49 additions and 11 deletions

View File

@@ -37,8 +37,8 @@ var Cmd = &base.Command{
// llgo cmptest // llgo cmptest
var CmpTestCmd = &base.Command{ var CmpTestCmd = &base.Command{
UsageLine: "llgo cmptest [build flags] package [arguments...]", UsageLine: "llgo cmptest [-genexpect] [build flags] package [arguments...]",
Short: "Compile programs by llgo and go, run them and do comparative tests (stdout/stderr/exitcode)", Short: "Compile and run with llgo, compare result (stdout/stderr/exitcode) with go or llgo.expect; generate llgo.expect file if -genexpect is specified",
} }
func init() { func init() {
@@ -55,9 +55,13 @@ func runCmpTest(cmd *base.Command, args []string) {
} }
func runCmdEx(cmd *base.Command, args []string, mode build.Mode) { func runCmdEx(cmd *base.Command, args []string, mode build.Mode) {
conf := build.NewDefaultConf(mode)
if mode == build.ModeCmpTest && len(args) > 0 && args[0] == "-genexpect" {
conf.GenExpect = true
args = args[1:]
}
args, runArgs, err := parseRunArgs(args) args, runArgs, err := parseRunArgs(args)
check(err) check(err)
conf := build.NewDefaultConf(mode)
conf.RunArgs = runArgs conf.RunArgs = runArgs
build.Do(args, conf) build.Do(args, conf)
} }

View File

@@ -64,11 +64,12 @@ func needLLFile(mode Mode) bool {
} }
type Config struct { type Config struct {
BinPath string BinPath string
AppExt string // ".exe" on Windows, empty on Unix AppExt string // ".exe" on Windows, empty on Unix
OutFile string // only valid for ModeBuild when len(pkgs) == 1 OutFile string // only valid for ModeBuild when len(pkgs) == 1
RunArgs []string // only valid for ModeRun RunArgs []string // only valid for ModeRun
Mode Mode GenExpect bool // only valid for ModeCmpTest
Mode Mode
} }
func NewDefaultConf(mode Mode) *Config { func NewDefaultConf(mode Mode) *Config {
@@ -462,7 +463,7 @@ func linkMainPkg(ctx *context, pkg *packages.Package, pkgs []*aPackage, llFiles
os.Exit(s.ExitCode()) os.Exit(s.ExitCode())
} }
case ModeCmpTest: case ModeCmpTest:
cmpTest("", pkgPath, app, conf.RunArgs) cmpTest(filepath.Dir(pkg.GoFiles[0]), pkgPath, app, conf.GenExpect, conf.RunArgs)
} }
return return
} }

View File

@@ -24,18 +24,51 @@ import (
"log" "log"
"os" "os"
"os/exec" "os/exec"
"path/filepath"
) )
func cmpTest(dir, pkgPath, llApp string, runArgs []string) { func cmpTest(dir, pkgPath, llApp string, genExpect bool, runArgs []string) {
var goOut, goErr bytes.Buffer
var llgoOut, llgoErr bytes.Buffer var llgoOut, llgoErr bytes.Buffer
var llgoRunErr = runApp(runArgs, dir, &llgoOut, &llgoErr, llApp) var llgoRunErr = runApp(runArgs, dir, &llgoOut, &llgoErr, llApp)
llgoExpect := formatExpect(llgoOut.Bytes(), llgoErr.Bytes(), llgoRunErr)
llgoExpectFile := filepath.Join(dir, "llgo.expect")
if genExpect {
if _, err := os.Stat(llgoExpectFile); !errors.Is(err, os.ErrNotExist) {
fatal(fmt.Errorf("llgo.expect file already exists: %s", llgoExpectFile))
}
if err := os.WriteFile(llgoExpectFile, llgoExpect, 0644); err != nil {
fatal(err)
}
return
}
if b, err := os.ReadFile(llgoExpectFile); err == nil {
checkEqual("llgo.expect", llgoExpect, b)
return
} else if !errors.Is(err, os.ErrNotExist) {
fatal(err)
}
var goOut, goErr bytes.Buffer
var goRunErr = runApp(runArgs, dir, &goOut, &goErr, "go", "run", pkgPath) var goRunErr = runApp(runArgs, dir, &goOut, &goErr, "go", "run", pkgPath)
checkEqual("output", llgoOut.Bytes(), goOut.Bytes()) checkEqual("output", llgoOut.Bytes(), goOut.Bytes())
checkEqual("stderr", llgoErr.Bytes(), goErr.Bytes()) checkEqual("stderr", llgoErr.Bytes(), goErr.Bytes())
checkEqualRunErr(llgoRunErr, goRunErr) checkEqualRunErr(llgoRunErr, goRunErr)
} }
func formatExpect(stdout, stderr []byte, runErr error) []byte {
var exitCode int
if runErr != nil {
if ee, ok := runErr.(*exec.ExitError); ok {
exitCode = ee.ExitCode()
} else { // This should never happen, but just in case.
exitCode = 255
}
}
return []byte(fmt.Sprintf("#stdout\n%s\n#stderr\n%s\n#exit %d\n", stdout, stderr, exitCode))
}
func checkEqualRunErr(llgoRunErr, goRunErr error) { func checkEqualRunErr(llgoRunErr, goRunErr error) {
if llgoRunErr == goRunErr { if llgoRunErr == goRunErr {
return return