memcache_test.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /*
  2. Copyright 2011 Google Inc.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. // Package memcache provides a client for the memcached cache server.
  14. package memcache
  15. import (
  16. "fmt"
  17. "net"
  18. "os"
  19. "os/exec"
  20. "strings"
  21. "testing"
  22. "time"
  23. )
  24. const testServer = "localhost:11211"
  25. func setup(t *testing.T) bool {
  26. c, err := net.Dial("tcp", testServer)
  27. if err != nil {
  28. t.Skipf("skipping test; no server running at %s", testServer)
  29. }
  30. c.Write([]byte("flush_all\r\n"))
  31. c.Close()
  32. return true
  33. }
  34. func TestLocalhost(t *testing.T) {
  35. if !setup(t) {
  36. return
  37. }
  38. testWithClient(t, New(testServer))
  39. }
  40. // Run the memcached binary as a child process and connect to its unix socket.
  41. func TestUnixSocket(t *testing.T) {
  42. sock := fmt.Sprintf("/tmp/test-gomemcache-%d.sock", os.Getpid())
  43. cmd := exec.Command("memcached", "-s", sock)
  44. if err := cmd.Start(); err != nil {
  45. t.Skipf("skipping test; couldn't find memcached")
  46. return
  47. }
  48. defer cmd.Wait()
  49. defer cmd.Process.Kill()
  50. // Wait a bit for the socket to appear.
  51. for i := 0; i < 10; i++ {
  52. if _, err := os.Stat(sock); err == nil {
  53. break
  54. }
  55. time.Sleep(time.Duration(25*i) * time.Millisecond)
  56. }
  57. testWithClient(t, New(sock))
  58. }
  59. func mustSetF(t *testing.T, c *Client) func(*Item) {
  60. return func(it *Item) {
  61. if err := c.Set(it); err != nil {
  62. t.Fatalf("failed to Set %#v: %v", *it, err)
  63. }
  64. }
  65. }
  66. func testWithClient(t *testing.T, c *Client) {
  67. checkErr := func(err error, format string, args ...interface{}) {
  68. if err != nil {
  69. t.Fatalf(format, args...)
  70. }
  71. }
  72. mustSet := mustSetF(t, c)
  73. // Set
  74. foo := &Item{Key: "foo", Value: []byte("fooval"), Flags: 123}
  75. err := c.Set(foo)
  76. checkErr(err, "first set(foo): %v", err)
  77. err = c.Set(foo)
  78. checkErr(err, "second set(foo): %v", err)
  79. // Get
  80. it, err := c.Get("foo")
  81. checkErr(err, "get(foo): %v", err)
  82. if it.Key != "foo" {
  83. t.Errorf("get(foo) Key = %q, want foo", it.Key)
  84. }
  85. if string(it.Value) != "fooval" {
  86. t.Errorf("get(foo) Value = %q, want fooval", string(it.Value))
  87. }
  88. if it.Flags != 123 {
  89. t.Errorf("get(foo) Flags = %v, want 123", it.Flags)
  90. }
  91. // Add
  92. bar := &Item{Key: "bar", Value: []byte("barval")}
  93. err = c.Add(bar)
  94. checkErr(err, "first add(foo): %v", err)
  95. if err := c.Add(bar); err != ErrNotStored {
  96. t.Fatalf("second add(foo) want ErrNotStored, got %v", err)
  97. }
  98. // GetMulti
  99. m, err := c.GetMulti([]string{"foo", "bar"})
  100. checkErr(err, "GetMulti: %v", err)
  101. if g, e := len(m), 2; g != e {
  102. t.Errorf("GetMulti: got len(map) = %d, want = %d", g, e)
  103. }
  104. if _, ok := m["foo"]; !ok {
  105. t.Fatalf("GetMulti: didn't get key 'foo'")
  106. }
  107. if _, ok := m["bar"]; !ok {
  108. t.Fatalf("GetMulti: didn't get key 'bar'")
  109. }
  110. if g, e := string(m["foo"].Value), "fooval"; g != e {
  111. t.Errorf("GetMulti: foo: got %q, want %q", g, e)
  112. }
  113. if g, e := string(m["bar"].Value), "barval"; g != e {
  114. t.Errorf("GetMulti: bar: got %q, want %q", g, e)
  115. }
  116. // Delete
  117. err = c.Delete("foo")
  118. checkErr(err, "Delete: %v", err)
  119. it, err = c.Get("foo")
  120. if err != ErrCacheMiss {
  121. t.Errorf("post-Delete want ErrCacheMiss, got %v", err)
  122. }
  123. // Incr/Decr
  124. mustSet(&Item{Key: "num", Value: []byte("42")})
  125. n, err := c.Increment("num", 8)
  126. checkErr(err, "Increment num + 8: %v", err)
  127. if n != 50 {
  128. t.Fatalf("Increment num + 8: want=50, got=%d", n)
  129. }
  130. n, err = c.Decrement("num", 49)
  131. checkErr(err, "Decrement: %v", err)
  132. if n != 1 {
  133. t.Fatalf("Decrement 49: want=1, got=%d", n)
  134. }
  135. err = c.Delete("num")
  136. checkErr(err, "delete num: %v", err)
  137. n, err = c.Increment("num", 1)
  138. if err != ErrCacheMiss {
  139. t.Fatalf("increment post-delete: want ErrCacheMiss, got %v", err)
  140. }
  141. mustSet(&Item{Key: "num", Value: []byte("not-numeric")})
  142. n, err = c.Increment("num", 1)
  143. if err == nil || !strings.Contains(err.Error(), "client error") {
  144. t.Fatalf("increment non-number: want client error, got %v", err)
  145. }
  146. testTouchWithClient(t, c)
  147. // Test Delete All
  148. err = c.DeleteAll()
  149. checkErr(err, "DeleteAll: %v", err)
  150. it, err = c.Get("bar")
  151. if err != ErrCacheMiss {
  152. t.Errorf("post-DeleteAll want ErrCacheMiss, got %v", err)
  153. }
  154. }
  155. func testTouchWithClient(t *testing.T, c *Client) {
  156. if testing.Short() {
  157. t.Log("Skipping testing memcache Touch with testing in Short mode")
  158. return
  159. }
  160. mustSet := mustSetF(t, c)
  161. const secondsToExpiry = int32(2)
  162. // We will set foo and bar to expire in 2 seconds, then we'll keep touching
  163. // foo every second
  164. // After 3 seconds, we expect foo to be available, and bar to be expired
  165. foo := &Item{Key: "foo", Value: []byte("fooval"), Expiration: secondsToExpiry}
  166. bar := &Item{Key: "bar", Value: []byte("barval"), Expiration: secondsToExpiry}
  167. setTime := time.Now()
  168. mustSet(foo)
  169. mustSet(bar)
  170. for s := 0; s < 3; s++ {
  171. time.Sleep(time.Duration(1 * time.Second))
  172. err := c.Touch(foo.Key, secondsToExpiry)
  173. if nil != err {
  174. t.Errorf("error touching foo: %v", err.Error())
  175. }
  176. }
  177. _, err := c.Get("foo")
  178. if err != nil {
  179. if err == ErrCacheMiss {
  180. t.Fatalf("touching failed to keep item foo alive")
  181. } else {
  182. t.Fatalf("unexpected error retrieving foo after touching: %v", err.Error())
  183. }
  184. }
  185. _, err = c.Get("bar")
  186. if nil == err {
  187. t.Fatalf("item bar did not expire within %v seconds", time.Now().Sub(setTime).Seconds())
  188. } else {
  189. if err != ErrCacheMiss {
  190. t.Fatalf("unexpected error retrieving bar: %v", err.Error())
  191. }
  192. }
  193. }