mirror of https://github.com/01-edu/public.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
166 lines
3.8 KiB
166 lines
3.8 KiB
5 years ago
|
package student_test
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"fmt"
|
||
|
"io/ioutil"
|
||
|
"os"
|
||
|
"os/exec"
|
||
|
"path/filepath"
|
||
|
"sort"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"testing"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
type (
|
||
|
vect struct{ x, y int }
|
||
|
|
||
|
// the tetromino is described with three vectors
|
||
|
// example :
|
||
|
//
|
||
|
// [#][#][#]
|
||
|
// [#]
|
||
|
//
|
||
|
// tetromino{{1, 0}, {2, 0}, {1, 1}}
|
||
|
tetromino [3]vect
|
||
|
)
|
||
|
|
||
|
// load a tetromino composed of the rune r in the map s
|
||
|
// example :
|
||
|
//
|
||
|
// read(`......
|
||
|
// ......
|
||
|
// ......
|
||
|
// ....X.
|
||
|
// ..XXX.
|
||
|
// ......
|
||
|
// `,
|
||
|
// 'X',
|
||
|
// ) == tetromino{{-2, 1}, {-1, 1}, {0, 1}}
|
||
|
func read(s string, r rune) (t tetromino) {
|
||
|
var origin vect
|
||
|
i := 0
|
||
|
first := true
|
||
|
lines := strings.Split(s, "\n")
|
||
|
for y, line := range lines {
|
||
|
for x := range line {
|
||
|
if []rune(line)[x] == r {
|
||
|
if first {
|
||
|
first = false
|
||
|
origin = vect{x, y}
|
||
|
} else {
|
||
|
t[i] = vect{x - origin.x, y - origin.y}
|
||
|
i++
|
||
|
if i == 3 {
|
||
|
return tetromino{}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func TestTetrisOptimizer(t *testing.T) {
|
||
|
var (
|
||
|
timeout = 30 * time.Second
|
||
|
samples = "./solutions/tetrisoptimizer/samples"
|
||
|
student = "./student/tetrisoptimizer"
|
||
|
exe = filepath.Join(student, "a.out")
|
||
|
)
|
||
|
// load samples
|
||
|
f, err := os.Open(samples)
|
||
|
if err != nil {
|
||
|
t.Fatal("Cannot open directory", err)
|
||
|
}
|
||
|
defer f.Close()
|
||
|
filenames, err := f.Readdirnames(0)
|
||
|
if err != nil {
|
||
|
t.Fatal("Cannot read directory", err)
|
||
|
}
|
||
|
|
||
|
// separate samples into good (valid) and bad (invalid) files
|
||
|
var goodFiles, badFiles []string
|
||
|
for _, filename := range filenames {
|
||
|
if strings.HasPrefix(filename, "good") {
|
||
|
goodFiles = append(goodFiles, filepath.Join(samples, filename))
|
||
|
} else if strings.HasPrefix(filename, "bad") {
|
||
|
badFiles = append(badFiles, filepath.Join(samples, filename))
|
||
|
}
|
||
|
}
|
||
|
sort.Strings(badFiles)
|
||
|
sort.Strings(goodFiles)
|
||
|
|
||
|
// compile student code
|
||
|
cmd := exec.Command("go", "build", "-o", exe, "-trimpath", "-ldflags", "-s -w", student)
|
||
|
cmd.Env = append(os.Environ(), "CGO_ENABLED=0", "GOARCH=amd64")
|
||
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||
|
t.Fatal("Cannot compile :", string(out))
|
||
|
}
|
||
|
|
||
|
ctx, cancel := context.WithTimeout(context.Background(), timeout)
|
||
|
defer cancel()
|
||
|
|
||
|
// test invalid cases
|
||
|
for _, badFile := range badFiles {
|
||
|
b, _ := exec.CommandContext(ctx, exe, badFile).CombinedOutput()
|
||
|
if string(b) != "ERROR\n" {
|
||
|
t.Fatal(`Failed to handle bad format, should output : "ERROR\n"`)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// test valid cases
|
||
|
score := .0
|
||
|
defer func() {
|
||
|
fmt.Println("score =", score)
|
||
|
}()
|
||
|
for _, goodFile := range goodFiles {
|
||
|
size, _ := strconv.Atoi(strings.Split(goodFile, "-")[2])
|
||
|
b, err := exec.CommandContext(ctx, exe, goodFile).Output()
|
||
|
if ctx.Err() == context.DeadlineExceeded {
|
||
|
return
|
||
|
}
|
||
|
if err != nil {
|
||
|
t.Fatal("Failed to process a valid map : execution failed")
|
||
|
}
|
||
|
s := string(b)
|
||
|
lines := strings.Split(s, "\n")
|
||
|
if lines[len(lines)-1] != "" {
|
||
|
t.Fatal(`Failed to process a valid map : missing final '\n'`)
|
||
|
}
|
||
|
lines = lines[:len(lines)-1]
|
||
|
for _, line := range lines {
|
||
|
if len(line) != len(lines) {
|
||
|
t.Fatal("Failed to process a valid map : invalid square, it is expected as many lines as characters")
|
||
|
}
|
||
|
}
|
||
|
if len(lines) < size {
|
||
|
t.Fatal("Failed to process a valid map : the square cannot be that small")
|
||
|
}
|
||
|
b, err = ioutil.ReadFile(goodFile)
|
||
|
if err != nil {
|
||
|
t.Fatal("Failed to read a valid map")
|
||
|
}
|
||
|
pieces := strings.Split(string(b), "\n\n")
|
||
|
surface := len(lines) * len(lines)
|
||
|
if strings.Count(s, ".") != surface-len(pieces)*4 {
|
||
|
t.Fatal("Failed to process a valid map : the number of holes (character '.') is not correct")
|
||
|
}
|
||
|
letter := 'A'
|
||
|
for _, piece := range pieces {
|
||
|
if read(s, letter) != read(piece, '#') {
|
||
|
t.Fatal("Failed to process a valid map : a tetromino is missing")
|
||
|
}
|
||
|
letter += 1
|
||
|
}
|
||
|
score += float64(size*size) / float64(surface)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO:
|
||
|
// Ajouter des cas d'erreurs :
|
||
|
// mauvais arguments
|
||
|
// mauvais types de fichiers (liens, dossiers)
|