All checks were successful
		
		
	
	Test / Create distribution (push) Successful in 33s
				
			Test / Sandbox (push) Successful in 2m15s
				
			Test / Hakurei (push) Successful in 3m11s
				
			Test / Hpkg (push) Successful in 4m0s
				
			Test / Sandbox (race detector) (push) Successful in 4m4s
				
			Test / Hakurei (race detector) (push) Successful in 4m52s
				
			Test / Flake checks (push) Successful in 1m30s
				
			This is a pretty solid implementation backed by robust tests, with a much cleaner interface. Signed-off-by: Ophestra <cat@gensokyo.uk>
		
			
				
	
	
		
			264 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			264 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2018 The 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.
 | 
						|
 | 
						|
// js and wasip1 do not support inter-process file locking.
 | 
						|
//
 | 
						|
//go:build !js && !wasip1
 | 
						|
 | 
						|
package lockedfile_test
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"testing"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"hakurei.app/container"
 | 
						|
	"hakurei.app/internal/lockedfile"
 | 
						|
	"hakurei.app/internal/lockedfile/internal/testexec"
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	quiescent            = 10 * time.Millisecond
 | 
						|
	probablyStillBlocked = 10 * time.Second
 | 
						|
)
 | 
						|
 | 
						|
func mustBlock(t *testing.T, desc string, f func()) (wait func(*testing.T)) {
 | 
						|
	t.Helper()
 | 
						|
 | 
						|
	done := make(chan struct{})
 | 
						|
	go func() {
 | 
						|
		f()
 | 
						|
		close(done)
 | 
						|
	}()
 | 
						|
 | 
						|
	timer := time.NewTimer(quiescent)
 | 
						|
	defer timer.Stop()
 | 
						|
	select {
 | 
						|
	case <-done:
 | 
						|
		t.Fatalf("%s unexpectedly did not block", desc)
 | 
						|
	case <-timer.C:
 | 
						|
	}
 | 
						|
 | 
						|
	return func(t *testing.T) {
 | 
						|
		logTimer := time.NewTimer(quiescent)
 | 
						|
		defer logTimer.Stop()
 | 
						|
 | 
						|
		select {
 | 
						|
		case <-logTimer.C:
 | 
						|
			// We expect the operation to have unblocked by now,
 | 
						|
			// but maybe it's just slow. Write to the test log
 | 
						|
			// in case the test times out, but don't fail it.
 | 
						|
			t.Helper()
 | 
						|
			t.Logf("%s is unexpectedly still blocked after %v", desc, quiescent)
 | 
						|
 | 
						|
			// Wait for the operation to actually complete, no matter how long it
 | 
						|
			// takes. If the test has deadlocked, this will cause the test to time out
 | 
						|
			// and dump goroutines.
 | 
						|
			<-done
 | 
						|
 | 
						|
		case <-done:
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func TestMutexExcludes(t *testing.T) {
 | 
						|
	t.Parallel()
 | 
						|
 | 
						|
	path := filepath.Join(t.TempDir(), "lock")
 | 
						|
	mu := lockedfile.MutexAt(path)
 | 
						|
	t.Logf("mu := MutexAt(_)")
 | 
						|
 | 
						|
	unlock, err := mu.Lock()
 | 
						|
	if err != nil {
 | 
						|
		t.Fatalf("mu.Lock: %v", err)
 | 
						|
	}
 | 
						|
	t.Logf("unlock, _  := mu.Lock()")
 | 
						|
 | 
						|
	mu2 := lockedfile.MutexAt(mu.Path)
 | 
						|
	t.Logf("mu2 := MutexAt(mu.Path)")
 | 
						|
 | 
						|
	wait := mustBlock(t, "mu2.Lock()", func() {
 | 
						|
		unlock2, err := mu2.Lock()
 | 
						|
		if err != nil {
 | 
						|
			t.Errorf("mu2.Lock: %v", err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
		t.Logf("unlock2, _ := mu2.Lock()")
 | 
						|
		t.Logf("unlock2()")
 | 
						|
		unlock2()
 | 
						|
	})
 | 
						|
 | 
						|
	t.Logf("unlock()")
 | 
						|
	unlock()
 | 
						|
	wait(t)
 | 
						|
}
 | 
						|
 | 
						|
func TestReadWaitsForLock(t *testing.T) {
 | 
						|
	t.Parallel()
 | 
						|
 | 
						|
	path := filepath.Join(t.TempDir(), "timestamp.txt")
 | 
						|
	f, err := lockedfile.Create(path)
 | 
						|
	if err != nil {
 | 
						|
		t.Fatalf("Create: %v", err)
 | 
						|
	}
 | 
						|
	defer f.Close()
 | 
						|
 | 
						|
	const (
 | 
						|
		part1 = "part 1\n"
 | 
						|
		part2 = "part 2\n"
 | 
						|
	)
 | 
						|
	_, err = f.WriteString(part1)
 | 
						|
	if err != nil {
 | 
						|
		t.Fatalf("WriteString: %v", err)
 | 
						|
	}
 | 
						|
	t.Logf("WriteString(%q) = <nil>", part1)
 | 
						|
 | 
						|
	wait := mustBlock(t, "Read", func() {
 | 
						|
		b, err := lockedfile.Read(path)
 | 
						|
		if err != nil {
 | 
						|
			t.Errorf("Read: %v", err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
 | 
						|
		const want = part1 + part2
 | 
						|
		got := string(b)
 | 
						|
		if got == want {
 | 
						|
			t.Logf("Read(_) = %q", got)
 | 
						|
		} else {
 | 
						|
			t.Errorf("Read(_) = %q, _; want %q", got, want)
 | 
						|
		}
 | 
						|
	})
 | 
						|
 | 
						|
	_, err = f.WriteString(part2)
 | 
						|
	if err != nil {
 | 
						|
		t.Errorf("WriteString: %v", err)
 | 
						|
	} else {
 | 
						|
		t.Logf("WriteString(%q) = <nil>", part2)
 | 
						|
	}
 | 
						|
	f.Close()
 | 
						|
 | 
						|
	wait(t)
 | 
						|
}
 | 
						|
 | 
						|
func TestCanLockExistingFile(t *testing.T) {
 | 
						|
	t.Parallel()
 | 
						|
 | 
						|
	path := filepath.Join(t.TempDir(), "existing.txt")
 | 
						|
	if err := os.WriteFile(path, []byte("ok"), 0777); err != nil {
 | 
						|
		t.Fatalf("os.WriteFile: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	f, err := lockedfile.Edit(path)
 | 
						|
	if err != nil {
 | 
						|
		t.Fatalf("first Edit: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	wait := mustBlock(t, "Edit", func() {
 | 
						|
		other, err := lockedfile.Edit(path)
 | 
						|
		if err != nil {
 | 
						|
			t.Errorf("second Edit: %v", err)
 | 
						|
		}
 | 
						|
		other.Close()
 | 
						|
	})
 | 
						|
 | 
						|
	f.Close()
 | 
						|
	wait(t)
 | 
						|
}
 | 
						|
 | 
						|
// TestSpuriousEDEADLK verifies that the spurious EDEADLK reported in
 | 
						|
// https://golang.org/issue/32817 no longer occurs.
 | 
						|
func TestSpuriousEDEADLK(t *testing.T) {
 | 
						|
	// 	P.1 locks file A.
 | 
						|
	// 	Q.3 locks file B.
 | 
						|
	// 	Q.3 blocks on file A.
 | 
						|
	// 	P.2 blocks on file B. (Spurious EDEADLK occurs here.)
 | 
						|
	// 	P.1 unlocks file A.
 | 
						|
	// 	Q.3 unblocks and locks file A.
 | 
						|
	// 	Q.3 unlocks files A and B.
 | 
						|
	// 	P.2 unblocks and locks file B.
 | 
						|
	// 	P.2 unlocks file B.
 | 
						|
 | 
						|
	dirVar := t.Name() + "DIR"
 | 
						|
 | 
						|
	if dir := os.Getenv(dirVar); dir != "" {
 | 
						|
		// Q.3 locks file B.
 | 
						|
		b, err := lockedfile.Edit(filepath.Join(dir, "B"))
 | 
						|
		if err != nil {
 | 
						|
			t.Fatal(err)
 | 
						|
		}
 | 
						|
		defer b.Close()
 | 
						|
 | 
						|
		if err := os.WriteFile(filepath.Join(dir, "locked"), []byte("ok"), 0666); err != nil {
 | 
						|
			t.Fatal(err)
 | 
						|
		}
 | 
						|
 | 
						|
		// Q.3 blocks on file A.
 | 
						|
		a, err := lockedfile.Edit(filepath.Join(dir, "A"))
 | 
						|
		// Q.3 unblocks and locks file A.
 | 
						|
		if err != nil {
 | 
						|
			t.Fatal(err)
 | 
						|
		}
 | 
						|
		defer a.Close()
 | 
						|
 | 
						|
		// Q.3 unlocks files A and B.
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	dir := t.TempDir()
 | 
						|
 | 
						|
	// P.1 locks file A.
 | 
						|
	a, err := lockedfile.Edit(filepath.Join(dir, "A"))
 | 
						|
	if err != nil {
 | 
						|
		t.Fatal(err)
 | 
						|
	}
 | 
						|
 | 
						|
	cmd := testexec.CommandContext(t, t.Context(), container.MustExecutable(nil), "-test.run=^"+t.Name()+"$")
 | 
						|
	cmd.Env = append(os.Environ(), fmt.Sprintf("%s=%s", dirVar, dir))
 | 
						|
 | 
						|
	qDone := make(chan struct{})
 | 
						|
	waitQ := mustBlock(t, "Edit A and B in subprocess", func() {
 | 
						|
		out, err := cmd.CombinedOutput()
 | 
						|
		if err != nil {
 | 
						|
			t.Errorf("%v:\n%s", err, out)
 | 
						|
		}
 | 
						|
		close(qDone)
 | 
						|
	})
 | 
						|
 | 
						|
	// Wait until process Q has either failed or locked file B.
 | 
						|
	// Otherwise, P.2 might not block on file B as intended.
 | 
						|
locked:
 | 
						|
	for {
 | 
						|
		if _, err := os.Stat(filepath.Join(dir, "locked")); !os.IsNotExist(err) {
 | 
						|
			break locked
 | 
						|
		}
 | 
						|
		timer := time.NewTimer(1 * time.Millisecond)
 | 
						|
		select {
 | 
						|
		case <-qDone:
 | 
						|
			timer.Stop()
 | 
						|
			break locked
 | 
						|
		case <-timer.C:
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	waitP2 := mustBlock(t, "Edit B", func() {
 | 
						|
		// P.2 blocks on file B. (Spurious EDEADLK occurs here.)
 | 
						|
		b, err := lockedfile.Edit(filepath.Join(dir, "B"))
 | 
						|
		// P.2 unblocks and locks file B.
 | 
						|
		if err != nil {
 | 
						|
			t.Error(err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
		// P.2 unlocks file B.
 | 
						|
		b.Close()
 | 
						|
	})
 | 
						|
 | 
						|
	// P.1 unlocks file A.
 | 
						|
	a.Close()
 | 
						|
 | 
						|
	waitQ(t)
 | 
						|
	waitP2(t)
 | 
						|
}
 |