perkeep/cmd/pk-put/camput_test.go

240 lines
6.3 KiB
Go

/*
Copyright 2011 The Perkeep Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package main
import (
"bytes"
"io"
"os"
"path/filepath"
"runtime"
"strings"
"testing"
"time"
"perkeep.org/pkg/cmdmain"
)
// env is the environment that a pk-put test runs within.
type env struct {
// stdin is the standard input, or /dev/null if nil
stdin io.Reader
// Timeout optionally specifies the timeout on the command.
Timeout time.Duration
// TODO(bradfitz): vfs files.
}
func (e *env) timeout() time.Duration {
if e.Timeout != 0 {
return e.Timeout
}
return 15 * time.Second
}
func (e *env) Run(args ...string) (out, err []byte, exitCode int) {
outbuf := new(bytes.Buffer)
errbuf := new(bytes.Buffer)
os.Args = append(os.Args[:1], args...)
cmdmain.Stdout, cmdmain.Stderr = outbuf, errbuf
if e.stdin == nil {
cmdmain.Stdin = strings.NewReader("")
} else {
cmdmain.Stdin = e.stdin
}
exitc := make(chan int, 1)
cmdmain.Exit = func(code int) {
exitc <- code
runtime.Goexit()
}
go func() {
cmdmain.Main()
cmdmain.Exit(0)
}()
select {
case exitCode = <-exitc:
case <-time.After(e.timeout()):
panic("timeout running command")
}
out = outbuf.Bytes()
err = errbuf.Bytes()
return
}
// TestUsageOnNoargs tests that we output a usage message when given no args, and return
// with a non-zero exit status.
func TestUsageOnNoargs(t *testing.T) {
var e env
out, err, code := e.Run()
if code != 1 {
t.Errorf("exit code = %d; want 1", code)
}
if len(out) != 0 {
t.Errorf("wanted nothing on stdout; got:\n%s", out)
}
if !bytes.Contains(err, []byte("Usage: pk-put")) {
t.Errorf("stderr doesn't contain usage. Got:\n%s", err)
}
}
// TestCommandUsage tests that we output a command-specific usage message and return
// with a non-zero exit status.
func TestCommandUsage(t *testing.T) {
var e env
out, err, code := e.Run("attr")
if code != 1 {
t.Errorf("exit code = %d; want 1", code)
}
if len(out) != 0 {
t.Errorf("wanted nothing on stdout; got:\n%s", out)
}
sub := "Attr takes 3 args: <permanode> <attr> <value>"
if !bytes.Contains(err, []byte(sub)) {
t.Errorf("stderr doesn't contain substring %q. Got:\n%s", sub, err)
}
}
func TestUploadingChangingDirectory(t *testing.T) {
// TODO(bradfitz):
// $ mkdir /tmp/somedir
// $ cp dev-pk-put /tmp/somedir
// $ ./dev-pk-put -file /tmp/somedir/ 2>&1 | tee /tmp/somedir/log
// ... verify it doesn't hang.
t.Logf("TODO")
}
func testWithTempDir(t *testing.T, fn func(tempDir string)) {
tempDir := t.TempDir()
confDir := filepath.Join(tempDir, "conf")
mustMkdir(t, confDir, 0700)
t.Setenv("CAMLI_CONFIG_DIR", confDir)
if err := os.WriteFile(filepath.Join(confDir, "client-config.json"), []byte("{}"), 0644); err != nil {
t.Fatal(err)
}
debugFlagOnce.Do(registerDebugFlags)
fn(tempDir)
}
// Tests that uploads of deep directory trees don't deadlock.
// See commit ee4550bff453526ebae460da1ad59f6e7f3efe77 for backstory
func TestUploadDirectories(t *testing.T) {
if testing.Short() {
t.Skip("skipping test in short mode.")
}
testWithTempDir(t, func(tempDir string) {
uploadRoot := filepath.Join(tempDir, "to_upload") // read from here
mustMkdir(t, uploadRoot, 0700)
blobDestDir := filepath.Join(tempDir, "blob_dest") // write to here
mustMkdir(t, blobDestDir, 0700)
// There are 10 stat cache workers. Simulate a slow lookup in
// the file-based ones (similar to reality), so the
// directory-based nodes make it to the upload worker first
// (where it would currently/previously deadlock waiting on
// children that are starved out) See
// ee4550bff453526ebae460da1ad59f6e7f3efe77.
testHookStatCache = func(el interface{}, ok bool) {
if !ok {
return
}
if ok && strings.HasSuffix(el.(*node).fullPath, ".txt") {
time.Sleep(50 * time.Millisecond)
}
}
defer func() { testHookStatCache = nil }()
dirIter := uploadRoot
for i := 0; i < 2; i++ {
dirPath := filepath.Join(dirIter, "dir")
mustMkdir(t, dirPath, 0700)
for _, baseFile := range []string{"file.txt", "FILE.txt"} {
filePath := filepath.Join(dirPath, baseFile)
if err := os.WriteFile(filePath, []byte("some file contents "+filePath), 0600); err != nil {
t.Fatalf("error writing to %s: %v", filePath, err)
}
t.Logf("Wrote file %s", filePath)
}
dirIter = dirPath
}
// Now set statCacheWorkers greater than uploadWorkers, so the
// sleep above can re-arrange the order that files get
// uploaded in, so the directory comes before the file. This
// was the old deadlock.
defer setAndRestore(&uploadWorkers, 1)()
defer setAndRestore(&statCacheWorkers, 5)()
e := &env{
Timeout: 5 * time.Second,
}
stdout, stderr, exit := e.Run(
"--blobdir="+blobDestDir,
"--havecache=false",
"--verbose=false", // useful to set true for debugging
"file",
uploadRoot)
if exit != 0 {
t.Fatalf("Exit status %d: stdout=[%s], stderr=[%s]", exit, stdout, stderr)
}
})
}
func TestCamputBlob(t *testing.T) {
if testing.Short() {
t.Skip("skipping test in short mode.")
}
testWithTempDir(t, func(tempDir string) {
blobDestDir := filepath.Join(tempDir, "blob_dest") // write to here
mustMkdir(t, blobDestDir, 0700)
e := &env{
Timeout: 5 * time.Second,
stdin: strings.NewReader("foo"),
}
stdout, stderr, exit := e.Run(
"--blobdir="+blobDestDir,
"--havecache=false",
"--verbose=false", // useful to set true for debugging
"blob", "-")
if exit != 0 {
t.Fatalf("Exit status %d: stdout=[%s], stderr=[%s]", exit, stdout, stderr)
}
if got, want := string(stdout), "sha224-0808f64e60d58979fcb676c96ec938270dea42445aeefcd3a4e6f8db\n"; got != want {
t.Errorf("Stdout = %q; want %q", got, want)
}
})
}
func mustMkdir(t *testing.T, fn string, mode int) {
if err := os.Mkdir(fn, 0700); err != nil {
t.Errorf("error creating dir %s: %v", fn, err)
}
}
func setAndRestore(dst *int, v int) func() {
old := *dst
*dst = v
return func() { *dst = old }
}