diff --git a/README.md b/README.md index cca98cf1..b4fe867f 100644 --- a/README.md +++ b/README.md @@ -159,9 +159,15 @@ package may be used to parse the JSON file output. First install the example notification command with `go get gotest.tools/gotestsum/contrib/notify`. The command will be downloaded to `$GOPATH/bin` as `notify`. Note that this -example `notify` command only works on macOS with +example `notify` command only works on Linux with `notify-send` and on macOS with [terminal-notifer](https://github.com/julienXX/terminal-notifier) installed. +On Linux, you need to have some "test-pass" and "test-fail" icons installed in your icon theme. +Some sample icons can be found in `contrib/notify`, and can be installed with `make install`. + +On Windows, you can install [notify-send.exe](https://github.com/vaskovsky/notify-send) +but it does not support custom icons so will have to use the basic "info" and "error". + ``` gotestsum --post-run-command notify ``` diff --git a/cmd/handler.go b/cmd/handler.go index 0820ee68..b2d74f9f 100644 --- a/cmd/handler.go +++ b/cmd/handler.go @@ -161,6 +161,7 @@ func postRunHook(opts *options, execution *testjson.Execution) error { if len(command) == 0 { return nil } + log.Debugf("exec: %s", command) cmd := exec.Command(command[0], command[1:]...) cmd.Stdout = opts.stdout diff --git a/contrib/notify/.gitignore b/contrib/notify/.gitignore new file mode 100644 index 00000000..f3b695d3 --- /dev/null +++ b/contrib/notify/.gitignore @@ -0,0 +1 @@ +notify diff --git a/contrib/notify/Makefile b/contrib/notify/Makefile new file mode 100644 index 00000000..73771c64 --- /dev/null +++ b/contrib/notify/Makefile @@ -0,0 +1,13 @@ +GO = go +INSTALL = install + +ICONS = icons/test-pass.svg icons/test-fail.svg +ICONDIR = $(HOME)/.icons # or /usr/share/icons + +build: + $(GO) build + +install: $(ICONS) + $(GO) install + $(INSTALL) -d $(ICONDIR) + $(INSTALL) $^ $(ICONDIR) diff --git a/contrib/notify/icons/README.html b/contrib/notify/icons/README.html new file mode 100644 index 00000000..4512a24e --- /dev/null +++ b/contrib/notify/icons/README.html @@ -0,0 +1 @@ +Pass Fail Vectors by Vecteezy diff --git a/contrib/notify/icons/test-fail.svg b/contrib/notify/icons/test-fail.svg new file mode 100644 index 00000000..f6b97c74 --- /dev/null +++ b/contrib/notify/icons/test-fail.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/contrib/notify/icons/test-pass.svg b/contrib/notify/icons/test-pass.svg new file mode 100644 index 00000000..2e96bdaf --- /dev/null +++ b/contrib/notify/icons/test-pass.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/contrib/notify/notify-macos.go b/contrib/notify/notify_darwin.go similarity index 85% rename from contrib/notify/notify-macos.go rename to contrib/notify/notify_darwin.go index 3f6822ca..ad683898 100644 --- a/contrib/notify/notify-macos.go +++ b/contrib/notify/notify_darwin.go @@ -43,9 +43,11 @@ func main() { "-group", "gotestsum", "-subtitle", subtitle, } - log.Printf("terminal-notifier %#v", args) - err := exec.Command("terminal-notifier", args...).Run() - if err != nil { + cmd := exec.Command("terminal-notifier", args...) + log.Printf("%#v", cmd.Args) + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + if err := cmd.Run(); err != nil { log.Fatalf("Failed to exec: %v", err) } } diff --git a/contrib/notify/notify_linux.go b/contrib/notify/notify_linux.go new file mode 100644 index 00000000..8da6c3e8 --- /dev/null +++ b/contrib/notify/notify_linux.go @@ -0,0 +1,57 @@ +package main + +import ( + "fmt" + "log" + "os" + "os/exec" + "strconv" +) + +func main() { + total := envInt("TOTAL") + skipped := envInt("SKIPPED") + failed := envInt("FAILED") + errors := envInt("ERRORS") + + icon := "test-pass" + title := "Passed" + switch { + case errors > 0: + icon = "dialog-warning" + title = "Errored" + case failed > 0: + icon = "test-fail" + title = "Failed" + case skipped > 0: + title = "Passed with skipped" + } + + subtitle := fmt.Sprintf("%d Tests Run", total) + if errors > 0 { + subtitle += fmt.Sprintf(", %d Errored", errors) + } + if failed > 0 { + subtitle += fmt.Sprintf(", %d Failed", failed) + } + if skipped > 0 { + subtitle += fmt.Sprintf(", %d Skipped", skipped) + } + + cmd := exec.Command("notify-send", "--icon", icon, title, subtitle) + log.Printf("%#v", cmd.Args) + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + if err := cmd.Run(); err != nil { + log.Fatalf("Failed to exec: %v", err) + } +} + +func envInt(name string) int { + val := os.Getenv("TESTS_" + name) + n, err := strconv.Atoi(val) + if err != nil { + return 0 + } + return n +} diff --git a/contrib/notify/notify_windows.go b/contrib/notify/notify_windows.go new file mode 100644 index 00000000..f3d384d5 --- /dev/null +++ b/contrib/notify/notify_windows.go @@ -0,0 +1,57 @@ +package main + +import ( + "fmt" + "log" + "os" + "os/exec" + "strconv" +) + +func main() { + total := envInt("TOTAL") + skipped := envInt("SKIPPED") + failed := envInt("FAILED") + errors := envInt("ERRORS") + + icon := "info" // Info 🛈 + title := "Passed" + switch { + case errors > 0: + icon = "important" // Warning ⚠ + title = "Errored" + case failed > 0: + icon = "error" // Error ⮾ + title = "Failed" + case skipped > 0: + title = "Passed with skipped" + } + + subtitle := fmt.Sprintf("%d Tests Run", total) + if errors > 0 { + subtitle += fmt.Sprintf(", %d Errored", errors) + } + if failed > 0 { + subtitle += fmt.Sprintf(", %d Failed", failed) + } + if skipped > 0 { + subtitle += fmt.Sprintf(", %d Skipped", skipped) + } + + cmd := exec.Command("notify-send.exe", "-i", icon, title, subtitle) + log.Printf("%#v", cmd.Args) + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + if err := cmd.Run(); err != nil { + log.Fatalf("Failed to exec: %v", err) + } +} + +func envInt(name string) int { + val := os.Getenv("TESTS_" + name) + n, err := strconv.Atoi(val) + if err != nil { + return 0 + } + return n +}