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

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

View File

@@ -24,18 +24,51 @@ import (
"log"
"os"
"os/exec"
"path/filepath"
)
func cmpTest(dir, pkgPath, llApp string, runArgs []string) {
var goOut, goErr bytes.Buffer
func cmpTest(dir, pkgPath, llApp string, genExpect bool, runArgs []string) {
var llgoOut, llgoErr bytes.Buffer
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)
checkEqual("output", llgoOut.Bytes(), goOut.Bytes())
checkEqual("stderr", llgoErr.Bytes(), goErr.Bytes())
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) {
if llgoRunErr == goRunErr {
return