From 12f1a6bf6a300e1fc20eb7fa1ac4c349fe09936f Mon Sep 17 00:00:00 2001 From: Fred Date: Fri, 28 Jun 2024 15:25:37 +0100 Subject: [PATCH 1/4] don't use ping to wait for some time a ping to localhost doesn't necessarily work on some firewalled computers --- config/mocks/NamedPropertySet.go | 2 +- config/mocks/ProfileInfo.go | 2 +- config/mocks/PropertyInfo.go | 2 +- config/mocks/SectionInfo.go | 2 +- lock/lock_test.go | 130 +++++++++++++++---------------- monitor/mocks/OutputAnalysis.go | 2 +- schedule/mocks/Handler.go | 2 +- 7 files changed, 67 insertions(+), 75 deletions(-) diff --git a/config/mocks/NamedPropertySet.go b/config/mocks/NamedPropertySet.go index dd922974..d2a8fc80 100644 --- a/config/mocks/NamedPropertySet.go +++ b/config/mocks/NamedPropertySet.go @@ -1,4 +1,4 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. +// Code generated by mockery v2.43.2. DO NOT EDIT. package mocks diff --git a/config/mocks/ProfileInfo.go b/config/mocks/ProfileInfo.go index f60e7e12..df08131c 100644 --- a/config/mocks/ProfileInfo.go +++ b/config/mocks/ProfileInfo.go @@ -1,4 +1,4 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. +// Code generated by mockery v2.43.2. DO NOT EDIT. package mocks diff --git a/config/mocks/PropertyInfo.go b/config/mocks/PropertyInfo.go index 0741bfa6..19e15a1b 100644 --- a/config/mocks/PropertyInfo.go +++ b/config/mocks/PropertyInfo.go @@ -1,4 +1,4 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. +// Code generated by mockery v2.43.2. DO NOT EDIT. package mocks diff --git a/config/mocks/SectionInfo.go b/config/mocks/SectionInfo.go index d7ef7ff5..38dd7bc9 100644 --- a/config/mocks/SectionInfo.go +++ b/config/mocks/SectionInfo.go @@ -1,4 +1,4 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. +// Code generated by mockery v2.43.2. DO NOT EDIT. package mocks diff --git a/lock/lock_test.go b/lock/lock_test.go index d45b645e..e2016c63 100644 --- a/lock/lock_test.go +++ b/lock/lock_test.go @@ -8,33 +8,41 @@ import ( "os/signal" "path/filepath" "regexp" - "runtime" "syscall" "testing" "time" + "github.com/creativeprojects/resticprofile/platform" "github.com/creativeprojects/resticprofile/shell" "github.com/shirou/gopsutil/v3/process" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) -const ( +var ( helperBinary = "./locktest" ) func init() { - if runtime.GOOS == "windows" { - return + if platform.IsWindows() { + helperBinary += ".exe" } - // compile helper command cmd := exec.Command("go", "build", "-o", helperBinary, "./test") cmd.Run() } -func TestLockIsAvailable(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestLockIsAvailable", time.Now().UnixNano(), os.Getpid())) +func getTempfile(t *testing.T) string { + t.Helper() + + tempfile := filepath.Join(t.TempDir(), fmt.Sprintf("%s.tmp", t.Name())) t.Log("Using temporary file", tempfile) + return tempfile +} + +func TestLockIsAvailable(t *testing.T) { + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() @@ -42,8 +50,9 @@ func TestLockIsAvailable(t *testing.T) { } func TestLockIsNotAvailable(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestLockIsNotAvailable", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() @@ -62,8 +71,9 @@ func TestLockIsNotAvailable(t *testing.T) { } func TestNoPID(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestNoPID", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() lock.TryAcquire() @@ -77,8 +87,9 @@ func TestNoPID(t *testing.T) { } func TestSetOnePID(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestSetPID", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() lock.TryAcquire() @@ -93,8 +104,9 @@ func TestSetOnePID(t *testing.T) { } func TestSetMorePID(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestSetMorePID", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() lock.TryAcquire() @@ -110,8 +122,9 @@ func TestSetMorePID(t *testing.T) { assert.Equal(t, int32(13), pid) } -// This test is using the shell package. This is just a convenient wrapper around cmd.exe and sh func TestProcessFinished(t *testing.T) { + t.Parallel() + childPID := 0 buffer := &bytes.Buffer{} @@ -135,8 +148,9 @@ func TestProcessFinished(t *testing.T) { assert.False(t, running) } -// This test is using the shell package. This is just a convenient wrapper around cmd.exe and sh func TestProcessNotFinished(t *testing.T) { + t.Parallel() + childPID := 0 buffer := &bytes.Buffer{} @@ -144,18 +158,8 @@ func TestProcessNotFinished(t *testing.T) { signal.Notify(c, os.Interrupt) defer signal.Reset(os.Interrupt) - // use ping to make sure the process is running for long enough to check its existence - var parameters []string - if runtime.GOOS == "windows" { - // it will run for 1 second - parameters = []string{"-n", "2", "127.0.0.1"} - } else { - // run for 200ms (don't need a whole second) - // 0.2 is the minimum in linux, 0.1 in darwin - parameters = []string{"-c", "2", "-i", "0.2", "127.0.0.1"} - } - - cmd := shell.NewSignalledCommand("ping", parameters, c) + // user the lock helper binary (we only need to wait for some time, we don't need the locking part) + cmd := shell.NewSignalledCommand(helperBinary, []string{"-wait", "100", "-lock", t.Name()}, c) cmd.Stdout = buffer // SetPID method is called right after we forked and have a PID available cmd.SetPID = func(pid int) { @@ -165,9 +169,7 @@ func TestProcessNotFinished(t *testing.T) { assert.True(t, running) } _, _, err := cmd.Run() - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) // at that point, the child process should be finished running, err := process.PidExists(int32(childPID)) @@ -176,8 +178,9 @@ func TestProcessNotFinished(t *testing.T) { } func TestForceLockIsAvailable(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestForceLockIsAvailable", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() @@ -185,8 +188,9 @@ func TestForceLockIsAvailable(t *testing.T) { } func TestForceLockWithNoPID(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestForceLockWithNoPID", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() @@ -199,10 +203,10 @@ func TestForceLockWithNoPID(t *testing.T) { assert.False(t, other.HasLocked()) } -// This test is using the shell package. This is just a convenient wrapper around cmd.exe and sh func TestForceLockWithExpiredPID(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestForceLockWithExpiredPID", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() @@ -230,10 +234,10 @@ func TestForceLockWithExpiredPID(t *testing.T) { assert.True(t, other.HasLocked()) } -// This test is using the shell package. This is just a convenient wrapper around cmd.exe and sh func TestForceLockWithRunningPID(t *testing.T) { - tempfile := filepath.Join(os.TempDir(), fmt.Sprintf("%s%d%d.tmp", "TestForceLockWithRunningPID", time.Now().UnixNano(), os.Getpid())) - t.Log("Using temporary file", tempfile) + t.Parallel() + + tempfile := getTempfile(t) lock := NewLock(tempfile) defer lock.Release() @@ -245,18 +249,8 @@ func TestForceLockWithRunningPID(t *testing.T) { signal.Notify(c, os.Interrupt) defer signal.Reset(os.Interrupt) - // use ping to make sure the process is running for long enough to check its existence - var parameters []string - if runtime.GOOS == "windows" { - // it will run for 1 second - parameters = []string{"-n", "2", "127.0.0.1"} - } else { - // run for 200ms (don't need a whole second) - // 0.2 is the minimum in linux, 0.1 in darwin - parameters = []string{"-c", "2", "-i", "0.2", "127.0.0.1"} - } - - cmd := shell.NewSignalledCommand("ping", parameters, c) + // user the lock helper binary (we only need to wait for some time, we don't need the locking part) + cmd := shell.NewSignalledCommand(helperBinary, []string{"-wait", "100", "-lock", t.Name()}, c) cmd.SetPID = func(pid int) { lock.SetPID(pid) // make sure we cannot break the lock right now @@ -266,18 +260,16 @@ func TestForceLockWithRunningPID(t *testing.T) { assert.False(t, other.HasLocked()) } _, _, err := cmd.Run() - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) } func TestLockWithNoInterruption(t *testing.T) { - if runtime.GOOS == "windows" { + t.Parallel() + + if platform.IsWindows() { t.Skip("cannot send a signal to a child process in Windows") } - lockfile := "TestLockWithNoInterruption.lock" - // make sure there's no remaining lockfile from a failed test - _ = os.Remove(lockfile) + lockfile := getTempfile(t) var err error buffer := &bytes.Buffer{} @@ -291,12 +283,12 @@ func TestLockWithNoInterruption(t *testing.T) { } func TestLockIsRemovedAfterInterruptSignal(t *testing.T) { - if runtime.GOOS == "windows" { + t.Parallel() + + if platform.IsWindows() { t.Skip("cannot send a signal to a child process in Windows") } - lockfile := "TestLockIsRemovedAfterInterruptSignal.lock" - // make sure there's no remaining lockfile from a failed test - _ = os.Remove(lockfile) + lockfile := getTempfile(t) var err error buffer := &bytes.Buffer{} @@ -317,12 +309,12 @@ func TestLockIsRemovedAfterInterruptSignal(t *testing.T) { } func TestLockIsRemovedAfterInterruptSignalInsideShell(t *testing.T) { - if runtime.GOOS == "windows" { + t.Parallel() + + if platform.IsWindows() { t.Skip("cannot send a signal to a child process in Windows") } - lockfile := "TestLockIsRemovedAfterInterruptSignal.lock" - // make sure there's no remaining lockfile from a failed test - _ = os.Remove(lockfile) + lockfile := getTempfile(t) var err error buffer := &bytes.Buffer{} diff --git a/monitor/mocks/OutputAnalysis.go b/monitor/mocks/OutputAnalysis.go index 28dd1393..b81bc16f 100644 --- a/monitor/mocks/OutputAnalysis.go +++ b/monitor/mocks/OutputAnalysis.go @@ -1,4 +1,4 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. +// Code generated by mockery v2.43.2. DO NOT EDIT. package mocks diff --git a/schedule/mocks/Handler.go b/schedule/mocks/Handler.go index ab2c8b9c..74971545 100644 --- a/schedule/mocks/Handler.go +++ b/schedule/mocks/Handler.go @@ -1,4 +1,4 @@ -// Code generated by mockery v2.42.1. DO NOT EDIT. +// Code generated by mockery v2.43.2. DO NOT EDIT. package mocks From e8ee5c4b77b39839298f2b4ec91ba5ee1cd9c7bb Mon Sep 17 00:00:00 2001 From: Fred Date: Fri, 28 Jun 2024 16:00:56 +0100 Subject: [PATCH 2/4] fix duplicated tests --- lock/lock_test.go | 15 +++++++-------- 1 file changed, 7 insertions(+), 8 deletions(-) diff --git a/lock/lock_test.go b/lock/lock_test.go index e2016c63..132560c6 100644 --- a/lock/lock_test.go +++ b/lock/lock_test.go @@ -28,7 +28,9 @@ func init() { helperBinary += ".exe" } cmd := exec.Command("go", "build", "-o", helperBinary, "./test") - cmd.Run() + if err := cmd.Run(); err != nil { + fmt.Fprintf(os.Stderr, "Error building helper binary: %s\n", err) + } } func getTempfile(t *testing.T) string { @@ -138,9 +140,7 @@ func TestProcessFinished(t *testing.T) { childPID = pid } _, _, err := cmd.Run() - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) // at that point, the child process should be finished running, err := process.PidExists(int32(childPID)) @@ -221,9 +221,8 @@ func TestForceLockWithExpiredPID(t *testing.T) { cmd := shell.NewSignalledCommand("echo", []string{"Hello World!"}, c) cmd.SetPID = lock.SetPID _, _, err := cmd.Run() - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) + // child process should be finished // let's close the lockfile handle manually (unix doesn't actually care, but windows would complain) lock.file.Close() @@ -318,7 +317,7 @@ func TestLockIsRemovedAfterInterruptSignalInsideShell(t *testing.T) { var err error buffer := &bytes.Buffer{} - cmd := exec.Command(helperBinary, "-wait", "600", "-lock", lockfile) + cmd := exec.Command("sh", "-c", helperBinary+" -wait 600 -lock "+lockfile) cmd.Stdout = buffer cmd.Stderr = buffer From d605d1907b8d2118830f5312accbf14546db8214 Mon Sep 17 00:00:00 2001 From: Fred Date: Fri, 28 Jun 2024 17:41:11 +0100 Subject: [PATCH 3/4] fix tests under windows --- .golangci.yml | 54 ---------------------------------------------- lock/lock_test.go | 55 +++++++++++------------------------------------ 2 files changed, 12 insertions(+), 97 deletions(-) diff --git a/.golangci.yml b/.golangci.yml index 6b9cf9d0..e69de29b 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -1,54 +0,0 @@ -linters-settings: - govet: - check-shadowing: true - gocyclo: - min-complexity: 20 - dupl: - threshold: 100 - goconst: - min-len: 3 - min-occurrences: 3 - lll: - line-length: 160 - nakedret: - max-func-lines: 20 - gocritic: - enabled-tags: - - performance - - style - - diagnostic - disabled-checks: - - dupImport # https://github.com/go-critic/go-critic/issues/845 - settings: - rangeValCopy: - sizeThreshold: 512 - hugeParam: - sizeThreshold: 512 -linters: - enable-all: true - disable: - - interfacer - - scopelint - - golint - - maligned - - godot - - gochecknoglobals - - exhaustivestruct - - wsl - - nestif - - goerr113 - - nlreturn - - wrapcheck - - misspell - - cyclop - - whitespace - - paralleltest - - fast: false - -run: - tests: true - -issues: - exclude-use-default: true - diff --git a/lock/lock_test.go b/lock/lock_test.go index 132560c6..31c2f3da 100644 --- a/lock/lock_test.go +++ b/lock/lock_test.go @@ -23,14 +23,16 @@ var ( helperBinary = "./locktest" ) -func init() { +func TestMain(m *testing.M) { if platform.IsWindows() { - helperBinary += ".exe" + helperBinary = `.\locktest.exe` } cmd := exec.Command("go", "build", "-o", helperBinary, "./test") if err := cmd.Run(); err != nil { fmt.Fprintf(os.Stderr, "Error building helper binary: %s\n", err) } + m.Run() + _ = os.Remove(helperBinary) } func getTempfile(t *testing.T) string { @@ -124,42 +126,14 @@ func TestSetMorePID(t *testing.T) { assert.Equal(t, int32(13), pid) } -func TestProcessFinished(t *testing.T) { +func TestProcessPID(t *testing.T) { t.Parallel() childPID := 0 buffer := &bytes.Buffer{} - c := make(chan os.Signal, 1) - signal.Notify(c, os.Interrupt) - defer signal.Reset(os.Interrupt) - - cmd := shell.NewSignalledCommand("echo", []string{"Hello World!"}, c) - cmd.Stdout = buffer - cmd.SetPID = func(pid int) { - childPID = pid - } - _, _, err := cmd.Run() - require.NoError(t, err) - - // at that point, the child process should be finished - running, err := process.PidExists(int32(childPID)) - assert.NoError(t, err) - assert.False(t, running) -} - -func TestProcessNotFinished(t *testing.T) { - t.Parallel() - - childPID := 0 - buffer := &bytes.Buffer{} - - c := make(chan os.Signal, 1) - signal.Notify(c, os.Interrupt) - defer signal.Reset(os.Interrupt) - - // user the lock helper binary (we only need to wait for some time, we don't need the locking part) - cmd := shell.NewSignalledCommand(helperBinary, []string{"-wait", "100", "-lock", t.Name()}, c) + // use the lock helper binary (we only need to wait for some time, we don't need the locking part) + cmd := shell.NewCommand(helperBinary, []string{"-wait", "200", "-lock", filepath.Join(t.TempDir(), t.Name())}) cmd.Stdout = buffer // SetPID method is called right after we forked and have a PID available cmd.SetPID = func(pid int) { @@ -243,13 +217,8 @@ func TestForceLockWithRunningPID(t *testing.T) { assert.True(t, lock.TryAcquire()) assert.True(t, lock.HasLocked()) - // run a child process - c := make(chan os.Signal, 1) - signal.Notify(c, os.Interrupt) - defer signal.Reset(os.Interrupt) - // user the lock helper binary (we only need to wait for some time, we don't need the locking part) - cmd := shell.NewSignalledCommand(helperBinary, []string{"-wait", "100", "-lock", t.Name()}, c) + cmd := shell.NewCommand(helperBinary, []string{"-wait", "100", "-lock", filepath.Join(t.TempDir(), t.Name())}) cmd.SetPID = func(pid int) { lock.SetPID(pid) // make sure we cannot break the lock right now @@ -291,14 +260,14 @@ func TestLockIsRemovedAfterInterruptSignal(t *testing.T) { var err error buffer := &bytes.Buffer{} - cmd := exec.Command(helperBinary, "-wait", "400", "-lock", lockfile) + cmd := exec.Command(helperBinary, "-wait", "2000", "-lock", lockfile) cmd.Stdout = buffer cmd.Stderr = buffer err = cmd.Start() require.NoError(t, err) - time.Sleep(100 * time.Millisecond) + time.Sleep(200 * time.Millisecond) err = cmd.Process.Signal(syscall.SIGINT) require.NoError(t, err) @@ -317,14 +286,14 @@ func TestLockIsRemovedAfterInterruptSignalInsideShell(t *testing.T) { var err error buffer := &bytes.Buffer{} - cmd := exec.Command("sh", "-c", helperBinary+" -wait 600 -lock "+lockfile) + cmd := exec.Command("sh", "-c", helperBinary+" -wait 2000 -lock "+lockfile) cmd.Stdout = buffer cmd.Stderr = buffer err = cmd.Start() require.NoError(t, err) - time.Sleep(100 * time.Millisecond) + time.Sleep(200 * time.Millisecond) err = cmd.Process.Signal(syscall.SIGINT) require.NoError(t, err) From addd36e7abbe1dcaf42544e92e0a3cd3745f9604 Mon Sep 17 00:00:00 2001 From: Fred Date: Fri, 28 Jun 2024 17:54:02 +0100 Subject: [PATCH 4/4] fix linux test by using "exec" --- lock/lock_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lock/lock_test.go b/lock/lock_test.go index 31c2f3da..e50a64ff 100644 --- a/lock/lock_test.go +++ b/lock/lock_test.go @@ -286,7 +286,7 @@ func TestLockIsRemovedAfterInterruptSignalInsideShell(t *testing.T) { var err error buffer := &bytes.Buffer{} - cmd := exec.Command("sh", "-c", helperBinary+" -wait 2000 -lock "+lockfile) + cmd := exec.Command("sh", "-c", "exec "+helperBinary+" -wait 2000 -lock "+lockfile) cmd.Stdout = buffer cmd.Stderr = buffer