2018-03-18 17:43:38 +01:00
|
|
|
package defaults
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2018-05-07 22:01:36 +02:00
|
|
|
"encoding/base64"
|
2018-03-18 17:43:38 +01:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"strings"
|
2018-03-26 21:54:17 +02:00
|
|
|
"syscall"
|
2018-03-18 17:43:38 +01:00
|
|
|
"testing"
|
|
|
|
|
2018-03-28 19:19:58 +02:00
|
|
|
"github.com/pkg/xattr"
|
2018-03-18 17:43:38 +01:00
|
|
|
|
2018-05-07 22:01:36 +02:00
|
|
|
"github.com/rfjakob/gocryptfs/internal/cryptocore"
|
2018-03-18 17:43:38 +01:00
|
|
|
"github.com/rfjakob/gocryptfs/tests/test_helpers"
|
|
|
|
)
|
|
|
|
|
|
|
|
// On modern Linux distributions, /tmp may be on tmpfs,
|
|
|
|
// which does not support user xattrs. Try /var/tmp instead.
|
|
|
|
var alternateTestParentDir = "/var/tmp/gocryptfs-xattr-test-parent"
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
2018-03-26 21:54:17 +02:00
|
|
|
if !xattrSupported(test_helpers.TmpDir) {
|
2018-03-18 17:43:38 +01:00
|
|
|
test_helpers.SwitchTestParentDir(alternateTestParentDir)
|
|
|
|
}
|
2018-03-26 21:54:17 +02:00
|
|
|
if !xattrSupported(test_helpers.TmpDir) {
|
2018-03-18 17:43:38 +01:00
|
|
|
fmt.Printf("xattrs not supported on %q", test_helpers.TmpDir)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
test_helpers.ResetTmpDir(true)
|
2018-05-07 22:01:36 +02:00
|
|
|
// Write deterministic diriv so encrypted filenames are deterministic.
|
|
|
|
os.Remove(test_helpers.DefaultCipherDir + "/gocryptfs.diriv")
|
|
|
|
diriv := []byte("1234567890123456")
|
|
|
|
err := ioutil.WriteFile(test_helpers.DefaultCipherDir+"/gocryptfs.diriv", diriv, 0400)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
2018-03-18 17:43:38 +01:00
|
|
|
test_helpers.MountOrExit(test_helpers.DefaultCipherDir, test_helpers.DefaultPlainDir, "-zerokey")
|
|
|
|
r := m.Run()
|
|
|
|
test_helpers.UnmountPanic(test_helpers.DefaultPlainDir)
|
|
|
|
os.RemoveAll(test_helpers.TmpDir)
|
|
|
|
os.Exit(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestXattrSetGetRm(t *testing.T) {
|
|
|
|
attr := "user.foo"
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestXattrSetGetRm"
|
|
|
|
err := ioutil.WriteFile(fn, nil, 0700)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("creating empty file failed: %v", err)
|
|
|
|
}
|
|
|
|
// Set
|
|
|
|
val1 := []byte("123456789")
|
2018-05-27 20:09:48 +02:00
|
|
|
err = xattr.LSet(fn, attr, val1)
|
2018-03-18 17:43:38 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Read back
|
2018-05-27 20:09:48 +02:00
|
|
|
val2, err := xattr.LGet(fn, attr)
|
2018-03-18 17:43:38 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(val1, val2) {
|
|
|
|
t.Fatalf("wrong readback value: %v != %v", val1, val2)
|
|
|
|
}
|
|
|
|
// Remove
|
2018-05-27 20:09:48 +02:00
|
|
|
err = xattr.LRemove(fn, attr)
|
2018-03-18 17:43:38 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Read back
|
2018-05-27 20:09:48 +02:00
|
|
|
val3, err := xattr.LGet(fn, attr)
|
2018-03-18 17:43:38 +01:00
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("attr is still there after deletion!? val3=%v", val3)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-25 21:02:33 +02:00
|
|
|
func TestXattrSetEmpty(t *testing.T) {
|
|
|
|
attr := "user.foo"
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestXattrSetEmpty1"
|
|
|
|
err := ioutil.WriteFile(fn, nil, 0700)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("creating empty file failed: %v", err)
|
|
|
|
}
|
|
|
|
// Make sure it does not exist already
|
2018-05-27 20:09:48 +02:00
|
|
|
_, err = xattr.LGet(fn, attr)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("we should have got an error here")
|
|
|
|
}
|
|
|
|
// Set empty value
|
2018-05-27 20:09:48 +02:00
|
|
|
err = xattr.LSet(fn, attr, nil)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Read back
|
2018-05-27 20:09:48 +02:00
|
|
|
val, err := xattr.LGet(fn, attr)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(val) != 0 {
|
|
|
|
t.Errorf("wrong length: want=0 have=%d", len(val))
|
|
|
|
}
|
|
|
|
// Overwrite empty value with something
|
|
|
|
val1 := []byte("xyz123")
|
2018-05-27 20:09:48 +02:00
|
|
|
err = xattr.LSet(fn, attr, val1)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Read back
|
2018-05-27 20:09:48 +02:00
|
|
|
val2, err := xattr.LGet(fn, attr)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(val1, val2) {
|
|
|
|
t.Fatalf("wrong readback value: %v != %v", val1, val2)
|
|
|
|
}
|
|
|
|
// Overwrite something with empty value
|
2018-05-27 20:09:48 +02:00
|
|
|
err = xattr.LSet(fn, attr, nil)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Read back
|
2018-05-27 20:09:48 +02:00
|
|
|
val, err = xattr.LGet(fn, attr)
|
2018-03-25 21:02:33 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(val) != 0 {
|
|
|
|
t.Errorf("wrong length: want=0 have=%d", len(val2))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-18 17:43:38 +01:00
|
|
|
func TestXattrList(t *testing.T) {
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestXattrList"
|
|
|
|
err := ioutil.WriteFile(fn, nil, 0700)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("creating empty file failed: %v", err)
|
|
|
|
}
|
|
|
|
val := []byte("xxxxxxxxyyyyyyyyyyyyyyyzzzzzzzzzzzzz")
|
|
|
|
num := 20
|
|
|
|
for i := 1; i <= num; i++ {
|
|
|
|
attr := fmt.Sprintf("user.TestXattrList.%02d", i)
|
2018-05-27 20:09:48 +02:00
|
|
|
err = xattr.LSet(fn, attr, val)
|
2018-03-18 17:43:38 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
2018-05-27 20:09:48 +02:00
|
|
|
names, err := xattr.LList(fn)
|
2018-03-18 17:43:38 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(names) != num {
|
|
|
|
t.Errorf("wrong number of names, want=%d have=%d", num, len(names))
|
|
|
|
}
|
|
|
|
for _, n := range names {
|
|
|
|
if !strings.HasPrefix(n, "user.TestXattrList.") {
|
|
|
|
t.Errorf("unexpected attr name: %q", n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-03-26 21:54:17 +02:00
|
|
|
|
|
|
|
func xattrSupported(path string) bool {
|
2018-05-27 20:09:48 +02:00
|
|
|
_, err := xattr.LGet(path, "user.xattrSupported-dummy-value")
|
2018-03-26 21:54:17 +02:00
|
|
|
if err == nil {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
err2 := err.(*xattr.Error)
|
|
|
|
if err2.Err == syscall.EOPNOTSUPP {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
2018-05-07 22:01:36 +02:00
|
|
|
|
|
|
|
func TestBase64XattrRead(t *testing.T) {
|
|
|
|
attrName := "user.test"
|
|
|
|
attrName2 := "user.test2"
|
|
|
|
encryptedAttrName := "user.gocryptfs.LB1kHHVrX1OEBdLmj3LTKw"
|
|
|
|
encryptedAttrName2 := "user.gocryptfs.d2yn5l7-0zUVqviADw-Oyw"
|
|
|
|
attrValue := fmt.Sprintf("test.%d", cryptocore.RandUint64())
|
|
|
|
|
|
|
|
fileName := "TestBase64Xattr"
|
|
|
|
encryptedFileName := "BaGak7jIoqAZQMlP0N5uCw"
|
|
|
|
|
|
|
|
plainFn := test_helpers.DefaultPlainDir + "/" + fileName
|
|
|
|
encryptedFn := test_helpers.DefaultCipherDir + "/" + encryptedFileName
|
|
|
|
err := ioutil.WriteFile(plainFn, nil, 0700)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("creating empty file failed: %v", err)
|
|
|
|
}
|
|
|
|
if _, err2 := os.Stat(encryptedFn); os.IsNotExist(err2) {
|
|
|
|
t.Fatalf("encrypted file does not exist: %v", err2)
|
|
|
|
}
|
2018-05-27 20:09:48 +02:00
|
|
|
xattr.LSet(plainFn, attrName, []byte(attrValue))
|
2018-05-07 22:01:36 +02:00
|
|
|
|
2018-05-27 20:09:48 +02:00
|
|
|
encryptedAttrValue, err1 := xattr.LGet(encryptedFn, encryptedAttrName)
|
2018-05-07 22:01:36 +02:00
|
|
|
if err1 != nil {
|
|
|
|
t.Fatal(err1)
|
|
|
|
}
|
|
|
|
|
2018-05-27 20:09:48 +02:00
|
|
|
xattr.LSet(encryptedFn, encryptedAttrName2, encryptedAttrValue)
|
|
|
|
plainValue, err := xattr.LGet(plainFn, attrName2)
|
2018-05-07 22:01:36 +02:00
|
|
|
|
|
|
|
if err != nil || string(plainValue) != attrValue {
|
|
|
|
t.Fatalf("Attribute binary value decryption error %s != %s %v", string(plainValue), attrValue, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
encryptedAttrValue64 := base64.RawURLEncoding.EncodeToString(encryptedAttrValue)
|
2018-05-27 20:09:48 +02:00
|
|
|
xattr.LSet(encryptedFn, encryptedAttrName2, []byte(encryptedAttrValue64))
|
2018-05-07 22:01:36 +02:00
|
|
|
|
2018-05-27 20:09:48 +02:00
|
|
|
plainValue, err = xattr.LGet(plainFn, attrName2)
|
2018-05-07 22:01:36 +02:00
|
|
|
if err != nil || string(plainValue) != attrValue {
|
|
|
|
t.Fatalf("Attribute base64-encoded value decryption error %s != %s %v", string(plainValue), attrValue, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remount with -wpanic=false so gocryptfs does not panics when it sees
|
|
|
|
// the broken xattrs
|
|
|
|
test_helpers.UnmountPanic(test_helpers.DefaultPlainDir)
|
|
|
|
test_helpers.MountOrExit(test_helpers.DefaultCipherDir, test_helpers.DefaultPlainDir, "-zerokey", "-wpanic=false")
|
|
|
|
|
|
|
|
brokenVals := []string{
|
|
|
|
"111",
|
|
|
|
"raw-test-long-block123",
|
|
|
|
"raw-test-long-block123-xyz11111111111111111111111111111111111111",
|
|
|
|
"$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$",
|
|
|
|
}
|
|
|
|
for _, val := range brokenVals {
|
2018-05-27 20:09:48 +02:00
|
|
|
xattr.LSet(encryptedFn, encryptedAttrName2, []byte(val))
|
|
|
|
plainValue, err = xattr.LGet(plainFn, attrName2)
|
2018-05-07 22:01:36 +02:00
|
|
|
err2, _ := err.(*xattr.Error)
|
|
|
|
if err == nil || err2.Err != syscall.EIO {
|
|
|
|
t.Fatalf("Incorrect handling of broken data %s %v", string(plainValue), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|