forked from joejulian/gio-cmd
ae8dd5433d
Signed-off-by: ddkwork
197 lines
4.6 KiB
Go
197 lines
4.6 KiB
Go
// SPDX-License-Identifier: Unlicense OR MIT
|
|
|
|
package main_test
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"image"
|
|
"image/png"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
"sync"
|
|
"text/template"
|
|
"time"
|
|
)
|
|
|
|
type WaylandTestDriver struct {
|
|
driverBase
|
|
|
|
runtimeDir string
|
|
socket string
|
|
display string
|
|
}
|
|
|
|
// No bars or anything fancy. Just a white background with our dimensions.
|
|
var tmplSwayConfig = template.Must(template.New("").Parse(`
|
|
output * bg #FFFFFF solid_color
|
|
output * mode {{.Width}}x{{.Height}}
|
|
default_border none
|
|
`))
|
|
|
|
var rxSwayReady = regexp.MustCompile(`Running compositor on wayland display '(.*)'`)
|
|
|
|
func (d *WaylandTestDriver) Start(path string) {
|
|
// We want os.Environ, so that it can e.g. find $DISPLAY to run within
|
|
// X11. wlroots env vars are documented at:
|
|
// https://github.com/swaywm/wlroots/blob/master/docs/env_vars.md
|
|
env := os.Environ()
|
|
if *headless {
|
|
env = append(env, "WLR_BACKENDS=headless")
|
|
}
|
|
|
|
d.needPrograms(
|
|
"sway", // to run a wayland compositor
|
|
"grim", // to take screenshots
|
|
"swaymsg", // to send input
|
|
)
|
|
|
|
// First, build the app.
|
|
dir := d.tempDir("gio-endtoend-wayland")
|
|
bin := filepath.Join(dir, "red")
|
|
flags := []string{"build", "-tags", "nox11", "-o=" + bin}
|
|
if raceEnabled {
|
|
flags = append(flags, "-race")
|
|
}
|
|
flags = append(flags, path)
|
|
cmd := exec.Command("go", flags...)
|
|
if out, err := cmd.CombinedOutput(); err != nil {
|
|
d.Fatalf("could not build app: %s:\n%s", err, out)
|
|
}
|
|
|
|
conf := filepath.Join(dir, "config")
|
|
f, err := os.Create(conf)
|
|
if err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
defer f.Close()
|
|
if err := tmplSwayConfig.Execute(f, struct{ Width, Height int }{
|
|
d.width, d.height,
|
|
}); err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
|
|
d.socket = filepath.Join(dir, "socket")
|
|
env = append(env, "SWAYSOCK="+d.socket)
|
|
d.runtimeDir = dir
|
|
env = append(env, "XDG_RUNTIME_DIR="+d.runtimeDir)
|
|
|
|
var wg sync.WaitGroup
|
|
d.Cleanup(wg.Wait)
|
|
|
|
// First, start sway.
|
|
{
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cmd := exec.CommandContext(ctx, "sway", "--config", conf, "--verbose")
|
|
cmd.Env = env
|
|
stderr, err := cmd.StderrPipe()
|
|
if err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
if err := cmd.Start(); err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
d.Cleanup(cancel)
|
|
d.Cleanup(func() {
|
|
// Give it a chance to exit gracefully, cleaning up
|
|
// after itself. After 10ms, the deferred cancel above
|
|
// will signal an os.Kill.
|
|
cmd.Process.Signal(os.Interrupt)
|
|
time.Sleep(10 * time.Millisecond)
|
|
})
|
|
|
|
// Wait for sway to be ready. We probably don't need a deadline
|
|
// here.
|
|
br := bufio.NewReader(stderr)
|
|
for {
|
|
line, err := br.ReadString('\n')
|
|
if err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
if m := rxSwayReady.FindStringSubmatch(line); m != nil {
|
|
d.display = m[1]
|
|
break
|
|
}
|
|
}
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
if err := cmd.Wait(); err != nil && ctx.Err() == nil && !strings.Contains(err.Error(), "interrupt") {
|
|
// Don't print all stderr, since we use --verbose.
|
|
// TODO(mvdan): if it's useful, probably filter
|
|
// errors and show them.
|
|
d.Error(err)
|
|
}
|
|
wg.Done()
|
|
}()
|
|
}
|
|
|
|
// Then, start our program on the sway compositor above.
|
|
{
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cmd := exec.CommandContext(ctx, bin)
|
|
cmd.Env = []string{"XDG_RUNTIME_DIR=" + d.runtimeDir, "WAYLAND_DISPLAY=" + d.display}
|
|
output, err := cmd.StdoutPipe()
|
|
if err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
cmd.Stderr = cmd.Stdout
|
|
d.output = output
|
|
if err := cmd.Start(); err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
d.Cleanup(cancel)
|
|
wg.Add(1)
|
|
go func() {
|
|
if err := cmd.Wait(); err != nil && ctx.Err() == nil {
|
|
d.Error(err)
|
|
}
|
|
wg.Done()
|
|
}()
|
|
}
|
|
|
|
// Wait for the gio app to render.
|
|
d.waitForFrame()
|
|
}
|
|
|
|
func (d *WaylandTestDriver) Screenshot() image.Image {
|
|
cmd := exec.Command("grim", "/dev/stdout")
|
|
cmd.Env = []string{"XDG_RUNTIME_DIR=" + d.runtimeDir, "WAYLAND_DISPLAY=" + d.display}
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
d.Errorf("%s", out)
|
|
d.Fatal(err)
|
|
}
|
|
img, err := png.Decode(bytes.NewReader(out))
|
|
if err != nil {
|
|
d.Fatal(err)
|
|
}
|
|
return img
|
|
}
|
|
|
|
func (d *WaylandTestDriver) swaymsg(args ...any) {
|
|
strs := []string{"--socket", d.socket}
|
|
for _, arg := range args {
|
|
strs = append(strs, fmt.Sprint(arg))
|
|
}
|
|
cmd := exec.Command("swaymsg", strs...)
|
|
if out, err := cmd.CombinedOutput(); err != nil {
|
|
d.Errorf("%s", out)
|
|
d.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func (d *WaylandTestDriver) Click(x, y int) {
|
|
d.swaymsg("seat", "-", "cursor", "set", x, y)
|
|
d.swaymsg("seat", "-", "cursor", "press", "button1")
|
|
d.swaymsg("seat", "-", "cursor", "release", "button1")
|
|
|
|
// Wait for the gio app to render after this click.
|
|
d.waitForFrame()
|
|
}
|