Browse Source

more logs

pull/8292/head
Chris Lu 13 hours ago
parent
commit
eded9a29b5
  1. 30
      weed/filer/empty_folder_cleanup/cleanup_queue.go
  2. 120
      weed/filer/empty_folder_cleanup/cleanup_queue_test.go
  3. 24
      weed/filer/empty_folder_cleanup/empty_folder_cleaner.go
  4. 12
      weed/filer/empty_folder_cleanup/empty_folder_cleaner_test.go
  5. 2
      weed/s3api/s3api_object_handlers_delete.go

30
weed/filer/empty_folder_cleanup/cleanup_queue.go

@ -21,8 +21,9 @@ type CleanupQueue struct {
// queueItem represents an item in the cleanup queue // queueItem represents an item in the cleanup queue
type queueItem struct { type queueItem struct {
folder string
queueTime time.Time
folder string
triggeredBy string
queueTime time.Time
} }
// NewCleanupQueue creates a new CleanupQueue with the specified limits // NewCleanupQueue creates a new CleanupQueue with the specified limits
@ -39,7 +40,7 @@ func NewCleanupQueue(maxSize int, maxAge time.Duration) *CleanupQueue {
// The item is inserted in time-sorted order (oldest at front) to handle out-of-order events. // The item is inserted in time-sorted order (oldest at front) to handle out-of-order events.
// If folder already exists with an older time, the time is updated and position adjusted. // If folder already exists with an older time, the time is updated and position adjusted.
// Returns true if the folder was newly added, false if it was updated. // Returns true if the folder was newly added, false if it was updated.
func (q *CleanupQueue) Add(folder string, eventTime time.Time) bool {
func (q *CleanupQueue) Add(folder string, triggeredBy string, eventTime time.Time) bool {
q.mu.Lock() q.mu.Lock()
defer q.mu.Unlock() defer q.mu.Unlock()
@ -51,23 +52,24 @@ func (q *CleanupQueue) Add(folder string, eventTime time.Time) bool {
// Remove from current position // Remove from current position
q.items.Remove(elem) q.items.Remove(elem)
// Re-insert with new time in sorted position // Re-insert with new time in sorted position
newElem := q.insertSorted(folder, eventTime)
newElem := q.insertSorted(folder, triggeredBy, eventTime)
q.itemsMap[folder] = newElem q.itemsMap[folder] = newElem
} }
return false return false
} }
// Insert new folder in sorted position // Insert new folder in sorted position
elem := q.insertSorted(folder, eventTime)
elem := q.insertSorted(folder, triggeredBy, eventTime)
q.itemsMap[folder] = elem q.itemsMap[folder] = elem
return true return true
} }
// insertSorted inserts an item in the correct position to maintain time ordering (oldest at front) // insertSorted inserts an item in the correct position to maintain time ordering (oldest at front)
func (q *CleanupQueue) insertSorted(folder string, eventTime time.Time) *list.Element {
func (q *CleanupQueue) insertSorted(folder string, triggeredBy string, eventTime time.Time) *list.Element {
item := &queueItem{ item := &queueItem{
folder: folder,
queueTime: eventTime,
folder: folder,
triggeredBy: triggeredBy,
queueTime: eventTime,
} }
// Find the correct position (insert before the first item with a later time) // Find the correct position (insert before the first item with a later time)
@ -135,35 +137,35 @@ func (q *CleanupQueue) shouldProcessLocked() bool {
// Pop removes and returns the oldest folder from the queue. // Pop removes and returns the oldest folder from the queue.
// Returns the folder and true if an item was available, or empty string and false if queue is empty. // Returns the folder and true if an item was available, or empty string and false if queue is empty.
func (q *CleanupQueue) Pop() (string, bool) {
func (q *CleanupQueue) Pop() (string, string, bool) {
q.mu.Lock() q.mu.Lock()
defer q.mu.Unlock() defer q.mu.Unlock()
front := q.items.Front() front := q.items.Front()
if front == nil { if front == nil {
return "", false
return "", "", false
} }
item := front.Value.(*queueItem) item := front.Value.(*queueItem)
q.items.Remove(front) q.items.Remove(front)
delete(q.itemsMap, item.folder) delete(q.itemsMap, item.folder)
return item.folder, true
return item.folder, item.triggeredBy, true
} }
// Peek returns the oldest folder without removing it. // Peek returns the oldest folder without removing it.
// Returns the folder and queue time if available, or empty values if queue is empty. // Returns the folder and queue time if available, or empty values if queue is empty.
func (q *CleanupQueue) Peek() (folder string, queueTime time.Time, ok bool) {
func (q *CleanupQueue) Peek() (folder string, triggeredBy string, queueTime time.Time, ok bool) {
q.mu.Lock() q.mu.Lock()
defer q.mu.Unlock() defer q.mu.Unlock()
front := q.items.Front() front := q.items.Front()
if front == nil { if front == nil {
return "", time.Time{}, false
return "", "", time.Time{}, false
} }
item := front.Value.(*queueItem) item := front.Value.(*queueItem)
return item.folder, item.queueTime, true
return item.folder, item.triggeredBy, item.queueTime, true
} }
// Len returns the current queue size. // Len returns the current queue size.

120
weed/filer/empty_folder_cleanup/cleanup_queue_test.go

@ -10,7 +10,7 @@ func TestCleanupQueue_Add(t *testing.T) {
now := time.Now() now := time.Now()
// Add first item // Add first item
if !q.Add("/buckets/b1/folder1", now) {
if !q.Add("/buckets/b1/folder1", "item1", now) {
t.Error("expected Add to return true for new item") t.Error("expected Add to return true for new item")
} }
if q.Len() != 1 { if q.Len() != 1 {
@ -18,7 +18,7 @@ func TestCleanupQueue_Add(t *testing.T) {
} }
// Add second item with later time // Add second item with later time
if !q.Add("/buckets/b1/folder2", now.Add(1*time.Second)) {
if !q.Add("/buckets/b1/folder2", "item2", now.Add(1*time.Second)) {
t.Error("expected Add to return true for new item") t.Error("expected Add to return true for new item")
} }
if q.Len() != 2 { if q.Len() != 2 {
@ -26,7 +26,7 @@ func TestCleanupQueue_Add(t *testing.T) {
} }
// Add duplicate with newer time - should update and reposition // Add duplicate with newer time - should update and reposition
if q.Add("/buckets/b1/folder1", now.Add(2*time.Second)) {
if q.Add("/buckets/b1/folder1", "item1-updated", now.Add(2*time.Second)) {
t.Error("expected Add to return false for existing item") t.Error("expected Add to return false for existing item")
} }
if q.Len() != 2 { if q.Len() != 2 {
@ -34,11 +34,14 @@ func TestCleanupQueue_Add(t *testing.T) {
} }
// folder1 should now be at the back (newer time) - verify by popping // folder1 should now be at the back (newer time) - verify by popping
folder1, _ := q.Pop()
folder2, _ := q.Pop()
folder1, triggered1, _ := q.Pop()
folder2, triggered2, _ := q.Pop()
if folder1 != "/buckets/b1/folder2" || folder2 != "/buckets/b1/folder1" { if folder1 != "/buckets/b1/folder2" || folder2 != "/buckets/b1/folder1" {
t.Errorf("expected folder1 to be moved to back, got %s, %s", folder1, folder2) t.Errorf("expected folder1 to be moved to back, got %s, %s", folder1, folder2)
} }
if triggered1 != "item2" || triggered2 != "item1-updated" {
t.Errorf("triggeredBy mismatch: got %s, %s", triggered1, triggered2)
}
} }
func TestCleanupQueue_Add_OutOfOrder(t *testing.T) { func TestCleanupQueue_Add_OutOfOrder(t *testing.T) {
@ -46,16 +49,23 @@ func TestCleanupQueue_Add_OutOfOrder(t *testing.T) {
baseTime := time.Now() baseTime := time.Now()
// Add items out of order // Add items out of order
q.Add("/buckets/b1/folder3", baseTime.Add(3*time.Second))
q.Add("/buckets/b1/folder1", baseTime.Add(1*time.Second))
q.Add("/buckets/b1/folder2", baseTime.Add(2*time.Second))
q.Add("/buckets/b1/folder3", "f3", baseTime.Add(3*time.Second))
q.Add("/buckets/b1/folder1", "f1", baseTime.Add(1*time.Second))
q.Add("/buckets/b1/folder2", "f2", baseTime.Add(2*time.Second))
// Items should be in time order (oldest first) - verify by popping // Items should be in time order (oldest first) - verify by popping
expected := []string{"/buckets/b1/folder1", "/buckets/b1/folder2", "/buckets/b1/folder3"}
expected := []struct {
folder string
item string
}{
{"/buckets/b1/folder1", "f1"},
{"/buckets/b1/folder2", "f2"},
{"/buckets/b1/folder3", "f3"},
}
for i, exp := range expected { for i, exp := range expected {
folder, ok := q.Pop()
if !ok || folder != exp {
t.Errorf("at index %d: expected %s, got %s", i, exp, folder)
folder, item, ok := q.Pop()
if !ok || folder != exp.folder || item != exp.item {
t.Errorf("at index %d: expected %s by %s, got %s by %s", i, exp.folder, exp.item, folder, item)
} }
} }
} }
@ -65,15 +75,15 @@ func TestCleanupQueue_Add_DuplicateWithOlderTime(t *testing.T) {
baseTime := time.Now() baseTime := time.Now()
// Add folder at t=5 // Add folder at t=5
q.Add("/buckets/b1/folder1", baseTime.Add(5*time.Second))
q.Add("/buckets/b1/folder1", "t5", baseTime.Add(5*time.Second))
// Try to add same folder with older time - should NOT update // Try to add same folder with older time - should NOT update
q.Add("/buckets/b1/folder1", baseTime.Add(2*time.Second))
q.Add("/buckets/b1/folder1", "t2", baseTime.Add(2*time.Second))
// Time should remain at t=5
_, queueTime, _ := q.Peek()
if queueTime != baseTime.Add(5*time.Second) {
t.Errorf("expected time to remain unchanged, got %v", queueTime)
// Time and triggeredBy should remain at t=5
_, triggered, queueTime, _ := q.Peek()
if queueTime != baseTime.Add(5*time.Second) || triggered != "t5" {
t.Errorf("expected time to remain unchanged, got %v and %s", queueTime, triggered)
} }
} }
@ -81,9 +91,9 @@ func TestCleanupQueue_Remove(t *testing.T) {
q := NewCleanupQueue(100, 10*time.Minute) q := NewCleanupQueue(100, 10*time.Minute)
now := time.Now() now := time.Now()
q.Add("/buckets/b1/folder1", now)
q.Add("/buckets/b1/folder2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder3", now.Add(2*time.Second))
q.Add("/buckets/b1/folder1", "i1", now)
q.Add("/buckets/b1/folder2", "i2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder3", "i3", now.Add(2*time.Second))
// Remove middle item // Remove middle item
if !q.Remove("/buckets/b1/folder2") { if !q.Remove("/buckets/b1/folder2") {
@ -102,8 +112,8 @@ func TestCleanupQueue_Remove(t *testing.T) {
} }
// Verify order is preserved by popping // Verify order is preserved by popping
folder1, _ := q.Pop()
folder3, _ := q.Pop()
folder1, _, _ := q.Pop()
folder3, _, _ := q.Pop()
if folder1 != "/buckets/b1/folder1" || folder3 != "/buckets/b1/folder3" { if folder1 != "/buckets/b1/folder1" || folder3 != "/buckets/b1/folder3" {
t.Errorf("unexpected order: %s, %s", folder1, folder3) t.Errorf("unexpected order: %s, %s", folder1, folder3)
} }
@ -114,30 +124,30 @@ func TestCleanupQueue_Pop(t *testing.T) {
now := time.Now() now := time.Now()
// Pop from empty queue // Pop from empty queue
folder, ok := q.Pop()
folder, item, ok := q.Pop()
if ok { if ok {
t.Error("expected Pop to return false for empty queue") t.Error("expected Pop to return false for empty queue")
} }
if folder != "" {
t.Errorf("expected empty folder, got %s", folder)
if folder != "" || item != "" {
t.Errorf("expected empty results, got %s by %s", folder, item)
} }
// Add items and pop in order // Add items and pop in order
q.Add("/buckets/b1/folder1", now)
q.Add("/buckets/b1/folder2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder3", now.Add(2*time.Second))
q.Add("/buckets/b1/folder1", "i1", now)
q.Add("/buckets/b1/folder2", "i2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder3", "i3", now.Add(2*time.Second))
folder, ok = q.Pop()
folder, _, ok = q.Pop()
if !ok || folder != "/buckets/b1/folder1" { if !ok || folder != "/buckets/b1/folder1" {
t.Errorf("expected folder1, got %s (ok=%v)", folder, ok) t.Errorf("expected folder1, got %s (ok=%v)", folder, ok)
} }
folder, ok = q.Pop()
folder, _, ok = q.Pop()
if !ok || folder != "/buckets/b1/folder2" { if !ok || folder != "/buckets/b1/folder2" {
t.Errorf("expected folder2, got %s (ok=%v)", folder, ok) t.Errorf("expected folder2, got %s (ok=%v)", folder, ok)
} }
folder, ok = q.Pop()
folder, _, ok = q.Pop()
if !ok || folder != "/buckets/b1/folder3" { if !ok || folder != "/buckets/b1/folder3" {
t.Errorf("expected folder3, got %s (ok=%v)", folder, ok) t.Errorf("expected folder3, got %s (ok=%v)", folder, ok)
} }
@ -153,16 +163,16 @@ func TestCleanupQueue_Peek(t *testing.T) {
now := time.Now() now := time.Now()
// Peek empty queue // Peek empty queue
folder, _, ok := q.Peek()
folder, item, _, ok := q.Peek()
if ok { if ok {
t.Error("expected Peek to return false for empty queue") t.Error("expected Peek to return false for empty queue")
} }
// Add item and peek // Add item and peek
q.Add("/buckets/b1/folder1", now)
folder, queueTime, ok := q.Peek()
if !ok || folder != "/buckets/b1/folder1" {
t.Errorf("expected folder1, got %s (ok=%v)", folder, ok)
q.Add("/buckets/b1/folder1", "i1", now)
folder, item, queueTime, ok := q.Peek()
if !ok || folder != "/buckets/b1/folder1" || item != "i1" {
t.Errorf("expected folder1 by i1, got %s by %s (ok=%v)", folder, item, ok)
} }
if queueTime != now { if queueTime != now {
t.Errorf("expected queue time %v, got %v", now, queueTime) t.Errorf("expected queue time %v, got %v", now, queueTime)
@ -178,7 +188,7 @@ func TestCleanupQueue_Contains(t *testing.T) {
q := NewCleanupQueue(100, 10*time.Minute) q := NewCleanupQueue(100, 10*time.Minute)
now := time.Now() now := time.Now()
q.Add("/buckets/b1/folder1", now)
q.Add("/buckets/b1/folder1", "i1", now)
if !q.Contains("/buckets/b1/folder1") { if !q.Contains("/buckets/b1/folder1") {
t.Error("expected Contains to return true") t.Error("expected Contains to return true")
@ -198,14 +208,14 @@ func TestCleanupQueue_ShouldProcess_MaxSize(t *testing.T) {
} }
// Add items below max // Add items below max
q.Add("/buckets/b1/folder1", now)
q.Add("/buckets/b1/folder2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder1", "i1", now)
q.Add("/buckets/b1/folder2", "i2", now.Add(1*time.Second))
if q.ShouldProcess() { if q.ShouldProcess() {
t.Error("queue below max should not need processing") t.Error("queue below max should not need processing")
} }
// Add item to reach max // Add item to reach max
q.Add("/buckets/b1/folder3", now.Add(2*time.Second))
q.Add("/buckets/b1/folder3", "i3", now.Add(2*time.Second))
if !q.ShouldProcess() { if !q.ShouldProcess() {
t.Error("queue at max should need processing") t.Error("queue at max should need processing")
} }
@ -216,7 +226,7 @@ func TestCleanupQueue_ShouldProcess_MaxAge(t *testing.T) {
// Add item with old event time // Add item with old event time
oldTime := time.Now().Add(-1 * time.Second) // 1 second ago oldTime := time.Now().Add(-1 * time.Second) // 1 second ago
q.Add("/buckets/b1/folder1", oldTime)
q.Add("/buckets/b1/folder1", "i1", oldTime)
// Item is older than maxAge, should need processing // Item is older than maxAge, should need processing
if !q.ShouldProcess() { if !q.ShouldProcess() {
@ -225,7 +235,7 @@ func TestCleanupQueue_ShouldProcess_MaxAge(t *testing.T) {
// Clear and add fresh item // Clear and add fresh item
q.Clear() q.Clear()
q.Add("/buckets/b1/folder2", time.Now())
q.Add("/buckets/b1/folder2", "i2", time.Now())
// Fresh item should not trigger processing // Fresh item should not trigger processing
if q.ShouldProcess() { if q.ShouldProcess() {
@ -237,9 +247,9 @@ func TestCleanupQueue_Clear(t *testing.T) {
q := NewCleanupQueue(100, 10*time.Minute) q := NewCleanupQueue(100, 10*time.Minute)
now := time.Now() now := time.Now()
q.Add("/buckets/b1/folder1", now)
q.Add("/buckets/b1/folder2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder3", now.Add(2*time.Second))
q.Add("/buckets/b1/folder1", "i1", now)
q.Add("/buckets/b1/folder2", "i2", now.Add(1*time.Second))
q.Add("/buckets/b1/folder3", "i3", now.Add(2*time.Second))
q.Clear() q.Clear()
@ -261,7 +271,7 @@ func TestCleanupQueue_OldestAge(t *testing.T) {
// Add item with time in the past // Add item with time in the past
oldTime := time.Now().Add(-5 * time.Minute) oldTime := time.Now().Add(-5 * time.Minute)
q.Add("/buckets/b1/folder1", oldTime)
q.Add("/buckets/b1/folder1", "i1", oldTime)
// Age should be approximately 5 minutes // Age should be approximately 5 minutes
age := q.OldestAge() age := q.OldestAge()
@ -283,12 +293,12 @@ func TestCleanupQueue_TimeOrder(t *testing.T) {
"/buckets/b1/e", "/buckets/b1/e",
} }
for i, item := range items { for i, item := range items {
q.Add(item, baseTime.Add(time.Duration(i)*time.Second))
q.Add(item, "i", baseTime.Add(time.Duration(i)*time.Second))
} }
// Pop should return in time order // Pop should return in time order
for i, expected := range items { for i, expected := range items {
got, ok := q.Pop()
got, _, ok := q.Pop()
if !ok { if !ok {
t.Errorf("Pop %d: expected item, got empty", i) t.Errorf("Pop %d: expected item, got empty", i)
} }
@ -303,17 +313,17 @@ func TestCleanupQueue_DuplicateWithNewerTime(t *testing.T) {
baseTime := time.Now() baseTime := time.Now()
// Add items // Add items
q.Add("/buckets/b1/folder1", baseTime)
q.Add("/buckets/b1/folder2", baseTime.Add(1*time.Second))
q.Add("/buckets/b1/folder3", baseTime.Add(2*time.Second))
q.Add("/buckets/b1/folder1", "i1", baseTime)
q.Add("/buckets/b1/folder2", "i2", baseTime.Add(1*time.Second))
q.Add("/buckets/b1/folder3", "i3", baseTime.Add(2*time.Second))
// Add duplicate with newer time - should update and reposition // Add duplicate with newer time - should update and reposition
q.Add("/buckets/b1/folder1", baseTime.Add(3*time.Second))
q.Add("/buckets/b1/folder1", "i1-new", baseTime.Add(3*time.Second))
// folder1 should now be at the back (newest time) - verify by popping // folder1 should now be at the back (newest time) - verify by popping
expected := []string{"/buckets/b1/folder2", "/buckets/b1/folder3", "/buckets/b1/folder1"} expected := []string{"/buckets/b1/folder2", "/buckets/b1/folder3", "/buckets/b1/folder1"}
for i, exp := range expected { for i, exp := range expected {
folder, ok := q.Pop()
folder, _, ok := q.Pop()
if !ok || folder != exp { if !ok || folder != exp {
t.Errorf("at index %d: expected %s, got %s", i, exp, folder) t.Errorf("at index %d: expected %s, got %s", i, exp, folder)
} }
@ -328,7 +338,7 @@ func TestCleanupQueue_Concurrent(t *testing.T) {
// Concurrent adds // Concurrent adds
go func() { go func() {
for i := 0; i < 100; i++ { for i := 0; i < 100; i++ {
q.Add("/buckets/b1/folder"+string(rune('A'+i%26)), now.Add(time.Duration(i)*time.Millisecond))
q.Add("/buckets/b1/folder"+string(rune('A'+i%26)), "item", now.Add(time.Duration(i)*time.Millisecond))
} }
done <- true done <- true
}() }()

24
weed/filer/empty_folder_cleanup/empty_folder_cleaner.go

@ -162,8 +162,8 @@ func (efc *EmptyFolderCleaner) OnDeleteEvent(directory string, entryName string,
} }
// Add to cleanup queue with event time (handles out-of-order events) // Add to cleanup queue with event time (handles out-of-order events)
if efc.cleanupQueue.Add(directory, eventTime) {
glog.V(3).Infof("EmptyFolderCleaner: queued %s for cleanup", directory)
if efc.cleanupQueue.Add(directory, entryName, eventTime) {
glog.V(3).Infof("EmptyFolderCleaner: queued %s for cleanup (triggered by %s)", directory, entryName)
} }
} }
@ -232,30 +232,30 @@ func (efc *EmptyFolderCleaner) processCleanupQueue() {
} }
// Pop the oldest item // Pop the oldest item
folder, ok := efc.cleanupQueue.Pop()
folder, triggeredBy, ok := efc.cleanupQueue.Pop()
if !ok { if !ok {
break break
} }
// Execute cleanup for this folder // Execute cleanup for this folder
efc.executeCleanup(folder)
efc.executeCleanup(folder, triggeredBy)
} }
} }
// executeCleanup performs the actual cleanup of an empty folder // executeCleanup performs the actual cleanup of an empty folder
func (efc *EmptyFolderCleaner) executeCleanup(folder string) {
func (efc *EmptyFolderCleaner) executeCleanup(folder string, triggeredBy string) {
efc.mu.Lock() efc.mu.Lock()
// Quick check: if we have cached count and it's > 0, skip // Quick check: if we have cached count and it's > 0, skip
if state, exists := efc.folderCounts[folder]; exists { if state, exists := efc.folderCounts[folder]; exists {
if state.roughCount > 0 { if state.roughCount > 0 {
glog.V(3).Infof("EmptyFolderCleaner: skipping %s, cached count=%d", folder, state.roughCount)
glog.V(3).Infof("EmptyFolderCleaner: skipping %s (triggered by %s), cached count=%d", folder, triggeredBy, state.roughCount)
efc.mu.Unlock() efc.mu.Unlock()
return return
} }
// If there was an add after our delete, skip // If there was an add after our delete, skip
if !state.lastAddTime.IsZero() && state.lastAddTime.After(state.lastDelTime) { if !state.lastAddTime.IsZero() && state.lastAddTime.After(state.lastDelTime) {
glog.V(3).Infof("EmptyFolderCleaner: skipping %s, add happened after delete", folder)
glog.V(3).Infof("EmptyFolderCleaner: skipping %s (triggered by %s), add happened after delete", folder, triggeredBy)
efc.mu.Unlock() efc.mu.Unlock()
return return
} }
@ -264,7 +264,7 @@ func (efc *EmptyFolderCleaner) executeCleanup(folder string) {
// Re-check ownership (topology might have changed) // Re-check ownership (topology might have changed)
if !efc.ownsFolder(folder) { if !efc.ownsFolder(folder) {
glog.V(3).Infof("EmptyFolderCleaner: no longer owner of %s, skipping", folder)
glog.V(3).Infof("EmptyFolderCleaner: no longer owner of %s (triggered by %s), skipping", folder, triggeredBy)
return return
} }
@ -319,7 +319,7 @@ func (efc *EmptyFolderCleaner) executeCleanup(folder string) {
} }
if !isImplicit { if !isImplicit {
glog.Infof("EmptyFolderCleaner: folder %s is not marked as implicit (source=%s attr=%s), skipping", folder, implicitSource, implicitAttr)
glog.Infof("EmptyFolderCleaner: folder %s (triggered by %s) is not marked as implicit (source=%s attr=%s), skipping", folder, triggeredBy, implicitSource, implicitAttr)
return return
} }
@ -340,14 +340,14 @@ func (efc *EmptyFolderCleaner) executeCleanup(folder string) {
efc.mu.Unlock() efc.mu.Unlock()
if count > 0 { if count > 0 {
glog.V(3).Infof("EmptyFolderCleaner: folder %s has %d items, not empty", folder, count)
glog.Infof("EmptyFolderCleaner: folder %s (triggered by %s) has %d items, not empty", folder, triggeredBy, count)
return return
} }
// Delete the empty folder // Delete the empty folder
glog.V(2).Infof("EmptyFolderCleaner: deleting empty folder %s", folder)
glog.Infof("EmptyFolderCleaner: deleting empty folder %s (triggered by %s)", folder, triggeredBy)
if err := efc.deleteFolder(ctx, folder); err != nil { if err := efc.deleteFolder(ctx, folder); err != nil {
glog.V(2).Infof("EmptyFolderCleaner: failed to delete empty folder %s: %v", folder, err)
glog.V(2).Infof("EmptyFolderCleaner: failed to delete empty folder %s (triggered by %s): %v", folder, triggeredBy, err)
return return
} }

12
weed/filer/empty_folder_cleanup/empty_folder_cleaner_test.go

@ -542,7 +542,7 @@ func TestEmptyFolderCleaner_cacheEviction_skipsEntriesInQueue(t *testing.T) {
// Add a stale cache entry // Add a stale cache entry
cleaner.folderCounts[folder] = &folderState{roughCount: 0, lastCheck: oldTime} cleaner.folderCounts[folder] = &folderState{roughCount: 0, lastCheck: oldTime}
// Also add to cleanup queue // Also add to cleanup queue
cleaner.cleanupQueue.Add(folder, time.Now())
cleaner.cleanupQueue.Add(folder, "item", time.Now())
// Run eviction // Run eviction
cleaner.evictStaleCacheEntries() cleaner.evictStaleCacheEntries()
@ -588,7 +588,7 @@ func TestEmptyFolderCleaner_queueFIFOOrder(t *testing.T) {
// Verify time-sorted order by popping // Verify time-sorted order by popping
for i, expected := range folders { for i, expected := range folders {
folder, ok := cleaner.cleanupQueue.Pop()
folder, _, ok := cleaner.cleanupQueue.Pop()
if !ok || folder != expected { if !ok || folder != expected {
t.Errorf("expected folder %s at index %d, got %s", expected, i, folder) t.Errorf("expected folder %s at index %d, got %s", expected, i, folder)
} }
@ -630,9 +630,9 @@ func TestEmptyFolderCleaner_processCleanupQueue_drainsAllOnceTriggered(t *testin
} }
now := time.Now() now := time.Now()
cleaner.cleanupQueue.Add("/buckets/test/folder1", now)
cleaner.cleanupQueue.Add("/buckets/test/folder2", now.Add(time.Millisecond))
cleaner.cleanupQueue.Add("/buckets/test/folder3", now.Add(2*time.Millisecond))
cleaner.cleanupQueue.Add("/buckets/test/folder1", "i1", now)
cleaner.cleanupQueue.Add("/buckets/test/folder2", "i2", now.Add(time.Millisecond))
cleaner.cleanupQueue.Add("/buckets/test/folder3", "i3", now.Add(2*time.Millisecond))
cleaner.processCleanupQueue() cleaner.processCleanupQueue()
@ -677,7 +677,7 @@ func TestEmptyFolderCleaner_executeCleanup_missingImplicitAttributeSkips(t *test
} }
folder := "/buckets/test/folder" folder := "/buckets/test/folder"
cleaner.executeCleanup(folder)
cleaner.executeCleanup(folder, "triggered_item")
if len(deleted) != 0 { if len(deleted) != 0 {
t.Fatalf("expected folder %s to be skipped, got deletions %v", folder, deleted) t.Fatalf("expected folder %s to be skipped, got deletions %v", folder, deleted)

2
weed/s3api/s3api_object_handlers_delete.go

@ -22,7 +22,7 @@ const (
func (s3a *S3ApiServer) DeleteObjectHandler(w http.ResponseWriter, r *http.Request) { func (s3a *S3ApiServer) DeleteObjectHandler(w http.ResponseWriter, r *http.Request) {
bucket, object := s3_constants.GetBucketAndObject(r) bucket, object := s3_constants.GetBucketAndObject(r)
glog.V(3).Infof("DeleteObjectHandler %s %s", bucket, object)
glog.Infof("DeleteObjectHandler %s %s", bucket, object)
if err := s3a.validateTableBucketObjectPath(bucket, object); err != nil { if err := s3a.validateTableBucketObjectPath(bucket, object); err != nil {
s3err.WriteErrorResponse(w, r, s3err.ErrAccessDenied) s3err.WriteErrorResponse(w, r, s3err.ErrAccessDenied)
return return

Loading…
Cancel
Save