xattr_test.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. // Copyright 2018 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. // +build darwin freebsd linux netbsd
  5. package unix_test
  6. import (
  7. "io/ioutil"
  8. "os"
  9. "runtime"
  10. "strings"
  11. "testing"
  12. "golang.org/x/sys/unix"
  13. )
  14. func TestXattr(t *testing.T) {
  15. defer chtmpdir(t)()
  16. f := "xattr1"
  17. touch(t, f)
  18. xattrName := "user.test"
  19. xattrDataSet := "gopher"
  20. err := unix.Setxattr(f, xattrName, []byte(xattrDataSet), 0)
  21. if err == unix.ENOTSUP || err == unix.EOPNOTSUPP {
  22. t.Skip("filesystem does not support extended attributes, skipping test")
  23. } else if err != nil {
  24. t.Fatalf("Setxattr: %v", err)
  25. }
  26. // find size
  27. size, err := unix.Listxattr(f, nil)
  28. if err != nil {
  29. t.Fatalf("Listxattr: %v", err)
  30. }
  31. if size <= 0 {
  32. t.Fatalf("Listxattr returned an empty list of attributes")
  33. }
  34. buf := make([]byte, size)
  35. read, err := unix.Listxattr(f, buf)
  36. if err != nil {
  37. t.Fatalf("Listxattr: %v", err)
  38. }
  39. xattrs := stringsFromByteSlice(buf[:read])
  40. xattrWant := xattrName
  41. if runtime.GOOS == "freebsd" {
  42. // On FreeBSD, the namespace is stored separately from the xattr
  43. // name and Listxattr doesn't return the namespace prefix.
  44. xattrWant = strings.TrimPrefix(xattrWant, "user.")
  45. }
  46. found := false
  47. for _, name := range xattrs {
  48. if name == xattrWant {
  49. found = true
  50. }
  51. }
  52. if !found {
  53. t.Errorf("Listxattr did not return previously set attribute '%s'", xattrName)
  54. }
  55. // find size
  56. size, err = unix.Getxattr(f, xattrName, nil)
  57. if err != nil {
  58. t.Fatalf("Getxattr: %v", err)
  59. }
  60. if size <= 0 {
  61. t.Fatalf("Getxattr returned an empty attribute")
  62. }
  63. xattrDataGet := make([]byte, size)
  64. _, err = unix.Getxattr(f, xattrName, xattrDataGet)
  65. if err != nil {
  66. t.Fatalf("Getxattr: %v", err)
  67. }
  68. got := string(xattrDataGet)
  69. if got != xattrDataSet {
  70. t.Errorf("Getxattr: expected attribute value %s, got %s", xattrDataSet, got)
  71. }
  72. err = unix.Removexattr(f, xattrName)
  73. if err != nil {
  74. t.Fatalf("Removexattr: %v", err)
  75. }
  76. n := "nonexistent"
  77. err = unix.Lsetxattr(n, xattrName, []byte(xattrDataSet), 0)
  78. if err != unix.ENOENT {
  79. t.Errorf("Lsetxattr: expected %v on non-existent file, got %v", unix.ENOENT, err)
  80. }
  81. _, err = unix.Lgetxattr(n, xattrName, nil)
  82. if err != unix.ENOENT {
  83. t.Errorf("Lgetxattr: %v", err)
  84. }
  85. s := "symlink1"
  86. err = os.Symlink(n, s)
  87. if err != nil {
  88. t.Fatal(err)
  89. }
  90. err = unix.Lsetxattr(s, xattrName, []byte(xattrDataSet), 0)
  91. if err != nil {
  92. // Linux and Android doen't support xattrs on symlinks according
  93. // to xattr(7), so just test that we get the proper error.
  94. if (runtime.GOOS != "linux" && runtime.GOOS != "android") || err != unix.EPERM {
  95. t.Fatalf("Lsetxattr: %v", err)
  96. }
  97. }
  98. }
  99. func TestFdXattr(t *testing.T) {
  100. file, err := ioutil.TempFile("", "TestFdXattr")
  101. if err != nil {
  102. t.Fatal(err)
  103. }
  104. defer os.Remove(file.Name())
  105. defer file.Close()
  106. fd := int(file.Fd())
  107. xattrName := "user.test"
  108. xattrDataSet := "gopher"
  109. err = unix.Fsetxattr(fd, xattrName, []byte(xattrDataSet), 0)
  110. if err == unix.ENOTSUP || err == unix.EOPNOTSUPP {
  111. t.Skip("filesystem does not support extended attributes, skipping test")
  112. } else if err != nil {
  113. t.Fatalf("Fsetxattr: %v", err)
  114. }
  115. // find size
  116. size, err := unix.Flistxattr(fd, nil)
  117. if err != nil {
  118. t.Fatalf("Flistxattr: %v", err)
  119. }
  120. if size <= 0 {
  121. t.Fatalf("Flistxattr returned an empty list of attributes")
  122. }
  123. buf := make([]byte, size)
  124. read, err := unix.Flistxattr(fd, buf)
  125. if err != nil {
  126. t.Fatalf("Flistxattr: %v", err)
  127. }
  128. xattrs := stringsFromByteSlice(buf[:read])
  129. xattrWant := xattrName
  130. if runtime.GOOS == "freebsd" {
  131. // On FreeBSD, the namespace is stored separately from the xattr
  132. // name and Listxattr doesn't return the namespace prefix.
  133. xattrWant = strings.TrimPrefix(xattrWant, "user.")
  134. }
  135. found := false
  136. for _, name := range xattrs {
  137. if name == xattrWant {
  138. found = true
  139. }
  140. }
  141. if !found {
  142. t.Errorf("Flistxattr did not return previously set attribute '%s'", xattrName)
  143. }
  144. // find size
  145. size, err = unix.Fgetxattr(fd, xattrName, nil)
  146. if err != nil {
  147. t.Fatalf("Fgetxattr: %v", err)
  148. }
  149. if size <= 0 {
  150. t.Fatalf("Fgetxattr returned an empty attribute")
  151. }
  152. xattrDataGet := make([]byte, size)
  153. _, err = unix.Fgetxattr(fd, xattrName, xattrDataGet)
  154. if err != nil {
  155. t.Fatalf("Fgetxattr: %v", err)
  156. }
  157. got := string(xattrDataGet)
  158. if got != xattrDataSet {
  159. t.Errorf("Fgetxattr: expected attribute value %s, got %s", xattrDataSet, got)
  160. }
  161. err = unix.Fremovexattr(fd, xattrName)
  162. if err != nil {
  163. t.Fatalf("Fremovexattr: %v", err)
  164. }
  165. }