go/filelock/file_lock_test.go

78 lines
2.0 KiB
Go

// Copyright 2014 The LevelDB-Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package filelock
import (
"bytes"
"flag"
"io/ioutil"
"os"
"os/exec"
"testing"
)
var lockFilename = flag.String("lockfile", "", "File to lock. Non-empty value pimples child process.")
func spawn(prog, filename string) ([]byte, error) {
return exec.Command(prog, "-lockfile", filename, "-test.v",
"-test.run=TestLock$").CombinedOutput()
}
// TestLock locks a file, spawns a second process that attempts to grab the
// lock to verify it fails.
// Then it closes the lock, and spawns a third copy to verify it can be
// relocked.
func TestLock(t *testing.T) {
child := *lockFilename != ""
var filename string
if child {
filename = *lockFilename
} else {
f, err := ioutil.TempFile("", "")
if err != nil {
t.Fatal(err)
}
filename = f.Name()
defer os.Remove(filename)
}
// Avoid truncating an existing, non-empty file.
fi, err := os.Stat(filename)
if err == nil && fi.Size() != 0 {
t.Fatalf("The file %s is not empty", filename)
}
t.Logf("Locking %s\n", filename)
lock, err := Lock(filename)
if err != nil {
t.Fatalf("Could not lock %s: %v", filename, err)
}
if !child {
t.Logf("Spawning child, should fail to grab lock.")
out, err := spawn(os.Args[0], filename)
if err == nil {
t.Fatalf("Attempt to grab open lock should have failed.\n%s", out)
}
if !bytes.Contains(out, []byte("Could not lock")) {
t.Fatalf("Child failed with unexpected output: %s\n", out)
}
t.Logf("Child failed to grab lock as expected.")
}
t.Logf("Unlocking %s", filename)
if err := lock.Close(); err != nil {
t.Fatalf("Could not unlock %s: %v", filename, err)
}
if !child {
t.Logf("Spawning child, should successfully grab lock.")
if out, err := spawn(os.Args[0], filename); err != nil {
t.Fatalf("Attempt to re-open lock should have succeeded: %v\n%s",
err, out)
}
t.Logf("Child grabbed lock.")
}
}