agent/cache: make edge case with prev/next idx == 0 handled better

pull/4275/head
Mitchell Hashimoto 2018-04-19 18:35:10 -07:00
parent 3b550d2b72
commit e0d964188c
No known key found for this signature in database
GPG Key ID: 744E147AA52F5B0A
2 changed files with 27 additions and 11 deletions

View File

@ -210,13 +210,7 @@ RETRY_GET:
// Touch the expiration and fix the heap.
c.entriesLock.Lock()
entry.Expiry.Reset()
idx := entry.Expiry.HeapIndex
heap.Fix(c.entriesExpiryHeap, entry.Expiry.HeapIndex)
if idx == 0 && entry.Expiry.HeapIndex == 0 {
// We didn't move and we were at the head of the heap.
// We need to let the loop know that the value changed.
c.entriesExpiryHeap.Notify()
}
c.entriesExpiryHeap.Fix(entry.Expiry)
c.entriesLock.Unlock()
return entry.Value, entry.Error

30
agent/cache/entry.go vendored
View File

@ -1,6 +1,7 @@
package cache
import (
"container/heap"
"time"
)
@ -51,9 +52,34 @@ type expiryHeap struct {
// NotifyCh is sent a value whenever the 0 index value of the heap
// changes. This can be used to detect when the earliest value
// changes.
//
// There is a single edge case where the heap will not automatically
// send a notification: if heap.Fix is called manually and the index
// changed is 0 and the change doesn't result in any moves (stays at index
// 0), then we won't detect the change. To work around this, please
// always call the expiryHeap.Fix method instead.
NotifyCh chan struct{}
}
// Identical to heap.Fix for this heap instance but will properly handle
// the edge case where idx == 0 and no heap modification is necessary,
// and still notify the NotifyCh.
//
// This is important for cache expiry since the expiry time may have been
// extended and if we don't send a message to the NotifyCh then we'll never
// reset the timer and the entry will be evicted early.
func (h *expiryHeap) Fix(entry *cacheEntryExpiry) {
idx := entry.HeapIndex
heap.Fix(h, idx)
// This is the edge case we handle: if the prev and current index
// is zero, it means the head-of-line didn't change while the value
// changed. Notify to reset our expiry worker.
if idx == 0 && entry.HeapIndex == 0 {
h.NotifyCh <- struct{}{}
}
}
func (h *expiryHeap) Len() int { return len(h.Entries) }
func (h *expiryHeap) Swap(i, j int) {
@ -97,7 +123,3 @@ func (h *expiryHeap) Pop() interface{} {
h.Entries = old[0 : n-1]
return x
}
func (h *expiryHeap) Notify() {
h.NotifyCh <- struct{}{}
}