123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666 |
- package memcache
- import (
- "bufio"
- "bytes"
- "errors"
- "fmt"
- "io"
- "io/ioutil"
- "net"
- "strconv"
- "strings"
- "sync"
- "time"
- )
- var (
-
- ErrCacheMiss = errors.New("memcache: cache miss")
-
-
-
-
- ErrCASConflict = errors.New("memcache: compare-and-swap conflict")
-
-
- ErrNotStored = errors.New("memcache: item not stored")
-
- ErrServerError = errors.New("memcache: server error")
-
- ErrNoStats = errors.New("memcache: no statistics available")
-
-
-
- ErrMalformedKey = errors.New("malformed: key is too long or contains invalid characters")
-
- ErrNoServers = errors.New("memcache: no servers configured or available")
- )
- const DefaultTimeout = 100 * time.Millisecond
- const (
- buffered = 8
- maxIdleConnsPerAddr = 2
- )
- func resumableError(err error) bool {
- switch err {
- case ErrCacheMiss, ErrCASConflict, ErrNotStored, ErrMalformedKey:
- return true
- }
- return false
- }
- func legalKey(key string) bool {
- if len(key) > 250 {
- return false
- }
- for i := 0; i < len(key); i++ {
- if key[i] <= ' ' || key[i] > 0x7e {
- return false
- }
- }
- return true
- }
- var (
- crlf = []byte("\r\n")
- space = []byte(" ")
- resultOK = []byte("OK\r\n")
- resultStored = []byte("STORED\r\n")
- resultNotStored = []byte("NOT_STORED\r\n")
- resultExists = []byte("EXISTS\r\n")
- resultNotFound = []byte("NOT_FOUND\r\n")
- resultDeleted = []byte("DELETED\r\n")
- resultEnd = []byte("END\r\n")
- resultOk = []byte("OK\r\n")
- resultTouched = []byte("TOUCHED\r\n")
- resultClientErrorPrefix = []byte("CLIENT_ERROR ")
- )
- func New(server ...string) *Client {
- ss := new(ServerList)
- ss.SetServers(server...)
- return NewFromSelector(ss)
- }
- func NewFromSelector(ss ServerSelector) *Client {
- return &Client{selector: ss}
- }
- type Client struct {
-
-
- Timeout time.Duration
- selector ServerSelector
- lk sync.Mutex
- freeconn map[string][]*conn
- }
- type Item struct {
-
- Key string
-
- Value []byte
-
-
- Flags uint32
-
-
-
- Expiration int32
-
- casid uint64
- }
- type conn struct {
- nc net.Conn
- rw *bufio.ReadWriter
- addr net.Addr
- c *Client
- }
- func (cn *conn) release() {
- cn.c.putFreeConn(cn.addr, cn)
- }
- func (cn *conn) extendDeadline() {
- cn.nc.SetDeadline(time.Now().Add(cn.c.netTimeout()))
- }
- func (cn *conn) condRelease(err *error) {
- if *err == nil || resumableError(*err) {
- cn.release()
- } else {
- cn.nc.Close()
- }
- }
- func (c *Client) putFreeConn(addr net.Addr, cn *conn) {
- c.lk.Lock()
- defer c.lk.Unlock()
- if c.freeconn == nil {
- c.freeconn = make(map[string][]*conn)
- }
- freelist := c.freeconn[addr.String()]
- if len(freelist) >= maxIdleConnsPerAddr {
- cn.nc.Close()
- return
- }
- c.freeconn[addr.String()] = append(freelist, cn)
- }
- func (c *Client) getFreeConn(addr net.Addr) (cn *conn, ok bool) {
- c.lk.Lock()
- defer c.lk.Unlock()
- if c.freeconn == nil {
- return nil, false
- }
- freelist, ok := c.freeconn[addr.String()]
- if !ok || len(freelist) == 0 {
- return nil, false
- }
- cn = freelist[len(freelist)-1]
- c.freeconn[addr.String()] = freelist[:len(freelist)-1]
- return cn, true
- }
- func (c *Client) netTimeout() time.Duration {
- if c.Timeout != 0 {
- return c.Timeout
- }
- return DefaultTimeout
- }
- type ConnectTimeoutError struct {
- Addr net.Addr
- }
- func (cte *ConnectTimeoutError) Error() string {
- return "memcache: connect timeout to " + cte.Addr.String()
- }
- func (c *Client) dial(addr net.Addr) (net.Conn, error) {
- type connError struct {
- cn net.Conn
- err error
- }
- nc, err := net.DialTimeout(addr.Network(), addr.String(), c.netTimeout())
- if err == nil {
- return nc, nil
- }
- if ne, ok := err.(net.Error); ok && ne.Timeout() {
- return nil, &ConnectTimeoutError{addr}
- }
- return nil, err
- }
- func (c *Client) getConn(addr net.Addr) (*conn, error) {
- cn, ok := c.getFreeConn(addr)
- if ok {
- cn.extendDeadline()
- return cn, nil
- }
- nc, err := c.dial(addr)
- if err != nil {
- return nil, err
- }
- cn = &conn{
- nc: nc,
- addr: addr,
- rw: bufio.NewReadWriter(bufio.NewReader(nc), bufio.NewWriter(nc)),
- c: c,
- }
- cn.extendDeadline()
- return cn, nil
- }
- func (c *Client) onItem(item *Item, fn func(*Client, *bufio.ReadWriter, *Item) error) error {
- addr, err := c.selector.PickServer(item.Key)
- if err != nil {
- return err
- }
- cn, err := c.getConn(addr)
- if err != nil {
- return err
- }
- defer cn.condRelease(&err)
- if err = fn(c, cn.rw, item); err != nil {
- return err
- }
- return nil
- }
- func (c *Client) FlushAll() error {
- return c.selector.Each(c.flushAllFromAddr)
- }
- func (c *Client) Get(key string) (item *Item, err error) {
- err = c.withKeyAddr(key, func(addr net.Addr) error {
- return c.getFromAddr(addr, []string{key}, func(it *Item) { item = it })
- })
- if err == nil && item == nil {
- err = ErrCacheMiss
- }
- return
- }
- func (c *Client) Touch(key string, seconds int32) (err error) {
- return c.withKeyAddr(key, func(addr net.Addr) error {
- return c.touchFromAddr(addr, []string{key}, seconds)
- })
- }
- func (c *Client) withKeyAddr(key string, fn func(net.Addr) error) (err error) {
- if !legalKey(key) {
- return ErrMalformedKey
- }
- addr, err := c.selector.PickServer(key)
- if err != nil {
- return err
- }
- return fn(addr)
- }
- func (c *Client) withAddrRw(addr net.Addr, fn func(*bufio.ReadWriter) error) (err error) {
- cn, err := c.getConn(addr)
- if err != nil {
- return err
- }
- defer cn.condRelease(&err)
- return fn(cn.rw)
- }
- func (c *Client) withKeyRw(key string, fn func(*bufio.ReadWriter) error) error {
- return c.withKeyAddr(key, func(addr net.Addr) error {
- return c.withAddrRw(addr, fn)
- })
- }
- func (c *Client) getFromAddr(addr net.Addr, keys []string, cb func(*Item)) error {
- return c.withAddrRw(addr, func(rw *bufio.ReadWriter) error {
- if _, err := fmt.Fprintf(rw, "gets %s\r\n", strings.Join(keys, " ")); err != nil {
- return err
- }
- if err := rw.Flush(); err != nil {
- return err
- }
- if err := parseGetResponse(rw.Reader, cb); err != nil {
- return err
- }
- return nil
- })
- }
- func (c *Client) flushAllFromAddr(addr net.Addr) error {
- return c.withAddrRw(addr, func(rw *bufio.ReadWriter) error {
- if _, err := fmt.Fprintf(rw, "flush_all\r\n"); err != nil {
- return err
- }
- if err := rw.Flush(); err != nil {
- return err
- }
- line, err := rw.ReadSlice('\n')
- if err != nil {
- return err
- }
- switch {
- case bytes.Equal(line, resultOk):
- break
- default:
- return fmt.Errorf("memcache: unexpected response line from flush_all: %q", string(line))
- }
- return nil
- })
- }
- func (c *Client) touchFromAddr(addr net.Addr, keys []string, expiration int32) error {
- return c.withAddrRw(addr, func(rw *bufio.ReadWriter) error {
- for _, key := range keys {
- if _, err := fmt.Fprintf(rw, "touch %s %d\r\n", key, expiration); err != nil {
- return err
- }
- if err := rw.Flush(); err != nil {
- return err
- }
- line, err := rw.ReadSlice('\n')
- if err != nil {
- return err
- }
- switch {
- case bytes.Equal(line, resultTouched):
- break
- case bytes.Equal(line, resultNotFound):
- return ErrCacheMiss
- default:
- return fmt.Errorf("memcache: unexpected response line from touch: %q", string(line))
- }
- }
- return nil
- })
- }
- func (c *Client) GetMulti(keys []string) (map[string]*Item, error) {
- var lk sync.Mutex
- m := make(map[string]*Item)
- addItemToMap := func(it *Item) {
- lk.Lock()
- defer lk.Unlock()
- m[it.Key] = it
- }
- keyMap := make(map[net.Addr][]string)
- for _, key := range keys {
- if !legalKey(key) {
- return nil, ErrMalformedKey
- }
- addr, err := c.selector.PickServer(key)
- if err != nil {
- return nil, err
- }
- keyMap[addr] = append(keyMap[addr], key)
- }
- ch := make(chan error, buffered)
- for addr, keys := range keyMap {
- go func(addr net.Addr, keys []string) {
- ch <- c.getFromAddr(addr, keys, addItemToMap)
- }(addr, keys)
- }
- var err error
- for _ = range keyMap {
- if ge := <-ch; ge != nil {
- err = ge
- }
- }
- return m, err
- }
- func parseGetResponse(r *bufio.Reader, cb func(*Item)) error {
- for {
- line, err := r.ReadSlice('\n')
- if err != nil {
- return err
- }
- if bytes.Equal(line, resultEnd) {
- return nil
- }
- it := new(Item)
- size, err := scanGetResponseLine(line, it)
- if err != nil {
- return err
- }
- it.Value, err = ioutil.ReadAll(io.LimitReader(r, int64(size)+2))
- if err != nil {
- return err
- }
- if !bytes.HasSuffix(it.Value, crlf) {
- return fmt.Errorf("memcache: corrupt get result read")
- }
- it.Value = it.Value[:size]
- cb(it)
- }
- }
- func scanGetResponseLine(line []byte, it *Item) (size int, err error) {
- pattern := "VALUE %s %d %d %d\r\n"
- dest := []interface{}{&it.Key, &it.Flags, &size, &it.casid}
- if bytes.Count(line, space) == 3 {
- pattern = "VALUE %s %d %d\r\n"
- dest = dest[:3]
- }
- n, err := fmt.Sscanf(string(line), pattern, dest...)
- if err != nil || n != len(dest) {
- return -1, fmt.Errorf("memcache: unexpected line in get response: %q", line)
- }
- return size, nil
- }
- func (c *Client) Set(item *Item) error {
- return c.onItem(item, (*Client).set)
- }
- func (c *Client) set(rw *bufio.ReadWriter, item *Item) error {
- return c.populateOne(rw, "set", item)
- }
- func (c *Client) Add(item *Item) error {
- return c.onItem(item, (*Client).add)
- }
- func (c *Client) add(rw *bufio.ReadWriter, item *Item) error {
- return c.populateOne(rw, "add", item)
- }
- func (c *Client) Replace(item *Item) error {
- return c.onItem(item, (*Client).replace)
- }
- func (c *Client) replace(rw *bufio.ReadWriter, item *Item) error {
- return c.populateOne(rw, "replace", item)
- }
- func (c *Client) CompareAndSwap(item *Item) error {
- return c.onItem(item, (*Client).cas)
- }
- func (c *Client) cas(rw *bufio.ReadWriter, item *Item) error {
- return c.populateOne(rw, "cas", item)
- }
- func (c *Client) populateOne(rw *bufio.ReadWriter, verb string, item *Item) error {
- if !legalKey(item.Key) {
- return ErrMalformedKey
- }
- var err error
- if verb == "cas" {
- _, err = fmt.Fprintf(rw, "%s %s %d %d %d %d\r\n",
- verb, item.Key, item.Flags, item.Expiration, len(item.Value), item.casid)
- } else {
- _, err = fmt.Fprintf(rw, "%s %s %d %d %d\r\n",
- verb, item.Key, item.Flags, item.Expiration, len(item.Value))
- }
- if err != nil {
- return err
- }
- if _, err = rw.Write(item.Value); err != nil {
- return err
- }
- if _, err := rw.Write(crlf); err != nil {
- return err
- }
- if err := rw.Flush(); err != nil {
- return err
- }
- line, err := rw.ReadSlice('\n')
- if err != nil {
- return err
- }
- switch {
- case bytes.Equal(line, resultStored):
- return nil
- case bytes.Equal(line, resultNotStored):
- return ErrNotStored
- case bytes.Equal(line, resultExists):
- return ErrCASConflict
- case bytes.Equal(line, resultNotFound):
- return ErrCacheMiss
- }
- return fmt.Errorf("memcache: unexpected response line from %q: %q", verb, string(line))
- }
- func writeReadLine(rw *bufio.ReadWriter, format string, args ...interface{}) ([]byte, error) {
- _, err := fmt.Fprintf(rw, format, args...)
- if err != nil {
- return nil, err
- }
- if err := rw.Flush(); err != nil {
- return nil, err
- }
- line, err := rw.ReadSlice('\n')
- return line, err
- }
- func writeExpectf(rw *bufio.ReadWriter, expect []byte, format string, args ...interface{}) error {
- line, err := writeReadLine(rw, format, args...)
- if err != nil {
- return err
- }
- switch {
- case bytes.Equal(line, resultOK):
- return nil
- case bytes.Equal(line, expect):
- return nil
- case bytes.Equal(line, resultNotStored):
- return ErrNotStored
- case bytes.Equal(line, resultExists):
- return ErrCASConflict
- case bytes.Equal(line, resultNotFound):
- return ErrCacheMiss
- }
- return fmt.Errorf("memcache: unexpected response line: %q", string(line))
- }
- func (c *Client) Delete(key string) error {
- return c.withKeyRw(key, func(rw *bufio.ReadWriter) error {
- return writeExpectf(rw, resultDeleted, "delete %s\r\n", key)
- })
- }
- func (c *Client) DeleteAll() error {
- return c.withKeyRw("", func(rw *bufio.ReadWriter) error {
- return writeExpectf(rw, resultDeleted, "flush_all\r\n")
- })
- }
- func (c *Client) Increment(key string, delta uint64) (newValue uint64, err error) {
- return c.incrDecr("incr", key, delta)
- }
- func (c *Client) Decrement(key string, delta uint64) (newValue uint64, err error) {
- return c.incrDecr("decr", key, delta)
- }
- func (c *Client) incrDecr(verb, key string, delta uint64) (uint64, error) {
- var val uint64
- err := c.withKeyRw(key, func(rw *bufio.ReadWriter) error {
- line, err := writeReadLine(rw, "%s %s %d\r\n", verb, key, delta)
- if err != nil {
- return err
- }
- switch {
- case bytes.Equal(line, resultNotFound):
- return ErrCacheMiss
- case bytes.HasPrefix(line, resultClientErrorPrefix):
- errMsg := line[len(resultClientErrorPrefix) : len(line)-2]
- return errors.New("memcache: client error: " + string(errMsg))
- }
- val, err = strconv.ParseUint(string(line[:len(line)-2]), 10, 64)
- if err != nil {
- return err
- }
- return nil
- })
- return val, err
- }
|