Complete atomicity for Add and Replace
This commit is contained in:
parent
88b75b4791
commit
e6e2a3d192
32
cache.go
32
cache.go
|
@ -113,6 +113,10 @@ func (c *cache) Set(k string, x interface{}, d time.Duration) {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
|
c.set(k, x, d)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cache) set(k string, x interface{}, d time.Duration) {
|
||||||
var e *time.Time
|
var e *time.Time
|
||||||
if d == 0 {
|
if d == 0 {
|
||||||
d = c.DefaultExpiration
|
d = c.DefaultExpiration
|
||||||
|
@ -127,27 +131,31 @@ func (c *cache) Set(k string, x interface{}, d time.Duration) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Add and Replace aren't completely atomic
|
|
||||||
|
|
||||||
// Adds an item to the cache only if an item doesn't already exist for the given key,
|
// Adds an item to the cache only if an item doesn't already exist for the given key,
|
||||||
// or if the existing item has expired. Returns an error if not.
|
// or if the existing item has expired. Returns an error if not.
|
||||||
func (c *cache) Add(k string, x interface{}, d time.Duration) error {
|
func (c *cache) Add(k string, x interface{}, d time.Duration) error {
|
||||||
_, found := c.Get(k)
|
c.mu.Lock()
|
||||||
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
|
_, found := c.get(k)
|
||||||
if found {
|
if found {
|
||||||
return fmt.Errorf("Item %s already exists", k)
|
return fmt.Errorf("Item %s already exists", k)
|
||||||
}
|
}
|
||||||
c.Set(k, x, d)
|
c.set(k, x, d)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sets a new value for the cache item only if it already exists. Returns an error if
|
// Sets a new value for the cache item only if it already exists. Returns an error if
|
||||||
// it does not.
|
// it does not.
|
||||||
func (c *cache) Replace(k string, x interface{}, d time.Duration) error {
|
func (c *cache) Replace(k string, x interface{}, d time.Duration) error {
|
||||||
_, found := c.Get(k)
|
c.mu.Lock()
|
||||||
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
|
_, found := c.get(k)
|
||||||
if !found {
|
if !found {
|
||||||
return fmt.Errorf("Item %s doesn't exist", k)
|
return fmt.Errorf("Item %s doesn't exist", k)
|
||||||
}
|
}
|
||||||
c.Set(k, x, d)
|
c.set(k, x, d)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -157,12 +165,16 @@ func (c *cache) Get(k string) (interface{}, bool) {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
|
return c.get(k)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cache) get(k string) (interface{}, bool) {
|
||||||
item, found := c.Items[k]
|
item, found := c.Items[k]
|
||||||
if !found {
|
if !found {
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
if item.Expired() {
|
if item.Expired() {
|
||||||
delete(c.Items, k)
|
c.delete(k)
|
||||||
return nil, false
|
return nil, false
|
||||||
}
|
}
|
||||||
return item.Object, true
|
return item.Object, true
|
||||||
|
@ -236,6 +248,10 @@ func (c *cache) Delete(k string) {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
|
c.delete(k)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *cache) delete(k string) {
|
||||||
delete(c.Items, k)
|
delete(c.Items, k)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -246,7 +262,7 @@ func (c *cache) DeleteExpired() {
|
||||||
|
|
||||||
for k, v := range c.Items {
|
for k, v := range c.Items {
|
||||||
if v.Expired() {
|
if v.Expired() {
|
||||||
delete(c.Items, k)
|
c.delete(k)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue