OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 package memory | |
6 | |
7 import ( | |
8 "infra/gae/libs/wrapper" | |
9 "infra/gae/libs/wrapper/gae/commonErrors" | |
10 "infra/gae/libs/wrapper/unsafe" | |
11 "sync" | |
12 "time" | |
13 | |
14 "golang.org/x/net/context" | |
15 | |
16 "appengine/memcache" | |
17 ) | |
18 | |
19 type memcacheData struct { | |
20 lock sync.Mutex | |
21 items map[string]*unsafe.Item | |
22 casID uint64 | |
23 } | |
24 | |
25 // memcacheImpl binds the current connection's memcache data to an | |
26 // implementation of {wrapper.Memcache, wrapper.Testable}. | |
M-A Ruel
2015/05/29 18:16:53
That's a great example that helps understanding th
| |
27 type memcacheImpl struct { | |
28 wrapper.Memcache | |
29 wrapper.BrokenFeatures | |
30 | |
31 // TODO(riannucci): bind+use namespace too | |
32 | |
33 data *memcacheData | |
34 timeNow func() time.Time | |
35 } | |
36 | |
37 // UseMC adds a wrapper.Memcache implementation to context, accessible | |
38 // by wrapper.GetMC(c) | |
39 func UseMC(c context.Context) context.Context { | |
40 lck := sync.Mutex{} | |
41 mcdMap := map[string]*memcacheData{} | |
42 | |
43 return wrapper.SetMCFactory(c, func(ic context.Context) wrapper.Memcache { | |
44 lck.Lock() | |
45 defer lck.Unlock() | |
46 | |
47 ns := curGID(ic).namespace | |
48 mcd, ok := mcdMap[ns] | |
49 if !ok { | |
50 mcd = &memcacheData{items: map[string]*unsafe.Item{}} | |
51 mcdMap[ns] = mcd | |
52 } | |
53 | |
54 return &memcacheImpl{ | |
55 wrapper.DummyMC(), | |
56 wrapper.BrokenFeatures{DefaultError: commonErrors.ErrSer verErrorMC}, | |
57 mcd, | |
58 func() time.Time { return wrapper.GetTimeNow(ic) }, | |
59 } | |
60 }) | |
61 } | |
62 | |
63 func (m *memcacheImpl) mkItemLocked(i *memcache.Item) *unsafe.Item { | |
64 m.data.casID++ | |
65 var exp time.Duration | |
66 if i.Expiration != 0 { | |
67 exp = time.Duration(m.timeNow().Add(i.Expiration).UnixNano()) | |
68 } | |
69 newItem := unsafe.Item{ | |
70 Key: i.Key, | |
71 Value: make([]byte, len(i.Value)), | |
72 Flags: i.Flags, | |
73 Expiration: exp, | |
74 CasID: m.data.casID, | |
75 } | |
76 copy(newItem.Value, i.Value) | |
77 return &newItem | |
78 } | |
79 | |
80 func copyBack(i *unsafe.Item) *memcache.Item { | |
81 ret := &memcache.Item{ | |
82 Key: i.Key, | |
83 Value: make([]byte, len(i.Value)), | |
84 Flags: i.Flags, | |
85 } | |
86 copy(ret.Value, i.Value) | |
87 unsafe.MCSetCasID(ret, i.CasID) | |
88 | |
89 return ret | |
90 } | |
91 | |
92 func (m *memcacheImpl) retrieve(key string) (*unsafe.Item, bool) { | |
93 ret, ok := m.data.items[key] | |
94 if ok && ret.Expiration != 0 && ret.Expiration < time.Duration(m.timeNow ().UnixNano()) { | |
95 ret = nil | |
96 ok = false | |
97 delete(m.data.items, key) | |
98 } | |
99 return ret, ok | |
100 } | |
101 | |
102 // Add implements context.MCSingleReadWriter.Add. | |
103 func (m *memcacheImpl) Add(i *memcache.Item) error { | |
104 if err := m.IsBroken(); err != nil { | |
105 return err | |
106 } | |
107 | |
108 m.data.lock.Lock() | |
109 defer m.data.lock.Unlock() | |
110 | |
111 if _, ok := m.retrieve(i.Key); !ok { | |
112 m.data.items[i.Key] = m.mkItemLocked(i) | |
113 return nil | |
114 } | |
115 return memcache.ErrNotStored | |
116 } | |
117 | |
118 // CompareAndSwap implements context.MCSingleReadWriter.CompareAndSwap. | |
119 func (m *memcacheImpl) CompareAndSwap(item *memcache.Item) error { | |
120 if err := m.IsBroken(); err != nil { | |
121 return err | |
122 } | |
123 | |
124 m.data.lock.Lock() | |
125 defer m.data.lock.Unlock() | |
126 | |
127 if cur, ok := m.retrieve(item.Key); ok { | |
128 if cur.CasID == unsafe.MCGetCasID(item) { | |
129 m.data.items[item.Key] = m.mkItemLocked(item) | |
130 } else { | |
131 return memcache.ErrCASConflict | |
132 } | |
133 } else { | |
134 return memcache.ErrNotStored | |
135 } | |
136 return nil | |
137 } | |
138 | |
139 // Set implements context.MCSingleReadWriter.Set. | |
140 func (m *memcacheImpl) Set(i *memcache.Item) error { | |
141 if err := m.IsBroken(); err != nil { | |
142 return err | |
143 } | |
144 | |
145 m.data.lock.Lock() | |
146 defer m.data.lock.Unlock() | |
147 | |
148 m.data.items[i.Key] = m.mkItemLocked(i) | |
149 return nil | |
150 } | |
151 | |
152 // Get implements context.MCSingleReadWriter.Get. | |
153 func (m *memcacheImpl) Get(key string) (*memcache.Item, error) { | |
154 if err := m.IsBroken(); err != nil { | |
155 return nil, err | |
156 } | |
157 | |
158 m.data.lock.Lock() | |
159 defer m.data.lock.Unlock() | |
160 | |
161 if val, ok := m.retrieve(key); ok { | |
162 return copyBack(val), nil | |
163 } | |
164 return nil, memcache.ErrCacheMiss | |
165 } | |
166 | |
167 // Delete implements context.MCSingleReadWriter.Delete. | |
168 func (m *memcacheImpl) Delete(key string) error { | |
169 if err := m.IsBroken(); err != nil { | |
170 return err | |
171 } | |
172 | |
173 m.data.lock.Lock() | |
174 defer m.data.lock.Unlock() | |
175 | |
176 if _, ok := m.retrieve(key); ok { | |
177 delete(m.data.items, key) | |
178 return nil | |
179 } | |
180 return memcache.ErrCacheMiss | |
181 } | |
OLD | NEW |