mirror of
https://github.com/kovidgoyal/kitty
synced 2026-06-08 14:18:26 +02:00
Add integration test for go atexit implementation
This commit is contained in:
@@ -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'):
|
||||||
|
|||||||
@@ -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()
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
Reference in New Issue
Block a user