2016-10-16 13:39:25 +02:00
|
|
|
|
// Tests and benchmarks performed with default settings only.
|
|
|
|
|
package defaults
|
|
|
|
|
|
|
|
|
|
import (
|
2016-11-07 20:58:12 +01:00
|
|
|
|
"bytes"
|
2017-05-01 18:36:25 +02:00
|
|
|
|
"io"
|
2017-07-11 23:19:58 +02:00
|
|
|
|
"io/ioutil"
|
2016-10-16 13:39:25 +02:00
|
|
|
|
"os"
|
2016-10-19 22:25:54 +02:00
|
|
|
|
"os/exec"
|
2020-07-23 22:55:07 +02:00
|
|
|
|
"path/filepath"
|
2017-05-01 18:36:25 +02:00
|
|
|
|
"runtime"
|
2018-05-17 23:14:33 +02:00
|
|
|
|
"strings"
|
2017-05-01 18:36:25 +02:00
|
|
|
|
"sync"
|
2016-10-16 13:39:25 +02:00
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
|
|
"github.com/rfjakob/gocryptfs/tests/test_helpers"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
|
|
|
|
test_helpers.ResetTmpDir(true)
|
2021-06-21 11:53:33 +02:00
|
|
|
|
// TestZerokey() in tests/cli verifies that mounting with `-zerokey` is equivalent
|
|
|
|
|
// to mounting with a config file with all-default options (just the masterkey
|
|
|
|
|
// set to all-zero).
|
2016-10-16 13:39:25 +02:00
|
|
|
|
test_helpers.MountOrExit(test_helpers.DefaultCipherDir, test_helpers.DefaultPlainDir, "-zerokey")
|
|
|
|
|
r := m.Run()
|
|
|
|
|
test_helpers.UnmountPanic(test_helpers.DefaultPlainDir)
|
|
|
|
|
os.Exit(r)
|
|
|
|
|
}
|
2016-10-19 22:25:54 +02:00
|
|
|
|
|
|
|
|
|
// Test that we get the right timestamp when extracting a tarball.
|
|
|
|
|
func Test1980Tar(t *testing.T) {
|
|
|
|
|
c := exec.Command("tar", "xzf", "1980.tar.gz", "-C", test_helpers.DefaultPlainDir)
|
|
|
|
|
c.Stderr = os.Stderr
|
|
|
|
|
c.Stdout = os.Stdout
|
|
|
|
|
err := c.Run()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
fi, err := os.Stat(test_helpers.DefaultPlainDir + "/1980.txt")
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
m := fi.ModTime().Unix()
|
|
|
|
|
if m != 315619323 {
|
|
|
|
|
t.Errorf("Wrong mtime: %d", m)
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-11-10 00:27:08 +01:00
|
|
|
|
|
2016-11-07 20:58:12 +01:00
|
|
|
|
// In gocryptfs before v1.2, the file header was only read once for each
|
|
|
|
|
// open. But truncating a file to zero will generate a new random file ID.
|
|
|
|
|
// The sequence below caused an I/O error to be returned.
|
|
|
|
|
func TestOpenTruncateRead(t *testing.T) {
|
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestTruncateWrite"
|
2018-12-27 12:03:00 +01:00
|
|
|
|
// First FD is used for write and truncate.
|
2016-11-07 20:58:12 +01:00
|
|
|
|
writeFd, err := os.Create(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
2018-03-05 21:56:30 +01:00
|
|
|
|
defer writeFd.Close()
|
2016-11-07 20:58:12 +01:00
|
|
|
|
abc := []byte("abc")
|
|
|
|
|
_, err = writeFd.WriteAt(abc, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
// Second FD is just for reading.
|
|
|
|
|
readFd, err := os.Open(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
2018-03-05 21:56:30 +01:00
|
|
|
|
defer readFd.Close()
|
2016-11-07 20:58:12 +01:00
|
|
|
|
content := make([]byte, 3)
|
|
|
|
|
_, err = readFd.ReadAt(content, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if !bytes.Equal(content, abc) {
|
|
|
|
|
t.Fatalf("wrong content: %s", string(content))
|
|
|
|
|
}
|
|
|
|
|
// Truncate to zero to generate a new file ID and write new content.
|
|
|
|
|
err = writeFd.Truncate(0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
xyz := []byte("xyz")
|
|
|
|
|
_, err = writeFd.WriteAt(xyz, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
// Try to read from the other FD.
|
|
|
|
|
_, err = readFd.ReadAt(content, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if !bytes.Equal(content, xyz) {
|
|
|
|
|
t.Fatalf("wrong content: %s", string(content))
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-01 17:49:37 +02:00
|
|
|
|
|
2017-05-01 18:36:25 +02:00
|
|
|
|
// TestWORead tries to read from a write-only FD.
|
2017-05-01 17:49:37 +02:00
|
|
|
|
func TestWORead(t *testing.T) {
|
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestWORead"
|
|
|
|
|
fd, err := os.OpenFile(fn, os.O_CREATE|os.O_WRONLY, 0600)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
defer fd.Close()
|
|
|
|
|
buf := make([]byte, 10)
|
|
|
|
|
_, err = fd.Read(buf)
|
|
|
|
|
if err == nil {
|
|
|
|
|
t.Error("Reading from write-only file should fail, but did not")
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-01 18:36:25 +02:00
|
|
|
|
|
|
|
|
|
// xfstests generic/124 triggers this warning:
|
|
|
|
|
// cipherSize 18 == header size: interrupted write?
|
|
|
|
|
// This test reproduces the problem.
|
|
|
|
|
func TestXfs124(t *testing.T) {
|
|
|
|
|
// GOMAXPROCS=8 and N=5000 seem to reliably trigger the problem. With N=1000,
|
|
|
|
|
// the test passes sometimes.
|
|
|
|
|
runtime.GOMAXPROCS(8)
|
|
|
|
|
N := 5000
|
|
|
|
|
|
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestXfs124"
|
|
|
|
|
fd, err := os.Create(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
defer fd.Close()
|
|
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
wg.Add(2)
|
|
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
|
buf := make([]byte, 10)
|
|
|
|
|
var err2 error
|
|
|
|
|
for i := 0; i < N; i++ {
|
|
|
|
|
err2 = fd.Truncate(0)
|
|
|
|
|
if err2 != nil {
|
|
|
|
|
panic(err2)
|
|
|
|
|
}
|
|
|
|
|
_, err2 = fd.WriteAt(buf, 0)
|
|
|
|
|
if err2 != nil {
|
|
|
|
|
panic(err2)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
wg.Done()
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
fd2, err := os.Open(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
defer fd2.Close()
|
|
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
|
buf := make([]byte, 10)
|
|
|
|
|
var err3 error
|
|
|
|
|
for i := 0; i < N; i++ {
|
|
|
|
|
_, err3 = fd2.ReadAt(buf, 0)
|
|
|
|
|
if err3 == io.EOF {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
if err3 != nil {
|
|
|
|
|
panic(err3)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
wg.Done()
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
}
|
2017-07-11 23:19:58 +02:00
|
|
|
|
|
|
|
|
|
func TestWrite0200File(t *testing.T) {
|
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestWrite0200File"
|
|
|
|
|
err := ioutil.WriteFile(fn, nil, 0200)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("creating empty file failed: %v", err)
|
|
|
|
|
}
|
|
|
|
|
fd, err := os.OpenFile(fn, os.O_WRONLY, 0)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
fi, err := fd.Stat()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
perms := fi.Mode().Perm()
|
|
|
|
|
if perms != 0200 {
|
|
|
|
|
t.Fatal("wrong initial permissions")
|
|
|
|
|
}
|
|
|
|
|
defer fd.Close()
|
|
|
|
|
_, err = fd.Write(make([]byte, 10))
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
perms = fi.Mode().Perm()
|
|
|
|
|
if perms != 0200 {
|
|
|
|
|
t.Fatal("wrong restored permissions")
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-05-15 23:00:47 +02:00
|
|
|
|
|
2020-07-12 13:43:34 +02:00
|
|
|
|
// TestMvWarnings:
|
2018-05-15 23:00:47 +02:00
|
|
|
|
// When xattr support was introduced, mv threw warnings like these:
|
|
|
|
|
// mv: preserving permissions for ‘b/x’: Operation not permitted
|
|
|
|
|
// because we returned EPERM when it tried to set system.posix_acl_access.
|
|
|
|
|
// Now we return EOPNOTSUPP and mv is happy.
|
|
|
|
|
func TestMvWarnings(t *testing.T) {
|
|
|
|
|
fn := test_helpers.TmpDir + "/TestMvWarnings"
|
|
|
|
|
err := ioutil.WriteFile(fn, nil, 0600)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("creating file failed: %v", err)
|
|
|
|
|
}
|
|
|
|
|
cmd := exec.Command("mv", fn, test_helpers.DefaultPlainDir)
|
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Log(string(out))
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if len(out) != 0 {
|
|
|
|
|
t.Fatalf("Got warnings from mv:\n%s", string(out))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-17 23:14:33 +02:00
|
|
|
|
// Check for this bug in symlink handling:
|
|
|
|
|
// $ ln -s /asd/asdasd/asdasd b/foo
|
|
|
|
|
// $ mv b/foo .
|
|
|
|
|
// mv: listing attributes of 'b/foo': No such file or directory
|
|
|
|
|
// strace shows:
|
|
|
|
|
// llistxattr("b/foo", NULL, 0) = -1 ENOENT (No such file or directory)
|
|
|
|
|
func TestMvWarningSymlink(t *testing.T) {
|
|
|
|
|
fn := test_helpers.DefaultPlainDir + "/TestMvWarningSymlink"
|
|
|
|
|
err := os.Symlink("/foo/bar/baz", fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
cmd := exec.Command("mv", fn, test_helpers.TmpDir)
|
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Log(string(out))
|
2019-01-20 17:20:30 +01:00
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
|
t.Skip("mv on darwin chokes on broken symlinks, see https://github.com/rfjakob/gocryptfs/issues/349")
|
|
|
|
|
}
|
2018-05-17 23:14:33 +02:00
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if len(out) != 0 {
|
|
|
|
|
t.Log(strings.TrimSpace(string(out)))
|
|
|
|
|
t.Fatal("Got warnings")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-12 13:43:34 +02:00
|
|
|
|
// See TestMvWarnings.
|
2018-05-15 23:00:47 +02:00
|
|
|
|
func TestCpWarnings(t *testing.T) {
|
|
|
|
|
fn := test_helpers.TmpDir + "/TestCpWarnings"
|
|
|
|
|
err := ioutil.WriteFile(fn, []byte("foo"), 0600)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("creating file failed: %v", err)
|
|
|
|
|
}
|
|
|
|
|
cmd := exec.Command("cp", "-a", fn, test_helpers.DefaultPlainDir)
|
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Log(string(out))
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if len(out) != 0 {
|
|
|
|
|
t.Fatalf("Got warnings from cp -a:\n%s", string(out))
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-07-23 22:55:07 +02:00
|
|
|
|
|
2020-08-16 19:58:47 +02:00
|
|
|
|
// TestSeekData tests that fs.FileLseeker is implemented
|
2020-07-23 22:55:07 +02:00
|
|
|
|
func TestSeekData(t *testing.T) {
|
|
|
|
|
fn := filepath.Join(test_helpers.DefaultPlainDir, t.Name())
|
|
|
|
|
f, err := os.Create(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
var oneTiB int64 = 1024 * 1024 * 1024 * 1024
|
|
|
|
|
if _, err = f.Seek(oneTiB, 0); err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if _, err = f.Write([]byte("foo")); err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
f.Close()
|
|
|
|
|
|
|
|
|
|
const SEEK_DATA = 3
|
|
|
|
|
|
|
|
|
|
f, err = os.Open(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
off, err := f.Seek(1024*1024, SEEK_DATA)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
if off < oneTiB-1024*1024 {
|
|
|
|
|
t.Errorf("off=%d, expected=%d\n", off, oneTiB)
|
|
|
|
|
}
|
|
|
|
|
f.Close()
|
|
|
|
|
}
|
2020-07-26 19:49:26 +02:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
TestMd5sumMaintainers tries to repro this interesting
|
|
|
|
|
bug that was seen during gocryptfs v2.0 development:
|
|
|
|
|
|
|
|
|
|
$ md5sum linux-3.0/MAINTAINERS linux-3.0/MAINTAINERS linux-3.0/MAINTAINERS linux-3.0/MAINTAINERS
|
|
|
|
|
279b6ab0491e7532132e8f32afe6c04d linux-3.0/MAINTAINERS <-- WRONG!!!!
|
|
|
|
|
99cc9f0dfd86e63231b94edd43a43e02 linux-3.0/MAINTAINERS <-- correct
|
|
|
|
|
99cc9f0dfd86e63231b94edd43a43e02 linux-3.0/MAINTAINERS
|
|
|
|
|
99cc9f0dfd86e63231b94edd43a43e02 linux-3.0/MAINTAINERS
|
|
|
|
|
|
|
|
|
|
strace shows:
|
|
|
|
|
|
|
|
|
|
Bad
|
|
|
|
|
---
|
|
|
|
|
fstat(3, {st_mode=S_IFREG|0644, st_size=196745, ...}) = 0
|
|
|
|
|
read(3, "\n\tList of maintainers and how to"..., 32768) = 32768
|
|
|
|
|
read(3, "M:\tSylwester Nawrocki <s.nawrock"..., 32768) = 32768
|
|
|
|
|
read(3, "rs/scsi/eata*\n\nEATA ISA/EISA/PCI"..., 32768) = 32768
|
|
|
|
|
read(3, "F:\tDocumentation/isapnp.txt\nF:\td"..., 32768) = 32768
|
|
|
|
|
read(3, "hunkeey@googlemail.com>\nL:\tlinux"..., 32768) = 32768
|
2021-06-26 16:09:04 +02:00
|
|
|
|
read(3, "ach-spear3xx/\n\nSPEAR6XX MACHINE "..., 32768) = 32768 <--- WRONG LENGTH!!!
|
2020-07-26 19:49:26 +02:00
|
|
|
|
read(3, "", 32768) = 0
|
2021-06-26 16:09:04 +02:00
|
|
|
|
lseek(3, 0, SEEK_CUR) = 196608 <--- WRONG LENGTH!!!
|
2020-07-26 19:49:26 +02:00
|
|
|
|
close(3) = 0
|
|
|
|
|
fstat(1, {st_mode=S_IFCHR|0620, st_rdev=makedev(0x88, 0x2), ...}) = 0
|
|
|
|
|
write(1, "279b6ab0491e7532132e8f32afe6c04d"..., 56279b6ab0491e7532132e8f32afe6c04d linux-3.0/MAINTAINERS
|
|
|
|
|
|
|
|
|
|
Good
|
|
|
|
|
----
|
|
|
|
|
fstat(3, {st_mode=S_IFREG|0644, st_size=195191, ...}) = 0
|
|
|
|
|
read(3, "\n\tList of maintainers and how to"..., 32768) = 32768
|
|
|
|
|
read(3, "M:\tSylwester Nawrocki <s.nawrock"..., 32768) = 32768
|
|
|
|
|
read(3, "rs/scsi/eata*\n\nEATA ISA/EISA/PCI"..., 32768) = 32768
|
|
|
|
|
read(3, "F:\tDocumentation/isapnp.txt\nF:\td"..., 32768) = 32768
|
|
|
|
|
read(3, "hunkeey@googlemail.com>\nL:\tlinux"..., 32768) = 32768
|
|
|
|
|
read(3, "ach-spear3xx/\n\nSPEAR6XX MACHINE "..., 32768) = 31351
|
|
|
|
|
read(3, "", 4096) = 0
|
|
|
|
|
lseek(3, 0, SEEK_CUR) = 195191
|
|
|
|
|
close(3) = 0
|
|
|
|
|
fstat(1, {st_mode=S_IFCHR|0620, st_rdev=makedev(0x88, 0x2), ...}) = 0
|
|
|
|
|
write(1, "99cc9f0dfd86e63231b94edd43a43e02"..., 5699cc9f0dfd86e63231b94edd43a43e02 linux-3.0/MAINTAINERS
|
|
|
|
|
*/
|
|
|
|
|
func TestMd5sumMaintainers(t *testing.T) {
|
|
|
|
|
fn := filepath.Join(test_helpers.DefaultPlainDir, t.Name())
|
|
|
|
|
f, err := os.Create(fn)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
// Size of the MAINTAINERS file = 195191
|
|
|
|
|
const sizeWant = 195191
|
|
|
|
|
content := make([]byte, sizeWant)
|
|
|
|
|
_, err = f.Write(content)
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
f.Close()
|
|
|
|
|
|
|
|
|
|
// Remount to clear the linux kernel attr cache
|
|
|
|
|
// (otherwise we would have to wait 2 seconds for the entry to expire)
|
|
|
|
|
test_helpers.UnmountPanic(test_helpers.DefaultPlainDir)
|
|
|
|
|
test_helpers.MountOrExit(test_helpers.DefaultCipherDir, test_helpers.DefaultPlainDir, "-zerokey")
|
|
|
|
|
|
|
|
|
|
cmd := exec.Command("md5sum", fn, fn, fn, fn)
|
|
|
|
|
out2, err := cmd.CombinedOutput()
|
|
|
|
|
out := string(out2)
|
|
|
|
|
|
|
|
|
|
// 195191 zero bytes have this md5sum
|
|
|
|
|
const md5Want = "b99bf6917f688068acd49126f3b1b005"
|
|
|
|
|
|
|
|
|
|
n := strings.Count(out, md5Want)
|
|
|
|
|
if n != 4 {
|
|
|
|
|
t.Errorf("found %d instead of %d instances of %q", n, 4, md5Want)
|
|
|
|
|
t.Logf("full output:\n%s", out)
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-26 16:09:04 +02:00
|
|
|
|
|
|
|
|
|
func TestMaxlen(t *testing.T) {
|
|
|
|
|
workDir := filepath.Join(test_helpers.DefaultPlainDir, t.Name())
|
|
|
|
|
if err := os.Mkdir(workDir, 0700); err != nil {
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
cmd := exec.Command("../../contrib/maxlen.bash", workDir)
|
2021-06-26 19:12:52 +02:00
|
|
|
|
cmd.Env = []string{"QUICK=1"}
|
2021-06-26 16:09:04 +02:00
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Log(string(out))
|
|
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
want := `
|
|
|
|
|
Maximum filename length: 255
|
|
|
|
|
Maximum path length with 100 chars per subdir: 4095
|
|
|
|
|
`
|
|
|
|
|
if !strings.HasSuffix(string(out), want) {
|
|
|
|
|
t.Errorf("wrong output: %s", string(out))
|
|
|
|
|
}
|
|
|
|
|
}
|