file_system_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. package fileSystem
  2. import (
  3. "math/rand"
  4. "strconv"
  5. "testing"
  6. "time"
  7. )
  8. func TestCreateAndGet(t *testing.T) {
  9. fs := New()
  10. fs.Create("/foobar", "bar", Permanent, 1, 1)
  11. // already exist, create should fail
  12. _, err := fs.Create("/foobar", "bar", Permanent, 1, 1)
  13. if err == nil {
  14. t.Fatal("Create should fail")
  15. }
  16. fs.Delete("/foobar", true, 1, 1)
  17. // this should create successfully
  18. createAndGet(fs, "/foobar", t)
  19. createAndGet(fs, "/foo/bar", t)
  20. createAndGet(fs, "/foo/foo/bar", t)
  21. // meet file, create should fail
  22. _, err = fs.Create("/foo/bar/bar", "bar", Permanent, 2, 1)
  23. if err == nil {
  24. t.Fatal("Create should fail")
  25. }
  26. // create a directory
  27. _, err = fs.Create("/fooDir", "", Permanent, 3, 1)
  28. if err != nil {
  29. t.Fatal("Cannot create /fooDir")
  30. }
  31. e, err := fs.Get("/fooDir", false, false, 3, 1)
  32. if err != nil || e.Dir != true {
  33. t.Fatal("Cannot create /fooDir ")
  34. }
  35. // create a file under directory
  36. _, err = fs.Create("/fooDir/bar", "bar", Permanent, 4, 1)
  37. if err != nil {
  38. t.Fatal("Cannot create /fooDir/bar = bar")
  39. }
  40. }
  41. func TestUpdateFile(t *testing.T) {
  42. fs := New()
  43. _, err := fs.Create("/foo/bar", "bar", Permanent, 1, 1)
  44. if err != nil {
  45. t.Fatalf("cannot create %s=bar [%s]", "/foo/bar", err.Error())
  46. }
  47. _, err = fs.Update("/foo/bar", "barbar", Permanent, 2, 1)
  48. if err != nil {
  49. t.Fatalf("cannot update %s=barbar [%s]", "/foo/bar", err.Error())
  50. }
  51. e, err := fs.Get("/foo/bar", false, false, 2, 1)
  52. if err != nil {
  53. t.Fatalf("cannot get %s [%s]", "/foo/bar", err.Error())
  54. }
  55. if e.Value != "barbar" {
  56. t.Fatalf("expect value of %s is barbar [%s]", "/foo/bar", e.Value)
  57. }
  58. // create a directory, update its ttl, to see if it will be deleted
  59. _, err = fs.Create("/foo/foo", "", Permanent, 3, 1)
  60. if err != nil {
  61. t.Fatalf("cannot create dir [%s] [%s]", "/foo/foo", err.Error())
  62. }
  63. _, err = fs.Create("/foo/foo/foo1", "bar1", Permanent, 4, 1)
  64. if err != nil {
  65. t.Fatal("cannot create [%s]", err.Error())
  66. }
  67. _, err = fs.Create("/foo/foo/foo2", "", Permanent, 5, 1)
  68. if err != nil {
  69. t.Fatal("cannot create [%s]", err.Error())
  70. }
  71. _, err = fs.Create("/foo/foo/foo2/boo", "boo1", Permanent, 6, 1)
  72. if err != nil {
  73. t.Fatal("cannot create [%s]", err.Error())
  74. }
  75. expire := time.Now().Add(time.Second * 2)
  76. _, err = fs.Update("/foo/foo", "", expire, 7, 1)
  77. if err != nil {
  78. t.Fatalf("cannot update dir [%s] [%s]", "/foo/foo", err.Error())
  79. }
  80. // sleep 50ms, it should still reach the node
  81. time.Sleep(time.Microsecond * 50)
  82. e, err = fs.Get("/foo/foo", true, false, 7, 1)
  83. if err != nil || e.Key != "/foo/foo" {
  84. t.Fatalf("cannot get dir before expiration [%s]", err.Error())
  85. }
  86. if e.KVPairs[0].Key != "/foo/foo/foo1" || e.KVPairs[0].Value != "bar1" {
  87. t.Fatalf("cannot get sub node before expiration [%s]", err.Error())
  88. }
  89. if e.KVPairs[1].Key != "/foo/foo/foo2" || e.KVPairs[1].Dir != true {
  90. t.Fatalf("cannot get sub dir before expiration [%s]", err.Error())
  91. }
  92. /*if e.KVPairs[2].Key != "/foo/foo/foo2/boo" || e.KVPairs[2].Value != "boo1" {
  93. t.Fatalf("cannot get sub node of sub dir before expiration [%s]", err.Error())
  94. }*/
  95. // wait for expiration
  96. time.Sleep(time.Second * 3)
  97. e, err = fs.Get("/foo/foo", true, false, 7, 1)
  98. if err == nil {
  99. t.Fatal("still can get dir after expiration [%s]")
  100. }
  101. _, err = fs.Get("/foo/foo/foo1", true, false, 7, 1)
  102. if err == nil {
  103. t.Fatal("still can get sub node after expiration [%s]")
  104. }
  105. _, err = fs.Get("/foo/foo/foo2", true, false, 7, 1)
  106. if err == nil {
  107. t.Fatal("still can get sub dir after expiration [%s]")
  108. }
  109. _, err = fs.Get("/foo/foo/foo2/boo", true, false, 7, 1)
  110. if err == nil {
  111. t.Fatalf("still can get sub node of sub dir after expiration [%s]", err.Error())
  112. }
  113. }
  114. func TestListDirectory(t *testing.T) {
  115. fs := New()
  116. // create dir /foo
  117. // set key-value /foo/foo=bar
  118. fs.Create("/foo/foo", "bar", Permanent, 1, 1)
  119. // create dir /foo/fooDir
  120. // set key-value /foo/fooDir/foo=bar
  121. fs.Create("/foo/fooDir/foo", "bar", Permanent, 2, 1)
  122. e, err := fs.Get("/foo", true, false, 2, 1)
  123. if err != nil {
  124. t.Fatalf("%v", err)
  125. }
  126. if len(e.KVPairs) != 2 {
  127. t.Fatalf("wrong number of kv pairs [%d/2]", len(e.KVPairs))
  128. }
  129. if e.KVPairs[0].Key != "/foo/foo" || e.KVPairs[0].Value != "bar" {
  130. t.Fatalf("wrong kv [/foo/foo/ / %s] -> [bar / %s]", e.KVPairs[0].Key, e.KVPairs[0].Value)
  131. }
  132. if e.KVPairs[1].Key != "/foo/fooDir" || e.KVPairs[1].Dir != true {
  133. t.Fatalf("wrong kv [/foo/fooDir/ / %s] -> [true / %v]", e.KVPairs[1].Key, e.KVPairs[1].Dir)
  134. }
  135. if e.KVPairs[1].KVPairs[0].Key != "/foo/fooDir/foo" || e.KVPairs[1].KVPairs[0].Value != "bar" {
  136. t.Fatalf("wrong kv [/foo/fooDir/foo / %s] -> [bar / %v]", e.KVPairs[1].KVPairs[0].Key, e.KVPairs[1].KVPairs[0].Value)
  137. }
  138. // test hidden node
  139. // create dir /foo/_hidden
  140. // set key-value /foo/_hidden/foo -> bar
  141. fs.Create("/foo/_hidden/foo", "bar", Permanent, 3, 1)
  142. e, _ = fs.Get("/foo", false, false, 2, 1)
  143. if len(e.KVPairs) != 2 {
  144. t.Fatalf("hidden node is not hidden! %s", e.KVPairs[2].Key)
  145. }
  146. }
  147. func TestRemove(t *testing.T) {
  148. fs := New()
  149. fs.Create("/foo", "bar", Permanent, 1, 1)
  150. _, err := fs.Delete("/foo", false, 1, 1)
  151. if err != nil {
  152. t.Fatalf("cannot delete %s [%s]", "/foo", err.Error())
  153. }
  154. _, err = fs.Get("/foo", false, false, 1, 1)
  155. if err == nil || err.Error() != "Key Not Found" {
  156. t.Fatalf("can get the node after deletion")
  157. }
  158. fs.Create("/foo/bar", "bar", Permanent, 1, 1)
  159. fs.Create("/foo/car", "car", Permanent, 1, 1)
  160. fs.Create("/foo/dar/dar", "dar", Permanent, 1, 1)
  161. _, err = fs.Delete("/foo", false, 1, 1)
  162. if err == nil {
  163. t.Fatalf("should not be able to delete a directory without recursive")
  164. }
  165. _, err = fs.Delete("/foo", true, 1, 1)
  166. if err != nil {
  167. t.Fatalf("cannot delete %s [%s]", "/foo", err.Error())
  168. }
  169. _, err = fs.Get("/foo", false, false, 1, 1)
  170. if err == nil || err.Error() != "Key Not Found" {
  171. t.Fatalf("can get the node after deletion ")
  172. }
  173. }
  174. func TestExpire(t *testing.T) {
  175. fs := New()
  176. expire := time.Now().Add(time.Second)
  177. fs.Create("/foo", "bar", expire, 1, 1)
  178. _, err := fs.InternalGet("/foo", 1, 1)
  179. if err != nil {
  180. t.Fatalf("can not get the node")
  181. }
  182. time.Sleep(time.Second * 2)
  183. _, err = fs.InternalGet("/foo", 1, 1)
  184. if err == nil {
  185. t.Fatalf("can get the node after expiration time")
  186. }
  187. // test if we can reach the node before expiration
  188. expire = time.Now().Add(time.Second)
  189. fs.Create("/foo", "bar", expire, 1, 1)
  190. time.Sleep(time.Millisecond * 50)
  191. _, err = fs.InternalGet("/foo", 1, 1)
  192. if err != nil {
  193. t.Fatalf("cannot get the node before expiration", err.Error())
  194. }
  195. expire = time.Now().Add(time.Second)
  196. fs.Create("/foo", "bar", expire, 1, 1)
  197. _, err = fs.Delete("/foo", false, 1, 1)
  198. if err != nil {
  199. t.Fatalf("cannot delete the node before expiration", err.Error())
  200. }
  201. }
  202. func TestTestAndSet(t *testing.T) { // TODO prevValue == nil ?
  203. fs := New()
  204. fs.Create("/foo", "bar", Permanent, 1, 1)
  205. // test on wrong previous value
  206. _, err := fs.TestAndSet("/foo", "barbar", 0, "car", Permanent, 2, 1)
  207. if err == nil {
  208. t.Fatal("test and set should fail barbar != bar")
  209. }
  210. // test on value
  211. e, err := fs.TestAndSet("/foo", "bar", 0, "car", Permanent, 3, 1)
  212. if err != nil {
  213. t.Fatal("test and set should succeed bar == bar")
  214. }
  215. if e.PrevValue != "bar" || e.Value != "car" {
  216. t.Fatalf("[%v/%v] [%v/%v]", e.PrevValue, "bar", e.Value, "car")
  217. }
  218. // test on index
  219. e, err = fs.TestAndSet("/foo", "", 3, "bar", Permanent, 4, 1)
  220. if err != nil {
  221. t.Fatal("test and set should succeed index 3 == 3")
  222. }
  223. if e.PrevValue != "car" || e.Value != "bar" {
  224. t.Fatalf("[%v/%v] [%v/%v]", e.PrevValue, "car", e.Value, "bar")
  225. }
  226. }
  227. func TestWatch(t *testing.T) {
  228. fs := New()
  229. // watch at a deeper path
  230. c, _ := fs.WatcherHub.watch("/foo/foo/foo", false, 0)
  231. fs.Create("/foo/foo/foo", "bar", Permanent, 1, 1)
  232. e := nonblockingRetrive(c)
  233. if e.Key != "/foo/foo/foo" {
  234. t.Fatal("watch for Create node fails")
  235. }
  236. c, _ = fs.WatcherHub.watch("/foo/foo/foo", false, 0)
  237. fs.Update("/foo/foo/foo", "car", Permanent, 2, 1)
  238. e = nonblockingRetrive(c)
  239. if e.Key != "/foo/foo/foo" {
  240. t.Fatal("watch for Update node fails")
  241. }
  242. c, _ = fs.WatcherHub.watch("/foo/foo/foo", false, 0)
  243. fs.TestAndSet("/foo/foo/foo", "car", 0, "bar", Permanent, 3, 1)
  244. e = nonblockingRetrive(c)
  245. if e.Key != "/foo/foo/foo" {
  246. t.Fatal("watch for TestAndSet node fails")
  247. }
  248. c, _ = fs.WatcherHub.watch("/foo/foo/foo", false, 0)
  249. fs.Delete("/foo", true, 4, 1) //recursively delete
  250. e = nonblockingRetrive(c)
  251. if e.Key != "/foo" {
  252. t.Fatal("watch for Delete node fails")
  253. }
  254. // watch at a prefix
  255. c, _ = fs.WatcherHub.watch("/foo", true, 0)
  256. fs.Create("/foo/foo/boo", "bar", Permanent, 5, 1)
  257. e = nonblockingRetrive(c)
  258. if e.Key != "/foo/foo/boo" {
  259. t.Fatal("watch for Create subdirectory fails")
  260. }
  261. c, _ = fs.WatcherHub.watch("/foo", true, 0)
  262. fs.Update("/foo/foo/boo", "foo", Permanent, 6, 1)
  263. e = nonblockingRetrive(c)
  264. if e.Key != "/foo/foo/boo" {
  265. t.Fatal("watch for Update subdirectory fails")
  266. }
  267. c, _ = fs.WatcherHub.watch("/foo", true, 0)
  268. fs.TestAndSet("/foo/foo/boo", "foo", 0, "bar", Permanent, 7, 1)
  269. e = nonblockingRetrive(c)
  270. if e.Key != "/foo/foo/boo" {
  271. t.Fatal("watch for TestAndSet subdirectory fails")
  272. }
  273. c, _ = fs.WatcherHub.watch("/foo", true, 0)
  274. fs.Delete("/foo/foo/boo", false, 8, 1)
  275. e = nonblockingRetrive(c)
  276. if e.Key != "/foo/foo/boo" {
  277. t.Fatal("watch for Delete subdirectory fails")
  278. }
  279. }
  280. func TestSort(t *testing.T) {
  281. fs := New()
  282. // simulating random creation
  283. keys := GenKeys(80, 4)
  284. i := uint64(1)
  285. for _, k := range keys {
  286. _, err := fs.Create(k, "bar", Permanent, i, 1)
  287. if err != nil {
  288. panic(err)
  289. } else {
  290. i++
  291. }
  292. }
  293. e, err := fs.Get("/foo", true, true, i, 1)
  294. if err != nil {
  295. t.Fatalf("get dir nodes failed [%s]", err.Error())
  296. }
  297. for i, k := range e.KVPairs[:len(e.KVPairs)-1] {
  298. if k.Key >= e.KVPairs[i+1].Key {
  299. t.Fatalf("sort failed, [%s] should be placed after [%s]", k.Key, e.KVPairs[i+1].Key)
  300. }
  301. if k.Dir {
  302. recursiveTestSort(k, t)
  303. }
  304. }
  305. if k := e.KVPairs[len(e.KVPairs)-1]; k.Dir {
  306. recursiveTestSort(k, t)
  307. }
  308. }
  309. func TestSaveAndRecover(t *testing.T) {
  310. fs := New()
  311. // simulating random creation
  312. keys := GenKeys(8, 4)
  313. i := uint64(1)
  314. for _, k := range keys {
  315. _, err := fs.Create(k, "bar", Permanent, i, 1)
  316. if err != nil {
  317. panic(err)
  318. } else {
  319. i++
  320. }
  321. }
  322. // create a node with expiration
  323. // test if we can reach the node before expiration
  324. expire := time.Now().Add(time.Second)
  325. fs.Create("/foo/foo", "bar", expire, 1, 1)
  326. b, err := fs.Save()
  327. cloneFs := New()
  328. time.Sleep(time.Second)
  329. cloneFs.Recovery(b)
  330. for i, k := range keys {
  331. _, err := cloneFs.Get(k, false, false, uint64(i), 1)
  332. if err != nil {
  333. panic(err)
  334. }
  335. }
  336. if fs.WatcherHub.EventHistory.StartIndex != cloneFs.WatcherHub.EventHistory.StartIndex {
  337. t.Fatal("Error recovered event history start index")
  338. }
  339. for i = 0; int(i) < fs.WatcherHub.EventHistory.Queue.Size; i++ {
  340. if fs.WatcherHub.EventHistory.Queue.Events[i].Key !=
  341. cloneFs.WatcherHub.EventHistory.Queue.Events[i].Key {
  342. t.Fatal("Error recovered event history")
  343. }
  344. }
  345. _, err = fs.Get("/foo/foo", false, false, 1, 1)
  346. if err == nil || err.Error() != "Key Not Found" {
  347. t.Fatalf("can get the node after deletion ")
  348. }
  349. }
  350. // GenKeys randomly generate num of keys with max depth
  351. func GenKeys(num int, depth int) []string {
  352. rand.Seed(time.Now().UnixNano())
  353. keys := make([]string, num)
  354. for i := 0; i < num; i++ {
  355. keys[i] = "/foo"
  356. depth := rand.Intn(depth) + 1
  357. for j := 0; j < depth; j++ {
  358. keys[i] += "/" + strconv.Itoa(rand.Int())
  359. }
  360. }
  361. return keys
  362. }
  363. func createAndGet(fs *FileSystem, path string, t *testing.T) {
  364. _, err := fs.Create(path, "bar", Permanent, 1, 1)
  365. if err != nil {
  366. t.Fatalf("cannot create %s=bar [%s]", path, err.Error())
  367. }
  368. e, err := fs.Get(path, false, false, 1, 1)
  369. if err != nil {
  370. t.Fatalf("cannot get %s [%s]", path, err.Error())
  371. }
  372. if e.Value != "bar" {
  373. t.Fatalf("expect value of %s is bar [%s]", path, e.Value)
  374. }
  375. }
  376. func recursiveTestSort(k KeyValuePair, t *testing.T) {
  377. for i, v := range k.KVPairs[:len(k.KVPairs)-1] {
  378. if v.Key >= k.KVPairs[i+1].Key {
  379. t.Fatalf("sort failed, [%s] should be placed after [%s]", v.Key, k.KVPairs[i+1].Key)
  380. }
  381. if v.Dir {
  382. recursiveTestSort(v, t)
  383. }
  384. }
  385. if v := k.KVPairs[len(k.KVPairs)-1]; v.Dir {
  386. recursiveTestSort(v, t)
  387. }
  388. }
  389. func nonblockingRetrive(c <-chan *Event) *Event {
  390. select {
  391. case e := <-c:
  392. return e
  393. default:
  394. return nil
  395. }
  396. }