You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
157 lines
3.9 KiB
Go
157 lines
3.9 KiB
Go
// Copyright 2015 Google Inc. All Rights Reserved.
|
|
//
|
|
// 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 interruptfs_test
|
|
|
|
import (
|
|
"bytes"
|
|
"os"
|
|
"os/exec"
|
|
"path"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/jacobsa/fuse/fuseutil"
|
|
"github.com/jacobsa/fuse/samples"
|
|
"github.com/jacobsa/fuse/samples/interruptfs"
|
|
. "github.com/jacobsa/oglematchers"
|
|
. "github.com/jacobsa/ogletest"
|
|
)
|
|
|
|
func TestInterruptFS(t *testing.T) { RunTests(t) }
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Boilerplate
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
type InterruptFSTest struct {
|
|
samples.SampleTest
|
|
fs *interruptfs.InterruptFS
|
|
}
|
|
|
|
func init() { RegisterTestSuite(&InterruptFSTest{}) }
|
|
|
|
var _ SetUpInterface = &InterruptFSTest{}
|
|
var _ TearDownInterface = &InterruptFSTest{}
|
|
|
|
func (t *InterruptFSTest) SetUp(ti *TestInfo) {
|
|
var err error
|
|
|
|
// Create the file system.
|
|
t.fs = interruptfs.New()
|
|
AssertEq(nil, err)
|
|
|
|
t.Server = fuseutil.NewFileSystemServer(t.fs)
|
|
|
|
// Mount it.
|
|
t.SampleTest.SetUp(ti)
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Test functions
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
func (t *InterruptFSTest) StatFoo() {
|
|
fi, err := os.Stat(path.Join(t.Dir, "foo"))
|
|
AssertEq(nil, err)
|
|
|
|
ExpectEq("foo", fi.Name())
|
|
ExpectEq(0777, fi.Mode())
|
|
ExpectFalse(fi.IsDir())
|
|
}
|
|
|
|
func (t *InterruptFSTest) InterruptedDuringRead() {
|
|
var err error
|
|
t.fs.EnableReadBlocking()
|
|
|
|
// Start a sub-process that attempts to read the file.
|
|
cmd := exec.Command("cat", path.Join(t.Dir, "foo"))
|
|
|
|
var cmdOutput bytes.Buffer
|
|
cmd.Stdout = &cmdOutput
|
|
cmd.Stderr = &cmdOutput
|
|
|
|
err = cmd.Start()
|
|
AssertEq(nil, err)
|
|
|
|
// Wait for the command in the background, writing to a channel when it is
|
|
// finished.
|
|
cmdErr := make(chan error)
|
|
go func() {
|
|
cmdErr <- cmd.Wait()
|
|
}()
|
|
|
|
// Wait for the read to make it to the file system.
|
|
t.fs.WaitForFirstRead()
|
|
|
|
// The command should be hanging on the read, and not yet have returned.
|
|
select {
|
|
case err = <-cmdErr:
|
|
AddFailure("Command returned early with error: %v", err)
|
|
AbortTest()
|
|
|
|
case <-time.After(10 * time.Millisecond):
|
|
}
|
|
|
|
// Send SIGINT.
|
|
cmd.Process.Signal(os.Interrupt)
|
|
|
|
// Now the command should return, with an appropriate error.
|
|
err = <-cmdErr
|
|
ExpectThat(err, Error(HasSubstr("signal")))
|
|
ExpectThat(err, Error(HasSubstr("interrupt")))
|
|
}
|
|
|
|
func (t *InterruptFSTest) InterruptedDuringFlush() {
|
|
var err error
|
|
t.fs.EnableFlushBlocking()
|
|
|
|
// Start a sub-process that attempts to read the file.
|
|
cmd := exec.Command("cat", path.Join(t.Dir, "foo"))
|
|
|
|
var cmdOutput bytes.Buffer
|
|
cmd.Stdout = &cmdOutput
|
|
cmd.Stderr = &cmdOutput
|
|
|
|
err = cmd.Start()
|
|
AssertEq(nil, err)
|
|
|
|
// Wait for the command in the background, writing to a channel when it is
|
|
// finished.
|
|
cmdErr := make(chan error)
|
|
go func() {
|
|
cmdErr <- cmd.Wait()
|
|
}()
|
|
|
|
// Wait for the flush to make it to the file system.
|
|
t.fs.WaitForFirstFlush()
|
|
|
|
// The command should be hanging on the flush, and not yet have returned.
|
|
select {
|
|
case err = <-cmdErr:
|
|
AddFailure("Command returned early with error: %v", err)
|
|
AbortTest()
|
|
|
|
case <-time.After(10 * time.Millisecond):
|
|
}
|
|
|
|
// Send SIGINT.
|
|
cmd.Process.Signal(os.Interrupt)
|
|
|
|
// Now the command should return, with an appropriate error.
|
|
err = <-cmdErr
|
|
ExpectThat(err, Error(HasSubstr("signal")))
|
|
ExpectThat(err, Error(HasSubstr("interrupt")))
|
|
}
|