Add integration test for go atexit implementation

This commit is contained in:
Kovid Goyal
2025-09-30 12:37:25 +05:30
parent 190e3e5891
commit 16d411943a
3 changed files with 63 additions and 3 deletions

View File

@@ -23,6 +23,11 @@ class Atexit(BaseTest):
def tearDown(self): def tearDown(self):
shutil.rmtree(self.tdir) shutil.rmtree(self.tdir)
def test_go_atexit(self):
cp = subprocess.run([kitten_exe(), '__atexit__', 'test'], cwd=self.tdir)
self.ae(cp.returncode, 0)
self.assertFalse(os.listdir(self.tdir))
def test_atexit(self): def test_atexit(self):
def r(action='close'): def r(action='close'):

View File

@@ -8,6 +8,7 @@ import (
"strings" "strings"
"github.com/kovidgoyal/kitty/tools/cli" "github.com/kovidgoyal/kitty/tools/cli"
"github.com/kovidgoyal/kitty/tools/utils"
"github.com/kovidgoyal/kitty/tools/utils/shm" "github.com/kovidgoyal/kitty/tools/utils/shm"
) )
@@ -33,6 +34,9 @@ func main() (rc int, err error) {
rc = 0 rc = 0
for _, line := range lines { for _, line := range lines {
if action, rest, found := strings.Cut(line, " "); found { if action, rest, found := strings.Cut(line, " "); found {
if !found {
continue
}
switch action { switch action {
case "unlink": case "unlink":
if err := os.Remove(rest); err != nil && !os.IsNotExist(err) { if err := os.Remove(rest); err != nil && !os.IsNotExist(err) {
@@ -55,6 +59,25 @@ func main() (rc int, err error) {
return return
} }
func do_test() (err error) {
if err = os.WriteFile("file", []byte("moose"), 0o600); err != nil {
return
}
if err = utils.AtExitUnlink("file"); err != nil {
return
}
if err = os.Mkdir("dir", 0o700); err != nil {
return
}
if err = utils.AtExitRmtree("dir"); err != nil {
return
}
if err = os.WriteFile("dir/sf", []byte("cat"), 0o600); err != nil {
return
}
return
}
func EntryPoint(root *cli.Command) { func EntryPoint(root *cli.Command) {
root.AddSubCommand(&cli.Command{ root.AddSubCommand(&cli.Command{
Name: "__atexit__", Name: "__atexit__",
@@ -62,6 +85,13 @@ func EntryPoint(root *cli.Command) {
OnlyArgsAllowed: true, OnlyArgsAllowed: true,
Run: func(cmd *cli.Command, args []string) (rc int, err error) { Run: func(cmd *cli.Command, args []string) (rc int, err error) {
if len(args) != 0 { if len(args) != 0 {
if args[0] == "test" {
rc = 0
if err = do_test(); err != nil {
rc = 1
}
return
}
return 1, fmt.Errorf("Usage: __atexit__") return 1, fmt.Errorf("Usage: __atexit__")
} }
return main() return main()

View File

@@ -1,10 +1,12 @@
package utils package utils
import ( import (
"flag"
"fmt" "fmt"
"io" "io"
"os" "os"
"os/exec" "os/exec"
"path/filepath"
"sync" "sync"
"sync/atomic" "sync/atomic"
) )
@@ -18,11 +20,21 @@ type worker struct {
var worker_started atomic.Bool var worker_started atomic.Bool
// IsTesting returns true if the code is being run by "go test".
func IsTesting() bool {
return flag.Lookup("test.v") != nil
}
var get_worker = sync.OnceValues(func() (*worker, error) { var get_worker = sync.OnceValues(func() (*worker, error) {
exe, err := os.Executable() exe, err := os.Executable()
if err != nil { if err != nil {
return nil, err return nil, err
} }
if IsTesting() {
if exe, err = filepath.Abs("../../kitty/launcher/kitten"); err != nil {
return nil, err
}
}
cmd := exec.Command(exe, "__atexit__") cmd := exec.Command(exe, "__atexit__")
cmd.Stdout = nil cmd.Stdout = nil
cmd.Stderr = os.Stderr cmd.Stderr = os.Stderr
@@ -32,23 +44,36 @@ var get_worker = sync.OnceValues(func() (*worker, error) {
return nil, err return nil, err
} }
ans.stdin_pipe = si ans.stdin_pipe = si
if err = cmd.Run(); err != nil { if err = cmd.Start(); err != nil {
return nil, err return nil, err
} }
worker_started.Store(true) worker_started.Store(true)
return &ans, nil return &ans, nil
}) })
func WaitForAtexitWorkerToFinish() { func WaitForAtexitWorkerToFinish() error {
if worker_started.Load() { if worker_started.Load() {
if w, err := get_worker(); err == nil { if w, err := get_worker(); err == nil {
w.stdin_pipe.Close() w.stdin_pipe.Close()
_ = w.cmd.Wait() return w.cmd.Wait()
} else {
return err
} }
} }
return nil
} }
func register(prefix, path string) error { func register(prefix, path string) error {
// no atexit cleanup is done as we dont have a good place to run
// WaitForAtexitWorkerToFinish() and anyway we may want to run tests in
// parallel, etc.
if IsTesting() {
return nil
}
path, err := filepath.Abs(path)
if err != nil {
return err
}
if w, err := get_worker(); err == nil { if w, err := get_worker(); err == nil {
_, err = fmt.Fprintln(w.stdin_pipe, prefix+" "+path) _, err = fmt.Fprintln(w.stdin_pipe, prefix+" "+path)
return err return err