Gopher2600/regression/regression.go

476 lines
13 KiB
Go

// This file is part of Gopher2600.
//
// Gopher2600 is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Gopher2600 is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Gopher2600. If not, see <https://www.gnu.org/licenses/>.
package regression
import (
"fmt"
"io"
"os"
"sort"
"strconv"
"strings"
"github.com/jetsetilly/gopher2600/database"
"github.com/jetsetilly/gopher2600/resources"
)
// ansi code for clear line.
const ansiClearLine = "\033[2K"
// the location of the regressionDB file and the location of any regression
// scripts. these should be wrapped by resources.ResourcePath().
const regressionPath = "regression"
const regressionDBFile = "db"
const regressionScripts = "scripts"
// Regressor is the generic entry type in the regressionDB.
type Regressor interface {
database.Entry
// String should return information about the entry in a human readable
// format. by contrast, machine readable representation is returned by the
// Serialise function
String() string
// perform the regression test for the regression type. the newRegression
// flag is for convenience really (or "logical binding", as the structured
// programmers would have it)
//
// message is the string that is to be printed during the regression
//
// returns: success boolean; any failure message (not always appropriate;
// and error state
regress(newRegression bool, output io.Writer, message string) (bool, string, error)
}
// when starting a database session we need to register what entries we will
// find in the database.
func initDBSession(db *database.Session) error {
if err := db.RegisterEntryType(videoEntryType, deserialiseVideoEntry); err != nil {
return err
}
if err := db.RegisterEntryType(playbackEntryType, deserialisePlaybackEntry); err != nil {
return err
}
if err := db.RegisterEntryType(logEntryType, deserialiseLogEntry); err != nil {
return err
}
return nil
}
// RegressList displays all entries in the database.
func RegressList(output io.Writer) error {
if output == nil {
return fmt.Errorf("regression: list: io.Writer should not be nil")
}
dbPth, err := resources.JoinPath(regressionPath, regressionDBFile)
if err != nil {
return fmt.Errorf("regression: list: %w", err)
}
db, err := database.StartSession(dbPth, database.ActivityReading, initDBSession)
if err != nil {
return fmt.Errorf("regression: list: %w", err)
}
defer db.EndSession(false)
return db.ForEach(func(key int, e database.Entry) error {
r, ok := e.(Regressor)
if !ok {
return fmt.Errorf("regression: list: %s does not implement the Regressor interfrace", e.EntryType())
}
output.Write([]byte(fmt.Sprintf("%03d %s\n", key, r.String())))
return nil
})
}
// RegressAdd adds a new regression handler to the database.
func RegressAdd(output io.Writer, reg Regressor) error {
if output == nil {
return fmt.Errorf("regression: add: io.Writer should not be nil")
}
dbPth, err := resources.JoinPath(regressionPath, regressionDBFile)
if err != nil {
return fmt.Errorf("regression: add: %w", err)
}
db, err := database.StartSession(dbPth, database.ActivityCreating, initDBSession)
if err != nil {
return fmt.Errorf("regression: add: %w", err)
}
defer db.EndSession(true)
msg := fmt.Sprintf("adding: %s", reg)
_, _, err = reg.regress(true, output, msg)
if err != nil {
return fmt.Errorf("regression: add: %w", err)
}
output.Write([]byte(ansiClearLine))
output.Write([]byte(fmt.Sprintf("\radded: %s\n", reg)))
return db.Add(reg)
}
// RegressRedux removes and adds an entry using the same parameters.
func RegressRedux(output io.Writer, confirmation io.Reader) error {
if output == nil {
return fmt.Errorf("regression: redux: io.Writer should not be nil")
}
if confirmation == nil {
return fmt.Errorf("regression: redux: io.Reader should not be nil")
}
output.Write([]byte("redux is a dangerous operation. it will rerun all compatible regression entries.\n"))
output.Write([]byte("redux? (y/n): "))
if !confirm(confirmation) {
return nil
}
output.Write([]byte("sure? (y/n): "))
if !confirm(confirmation) {
return nil
}
dbPth, err := resources.JoinPath(regressionPath, regressionDBFile)
if err != nil {
return fmt.Errorf("regression: redux: %w", err)
}
db, err := database.StartSession(dbPth, database.ActivityCreating, initDBSession)
if err != nil {
return fmt.Errorf("regression: redux: %w", err)
}
defer db.EndSession(true)
return db.ForEach(func(key int, e database.Entry) error {
switch reg := e.(type) {
case *VideoRegression:
err = redux(db, output, key, reg)
if err != nil {
return fmt.Errorf("regression: redux: %w", err)
}
case *LogRegression:
err = redux(db, output, key, reg)
if err != nil {
return fmt.Errorf("regression: redux: %w", err)
}
default:
output.Write([]byte(fmt.Sprintf("skipped: %s\n", reg)))
}
return nil
})
}
func redux(db *database.Session, output io.Writer, key int, reg Regressor) error {
err := db.Delete(key)
if err != nil {
return err
}
msg := fmt.Sprintf("reduxing: %s", reg)
_, _, err = reg.regress(true, output, msg)
if err != nil {
return err
}
output.Write([]byte(ansiClearLine))
output.Write([]byte(fmt.Sprintf("\rreduxed: %s\n", reg)))
err = db.Add(reg)
if err != nil {
return err
}
return nil
}
// CleanupScript removes orphaned script files from disk. An orphaned file is
// one that exists on disk but has no reference in the regression database
// file.
func RegressCleanup(output io.Writer, confirmation io.Reader) error {
if output == nil {
return fmt.Errorf("regression: list: io.Writer should not be nil")
}
if confirmation == nil {
return fmt.Errorf("regression: redux: io.Reader should not be nil")
}
output.Write([]byte("cleanup is a dangerous operation. it will delete all orphaned script files.\n"))
output.Write([]byte("cleanup? (y/n): "))
if !confirm(confirmation) {
return nil
}
dbPth, err := resources.JoinPath(regressionPath, regressionDBFile)
if err != nil {
return fmt.Errorf("regression: cleanup: %w", err)
}
db, err := database.StartSession(dbPth, database.ActivityReading, initDBSession)
if err != nil {
return fmt.Errorf("regression: cleanup: %w", err)
}
defer db.EndSession(false)
// gather list of all files referenced
filesReferenced := make([]string, 0)
err = db.ForEach(func(key int, e database.Entry) error {
switch reg := e.(type) {
case *VideoRegression:
if len(strings.TrimSpace(reg.stateFile)) > 0 {
filesReferenced = append(filesReferenced, reg.stateFile)
}
case *PlaybackRegression:
filesReferenced = append(filesReferenced, reg.Script)
case *LogRegression:
// no support required
default:
return fmt.Errorf("not supported (%s)", reg.EntryType())
}
return nil
})
if err != nil {
return fmt.Errorf("regression: cleanup: %w", err)
}
// gather list of files on disk in path
scriptPth, err := resources.JoinPath(regressionPath, regressionScripts)
if err != nil {
return fmt.Errorf("regression: list: %w", err)
}
var filesOnDisk []os.DirEntry
filesOnDisk, err = os.ReadDir(scriptPth)
if err != nil {
return fmt.Errorf("regression: cleanup: %w", err)
}
// prepare statistics
numDeleted := 0
numRemain := len(filesReferenced)
numErrors := 0
defer func() {
output.Write([]byte(fmt.Sprintf("regression cleanup: %d deleted, %d remain, %d errors\n", numDeleted, numRemain, numErrors)))
}()
// delete any files on disk that are not referenced
for _, e := range filesOnDisk {
found := false
n, err := resources.JoinPath(regressionPath, regressionScripts, e.Name())
if err != nil {
return fmt.Errorf("regression: cleanup: %w", err)
}
for _, f := range filesReferenced {
if f == n {
found = true
break // for range filesRefrenced
}
}
if !found {
output.Write([]byte(fmt.Sprintf("delete %s? (y/n): ", n)))
if confirm(confirmation) {
output.Write([]byte(ansiClearLine))
err := os.Remove(n)
if err != nil {
output.Write([]byte(fmt.Sprintf("\rerror deleting: %s (%s)\n", n, err)))
numErrors++
} else {
output.Write([]byte(fmt.Sprintf("\rdeleted: %s\n", n)))
numDeleted++
numRemain--
}
} else {
output.Write([]byte(ansiClearLine))
output.Write([]byte(fmt.Sprintf("\rnot deleting: %s\n", n)))
}
} else {
output.Write([]byte(fmt.Sprintf("\ris referenced: %s\n", n)))
}
}
return nil
}
// RegressDelete removes a cartridge from the regression db.
func RegressDelete(output io.Writer, confirmation io.Reader, key string) error {
if output == nil {
return fmt.Errorf("regression: delete: io.Writer should not be nil")
}
if confirmation == nil {
return fmt.Errorf("regression: delete: io.Reader should not be nil")
}
v, err := strconv.Atoi(key)
if err != nil {
return fmt.Errorf("regression: delete: invalid key [%s]", key)
}
dbPth, err := resources.JoinPath(regressionPath, regressionDBFile)
if err != nil {
return fmt.Errorf("regression: delete: %w", err)
}
db, err := database.StartSession(dbPth, database.ActivityModifying, initDBSession)
if err != nil {
return fmt.Errorf("regression: delete: %w", err)
}
defer db.EndSession(true)
ent, err := db.SelectKeys(nil, v)
if err != nil {
return fmt.Errorf("regression: delete: %w", err)
}
output.Write([]byte(fmt.Sprintf("%s\ndelete? (y/n): ", ent)))
if confirm(confirmation) {
err = db.Delete(v)
if err != nil {
return fmt.Errorf("regression: delete: %w", err)
}
output.Write([]byte(fmt.Sprintf("deleted test #%s from regression database\n", key)))
}
return nil
}
// RegressRun runs all the tests in the regression database. filterKeys
// list specified which entries to test. an empty keys list means that every
// entry should be tested.
func RegressRun(output io.Writer, verbose bool, filterKeys []string) error {
if output == nil {
return fmt.Errorf("regression: run: io.Writer should not be nil")
}
dbPth, err := resources.JoinPath(regressionPath, regressionDBFile)
if err != nil {
return fmt.Errorf("regression: run: %w", err)
}
db, err := database.StartSession(dbPth, database.ActivityReading, initDBSession)
if err != nil {
return fmt.Errorf("regression: run: %w", err)
}
defer db.EndSession(false)
// make sure any supplied keys list is in order
keysV := make([]int, 0, len(filterKeys))
for k := range filterKeys {
v, err := strconv.Atoi(filterKeys[k])
if err != nil {
return fmt.Errorf("regression: run: invalid key [%s]", filterKeys[k])
}
keysV = append(keysV, v)
}
sort.Ints(keysV)
numSucceed := 0
numFail := 0
numError := 0
defer func() {
output.Write([]byte(fmt.Sprintf("regression tests: %d succeed, %d fail", numSucceed, numFail)))
if numError > 0 {
output.Write([]byte(fmt.Sprintf(" [with %d errors]", numError)))
}
output.Write([]byte("\n"))
}()
// selectKeys() calls this onSelect function for every key entry
onSelect := func(ent database.Entry) error {
// database entry should also satisfy Regressor interface
reg, ok := ent.(Regressor)
if !ok {
return fmt.Errorf("regression: run: database entry does not satisfy Regressor interface")
}
// run regress() function with message. message does not have a
// trailing newline
msg := fmt.Sprintf("running: %s", reg)
ok, failm, err := reg.regress(false, output, msg)
// once regress() has completed we clear the line ready for the
// completion message
output.Write([]byte(ansiClearLine))
// print completion message depending on result of regress()
if err != nil {
numError++
output.Write([]byte(fmt.Sprintf("\rerror: %s\n", reg)))
// output any error message on following line
if verbose {
output.Write([]byte(fmt.Sprintf(" ^^ %s\n", err)))
}
// an error implies a fail
numFail++
} else if !ok {
numFail++
output.Write([]byte(fmt.Sprintf("\rfailure: %s\n", reg)))
if verbose && failm != "" {
output.Write([]byte(fmt.Sprintf(" ^^ %s\n", failm)))
}
} else {
numSucceed++
output.Write([]byte(fmt.Sprintf("\rsucceed: %s\n", reg)))
}
return nil
}
_, err = db.SelectKeys(onSelect, keysV...)
return nil
}
// returns true if response from user begins with 'y' or 'Y'.
func confirm(confirmation io.Reader) bool {
confirm := make([]byte, 32)
_, err := confirmation.Read(confirm)
if err != nil {
return false
}
if confirm[0] == 'y' || confirm[0] == 'Y' {
return true
}
return false
}