@@ -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)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
Reference in New Issue
Block a user