backup_test.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. package winio
  2. import (
  3. "io"
  4. "io/ioutil"
  5. "os"
  6. "syscall"
  7. "testing"
  8. )
  9. var testFileName string
  10. func TestMain(m *testing.M) {
  11. f, err := ioutil.TempFile("", "tmp")
  12. if err != nil {
  13. panic(err)
  14. }
  15. testFileName = f.Name()
  16. f.Close()
  17. defer os.Remove(testFileName)
  18. os.Exit(m.Run())
  19. }
  20. func makeTestFile(makeADS bool) error {
  21. os.Remove(testFileName)
  22. f, err := os.Create(testFileName)
  23. if err != nil {
  24. return err
  25. }
  26. defer f.Close()
  27. _, err = f.Write([]byte("testing 1 2 3\n"))
  28. if err != nil {
  29. return err
  30. }
  31. if makeADS {
  32. a, err := os.Create(testFileName + ":ads.txt")
  33. if err != nil {
  34. return err
  35. }
  36. defer a.Close()
  37. _, err = a.Write([]byte("alternate data stream\n"))
  38. if err != nil {
  39. return err
  40. }
  41. }
  42. return nil
  43. }
  44. func TestBackupRead(t *testing.T) {
  45. err := makeTestFile(true)
  46. if err != nil {
  47. t.Fatal(err)
  48. }
  49. f, err := os.Open(testFileName)
  50. if err != nil {
  51. t.Fatal(err)
  52. }
  53. defer f.Close()
  54. r := NewBackupFileReader(f, false)
  55. defer r.Close()
  56. b, err := ioutil.ReadAll(r)
  57. if err != nil {
  58. t.Fatal(err)
  59. }
  60. if len(b) == 0 {
  61. t.Fatal("no data")
  62. }
  63. }
  64. func TestBackupStreamRead(t *testing.T) {
  65. err := makeTestFile(true)
  66. if err != nil {
  67. t.Fatal(err)
  68. }
  69. f, err := os.Open(testFileName)
  70. if err != nil {
  71. t.Fatal(err)
  72. }
  73. defer f.Close()
  74. r := NewBackupFileReader(f, false)
  75. defer r.Close()
  76. br := NewBackupStreamReader(r)
  77. gotData := false
  78. gotAltData := false
  79. for {
  80. hdr, err := br.Next()
  81. if err == io.EOF {
  82. break
  83. }
  84. if err != nil {
  85. t.Fatal(err)
  86. }
  87. switch hdr.Id {
  88. case BackupData:
  89. if gotData {
  90. t.Fatal("duplicate data")
  91. }
  92. if hdr.Name != "" {
  93. t.Fatalf("unexpected name %s", hdr.Name)
  94. }
  95. b, err := ioutil.ReadAll(br)
  96. if err != nil {
  97. t.Fatal(err)
  98. }
  99. if string(b) != "testing 1 2 3\n" {
  100. t.Fatalf("incorrect data %v", b)
  101. }
  102. gotData = true
  103. case BackupAlternateData:
  104. if gotAltData {
  105. t.Fatal("duplicate alt data")
  106. }
  107. if hdr.Name != ":ads.txt:$DATA" {
  108. t.Fatalf("incorrect name %s", hdr.Name)
  109. }
  110. b, err := ioutil.ReadAll(br)
  111. if err != nil {
  112. t.Fatal(err)
  113. }
  114. if string(b) != "alternate data stream\n" {
  115. t.Fatalf("incorrect data %v", b)
  116. }
  117. gotAltData = true
  118. default:
  119. t.Fatalf("unknown stream ID %d", hdr.Id)
  120. }
  121. }
  122. if !gotData || !gotAltData {
  123. t.Fatal("missing stream")
  124. }
  125. }
  126. func TestBackupStreamWrite(t *testing.T) {
  127. f, err := os.Create(testFileName)
  128. if err != nil {
  129. t.Fatal(err)
  130. }
  131. defer f.Close()
  132. w := NewBackupFileWriter(f, false)
  133. defer w.Close()
  134. data := "testing 1 2 3\n"
  135. altData := "alternate stream\n"
  136. br := NewBackupStreamWriter(w)
  137. err = br.WriteHeader(&BackupHeader{Id: BackupData, Size: int64(len(data))})
  138. if err != nil {
  139. t.Fatal(err)
  140. }
  141. n, err := br.Write([]byte(data))
  142. if err != nil {
  143. t.Fatal(err)
  144. }
  145. if n != len(data) {
  146. t.Fatal("short write")
  147. }
  148. err = br.WriteHeader(&BackupHeader{Id: BackupAlternateData, Size: int64(len(altData)), Name: ":ads.txt:$DATA"})
  149. if err != nil {
  150. t.Fatal(err)
  151. }
  152. n, err = br.Write([]byte(altData))
  153. if err != nil {
  154. t.Fatal(err)
  155. }
  156. if n != len(altData) {
  157. t.Fatal("short write")
  158. }
  159. f.Close()
  160. b, err := ioutil.ReadFile(testFileName)
  161. if err != nil {
  162. t.Fatal(err)
  163. }
  164. if string(b) != data {
  165. t.Fatalf("wrong data %v", b)
  166. }
  167. b, err = ioutil.ReadFile(testFileName + ":ads.txt")
  168. if err != nil {
  169. t.Fatal(err)
  170. }
  171. if string(b) != altData {
  172. t.Fatalf("wrong data %v", b)
  173. }
  174. }
  175. func makeSparseFile() error {
  176. os.Remove(testFileName)
  177. f, err := os.Create(testFileName)
  178. if err != nil {
  179. return err
  180. }
  181. defer f.Close()
  182. const (
  183. FSCTL_SET_SPARSE = 0x000900c4
  184. FSCTL_SET_ZERO_DATA = 0x000980c8
  185. )
  186. err = syscall.DeviceIoControl(syscall.Handle(f.Fd()), FSCTL_SET_SPARSE, nil, 0, nil, 0, nil, nil)
  187. if err != nil {
  188. return err
  189. }
  190. _, err = f.Write([]byte("testing 1 2 3\n"))
  191. if err != nil {
  192. return err
  193. }
  194. _, err = f.Seek(1000000, 0)
  195. if err != nil {
  196. return err
  197. }
  198. _, err = f.Write([]byte("more data later\n"))
  199. if err != nil {
  200. return err
  201. }
  202. return nil
  203. }
  204. func TestBackupSparseFile(t *testing.T) {
  205. err := makeSparseFile()
  206. if err != nil {
  207. t.Fatal(err)
  208. }
  209. f, err := os.Open(testFileName)
  210. if err != nil {
  211. t.Fatal(err)
  212. }
  213. defer f.Close()
  214. r := NewBackupFileReader(f, false)
  215. defer r.Close()
  216. br := NewBackupStreamReader(r)
  217. for {
  218. hdr, err := br.Next()
  219. if err == io.EOF {
  220. break
  221. }
  222. if err != nil {
  223. t.Fatal(err)
  224. }
  225. t.Log(hdr)
  226. }
  227. }