concurrency_test.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. // Copyright (C) 2014 The Syncthing Authors.
  2. //
  3. // This program is free software: you can redistribute it and/or modify it
  4. // under the terms of the GNU General Public License as published by the Free
  5. // Software Foundation, either version 3 of the License, or (at your option)
  6. // any later version.
  7. //
  8. // This program is distributed in the hope that it will be useful, but WITHOUT
  9. // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  10. // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  11. // more details.
  12. //
  13. // You should have received a copy of the GNU General Public License along
  14. // with this program. If not, see <http://www.gnu.org/licenses/>.
  15. package files_test
  16. import (
  17. "crypto/rand"
  18. "log"
  19. "os"
  20. "sync"
  21. "testing"
  22. "time"
  23. "github.com/syndtr/goleveldb/leveldb"
  24. "github.com/syndtr/goleveldb/leveldb/opt"
  25. "github.com/syndtr/goleveldb/leveldb/util"
  26. )
  27. var keys [][]byte
  28. func init() {
  29. for i := 0; i < nItems; i++ {
  30. keys = append(keys, randomData(1))
  31. }
  32. }
  33. const nItems = 10000
  34. func randomData(prefix byte) []byte {
  35. data := make([]byte, 1+32+64+32)
  36. _, err := rand.Reader.Read(data)
  37. if err != nil {
  38. panic(err)
  39. }
  40. return append([]byte{prefix}, data...)
  41. }
  42. func setItems(db *leveldb.DB) error {
  43. batch := new(leveldb.Batch)
  44. for _, k1 := range keys {
  45. k2 := randomData(2)
  46. // k2 -> data
  47. batch.Put(k2, randomData(42))
  48. // k1 -> k2
  49. batch.Put(k1, k2)
  50. }
  51. if testing.Verbose() {
  52. log.Printf("batch write (set) %p", batch)
  53. }
  54. return db.Write(batch, nil)
  55. }
  56. func clearItems(db *leveldb.DB) error {
  57. snap, err := db.GetSnapshot()
  58. if err != nil {
  59. return err
  60. }
  61. defer snap.Release()
  62. // Iterate over k2
  63. it := snap.NewIterator(util.BytesPrefix([]byte{1}), nil)
  64. defer it.Release()
  65. batch := new(leveldb.Batch)
  66. for it.Next() {
  67. k1 := it.Key()
  68. k2 := it.Value()
  69. // k2 should exist
  70. _, err := snap.Get(k2, nil)
  71. if err != nil {
  72. return err
  73. }
  74. // Delete the k1 => k2 mapping first
  75. batch.Delete(k1)
  76. // Then the k2 => data mapping
  77. batch.Delete(k2)
  78. }
  79. if testing.Verbose() {
  80. log.Printf("batch write (clear) %p", batch)
  81. }
  82. return db.Write(batch, nil)
  83. }
  84. func scanItems(db *leveldb.DB) error {
  85. snap, err := db.GetSnapshot()
  86. if testing.Verbose() {
  87. log.Printf("snap create %p", snap)
  88. }
  89. if err != nil {
  90. return err
  91. }
  92. defer func() {
  93. if testing.Verbose() {
  94. log.Printf("snap release %p", snap)
  95. }
  96. snap.Release()
  97. }()
  98. // Iterate from the start of k2 space to the end
  99. it := snap.NewIterator(util.BytesPrefix([]byte{1}), nil)
  100. defer it.Release()
  101. i := 0
  102. for it.Next() {
  103. // k2 => k1 => data
  104. k1 := it.Key()
  105. k2 := it.Value()
  106. _, err := snap.Get(k2, nil)
  107. if err != nil {
  108. log.Printf("k1: %x", k1)
  109. log.Printf("k2: %x (missing)", k2)
  110. return err
  111. }
  112. i++
  113. }
  114. if testing.Verbose() {
  115. log.Println("scanned", i)
  116. }
  117. return nil
  118. }
  119. func TestConcurrentSetClear(t *testing.T) {
  120. if testing.Short() {
  121. return
  122. }
  123. dur := 30 * time.Second
  124. t0 := time.Now()
  125. var wg sync.WaitGroup
  126. os.RemoveAll("testdata/concurrent-set-clear.db")
  127. db, err := leveldb.OpenFile("testdata/concurrent-set-clear.db", &opt.Options{CachedOpenFiles: 10})
  128. if err != nil {
  129. t.Fatal(err)
  130. }
  131. defer os.RemoveAll("testdata/concurrent-set-clear.db")
  132. errChan := make(chan error, 3)
  133. wg.Add(1)
  134. go func() {
  135. defer wg.Done()
  136. for time.Since(t0) < dur {
  137. if err := setItems(db); err != nil {
  138. errChan <- err
  139. return
  140. }
  141. if err := clearItems(db); err != nil {
  142. errChan <- err
  143. return
  144. }
  145. }
  146. }()
  147. wg.Add(1)
  148. go func() {
  149. defer wg.Done()
  150. for time.Since(t0) < dur {
  151. if err := scanItems(db); err != nil {
  152. errChan <- err
  153. return
  154. }
  155. }
  156. }()
  157. go func() {
  158. wg.Wait()
  159. errChan <- nil
  160. }()
  161. err = <-errChan
  162. if err != nil {
  163. t.Error(err)
  164. }
  165. db.Close()
  166. }
  167. func TestConcurrentSetOnly(t *testing.T) {
  168. if testing.Short() {
  169. return
  170. }
  171. dur := 30 * time.Second
  172. t0 := time.Now()
  173. var wg sync.WaitGroup
  174. os.RemoveAll("testdata/concurrent-set-only.db")
  175. db, err := leveldb.OpenFile("testdata/concurrent-set-only.db", &opt.Options{CachedOpenFiles: 10})
  176. if err != nil {
  177. t.Fatal(err)
  178. }
  179. defer os.RemoveAll("testdata/concurrent-set-only.db")
  180. errChan := make(chan error, 3)
  181. wg.Add(1)
  182. go func() {
  183. defer wg.Done()
  184. for time.Since(t0) < dur {
  185. if err := setItems(db); err != nil {
  186. errChan <- err
  187. return
  188. }
  189. }
  190. }()
  191. wg.Add(1)
  192. go func() {
  193. defer wg.Done()
  194. for time.Since(t0) < dur {
  195. if err := scanItems(db); err != nil {
  196. errChan <- err
  197. return
  198. }
  199. }
  200. }()
  201. go func() {
  202. wg.Wait()
  203. errChan <- nil
  204. }()
  205. err = <-errChan
  206. if err != nil {
  207. t.Error(err)
  208. }
  209. }