common.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. package test
  2. import (
  3. "bytes"
  4. "os"
  5. "os/exec"
  6. "path/filepath"
  7. "strings"
  8. "syscall"
  9. "testing"
  10. )
  11. // A PlaybookTest executes a given Ansible playbook and checks the exit code and
  12. // output contents.
  13. type PlaybookTest struct {
  14. // inputs
  15. Path string
  16. // expected outputs
  17. ExitCode int
  18. Output []string // zero or more strings that should be in the output
  19. }
  20. // Run runs the PlaybookTest.
  21. func (p PlaybookTest) Run(t *testing.T) {
  22. // A PlaybookTest is intended to be run in parallel with other tests.
  23. t.Parallel()
  24. cmd := exec.Command("ansible-playbook", "-e", "testing_skip_some_requirements=1", "-i", "/dev/null", p.Path)
  25. cmd.Env = append(os.Environ(), "ANSIBLE_FORCE_COLOR=1")
  26. b, err := cmd.CombinedOutput()
  27. // Check exit code.
  28. if (err == nil) && (p.ExitCode != 0) {
  29. p.checkExitCode(t, 0, p.ExitCode, cmd, b)
  30. }
  31. if (err != nil) && (p.ExitCode == 0) {
  32. got, ok := getExitCode(err)
  33. if !ok {
  34. t.Logf("unexpected error (%T): %[1]v", err)
  35. p.logCmdAndOutput(t, cmd, b)
  36. t.FailNow()
  37. }
  38. p.checkExitCode(t, got, p.ExitCode, cmd, b)
  39. }
  40. // Check output contents.
  41. var missing []string
  42. for _, s := range p.Output {
  43. if !bytes.Contains(b, []byte(s)) {
  44. missing = append(missing, s)
  45. }
  46. }
  47. if len(missing) > 0 {
  48. t.Logf("missing in output: %q", missing)
  49. p.logCmdAndOutput(t, cmd, b)
  50. t.FailNow()
  51. }
  52. }
  53. // getExitCode returns an exit code and true if the exit code could be taken
  54. // from err, false otherwise.
  55. // The implementation is GOOS-specific, and currently only supports Linux.
  56. func getExitCode(err error) (int, bool) {
  57. exitErr, ok := err.(*exec.ExitError)
  58. if !ok {
  59. return -1, false
  60. }
  61. waitStatus, ok := exitErr.Sys().(syscall.WaitStatus)
  62. if !ok {
  63. return -1, false
  64. }
  65. return waitStatus.ExitStatus(), true
  66. }
  67. // checkExitCode marks the test as failed when got is different than want.
  68. func (p PlaybookTest) checkExitCode(t *testing.T, got, want int, cmd *exec.Cmd, output []byte) {
  69. if got == want {
  70. return
  71. }
  72. t.Logf("got exit code %v, want %v", got, want)
  73. p.logCmdAndOutput(t, cmd, output)
  74. t.FailNow()
  75. }
  76. // logCmdAndOutput logs how to re-run a command and a summary of the output of
  77. // its last execution for debugging.
  78. func (p PlaybookTest) logCmdAndOutput(t *testing.T, cmd *exec.Cmd, output []byte) {
  79. const maxLines = 10
  80. lines := bytes.Split(bytes.TrimRight(output, "\n"), []byte("\n"))
  81. if len(lines) > maxLines {
  82. lines = append([][]byte{[]byte("...")}, lines[len(lines)-maxLines:len(lines)]...)
  83. }
  84. output = bytes.Join(lines, []byte("\n"))
  85. dir, err := filepath.Abs(cmd.Dir)
  86. if err != nil {
  87. panic(err)
  88. }
  89. t.Logf("\n$ (cd %s && %s)\n%s", dir, strings.Join(cmd.Args, " "), output)
  90. }