mirror of
https://github.com/jhillyerd/inbucket.git
synced 2026-01-26 21:15:57 +00:00
feat: Monitor tab dynamically updates when messages are deleted (#337)
* WIP: msghub handles deletes, UI does not yet display them Signed-off-by: James Hillyerd <james@hillyerd.com> * socket and UI support message deletes Signed-off-by: James Hillyerd <james@hillyerd.com> * use Delete naming for consistency Signed-off-by: James Hillyerd <james@hillyerd.com> --------- Signed-off-by: James Hillyerd <james@hillyerd.com>
This commit is contained in:
@@ -15,6 +15,7 @@ const opChanLen = 100
|
||||
// Listener receives the contents of the history buffer, followed by new messages
|
||||
type Listener interface {
|
||||
Receive(msg event.MessageMetadata) error
|
||||
Delete(mailbox string, id string) error
|
||||
}
|
||||
|
||||
// Hub relays messages on to its listeners
|
||||
@@ -42,6 +43,12 @@ func New(historyLen int, extHost *extension.Host) *Hub {
|
||||
return nil
|
||||
})
|
||||
|
||||
extHost.Events.AfterMessageDeleted.AddListener("msghub",
|
||||
func(msg event.MessageMetadata) *extension.Void {
|
||||
hub.Delete(msg.Mailbox, msg.ID)
|
||||
return nil
|
||||
})
|
||||
|
||||
return hub
|
||||
}
|
||||
|
||||
@@ -68,7 +75,7 @@ func (hub *Hub) Dispatch(msg event.MessageMetadata) {
|
||||
h.history.Value = msg
|
||||
h.history = h.history.Next()
|
||||
|
||||
// Deliver message to all listeners, removing listeners if they return an error
|
||||
// Relay event to all listeners, removing listeners if they return an error.
|
||||
for l := range h.listeners {
|
||||
if err := l.Receive(msg); err != nil {
|
||||
delete(h.listeners, l)
|
||||
@@ -78,6 +85,37 @@ func (hub *Hub) Dispatch(msg event.MessageMetadata) {
|
||||
}
|
||||
}
|
||||
|
||||
// Delete removes the message from the history buffer and instructs listeners to do the same.
|
||||
func (hub *Hub) Delete(mailbox string, id string) {
|
||||
hub.opChan <- func(h *Hub) {
|
||||
if h.history == nil {
|
||||
return
|
||||
}
|
||||
|
||||
// Locate and remove history entry.
|
||||
p := h.history
|
||||
end := p
|
||||
for {
|
||||
if next, ok := p.Next().Value.(event.MessageMetadata); ok {
|
||||
if mailbox == next.Mailbox && id == next.ID {
|
||||
p.Unlink(1) // Remove next node.
|
||||
break
|
||||
}
|
||||
}
|
||||
if p = p.Next(); p == end {
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
// Relay event to all listeners, removing listeners if they return an error.
|
||||
for l := range h.listeners {
|
||||
if err := l.Delete(mailbox, id); err != nil {
|
||||
delete(h.listeners, l)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// AddListener registers a listener to receive broadcasted messages.
|
||||
func (hub *Hub) AddListener(l Listener) {
|
||||
hub.opChan <- func(h *Hub) {
|
||||
|
||||
@@ -3,6 +3,7 @@ package msghub
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"strconv"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
@@ -12,9 +13,11 @@ import (
|
||||
|
||||
// testListener implements the Listener interface, mock for unit tests
|
||||
type testListener struct {
|
||||
messages []*event.MessageMetadata // received messages
|
||||
wantMessages int // how many messages this listener wants to receive
|
||||
errorAfter int // when != 0, messages until Receive() begins returning error
|
||||
messages []*event.MessageMetadata // received messages
|
||||
deletes []string // received deletes
|
||||
wantEvents int // how many events this listener wants to receive
|
||||
errorAfter int // when != 0, event count until Receive() begins returning error
|
||||
gotEvents int
|
||||
|
||||
done chan struct{} // closed once we have received wantMessages
|
||||
overflow chan struct{} // closed if we receive wantMessages+1
|
||||
@@ -22,10 +25,11 @@ type testListener struct {
|
||||
|
||||
func newTestListener(want int) *testListener {
|
||||
l := &testListener{
|
||||
messages: make([]*event.MessageMetadata, 0, want*2),
|
||||
wantMessages: want,
|
||||
done: make(chan struct{}),
|
||||
overflow: make(chan struct{}),
|
||||
messages: make([]*event.MessageMetadata, 0, want*2),
|
||||
deletes: make([]string, 0, want*2),
|
||||
wantEvents: want,
|
||||
done: make(chan struct{}),
|
||||
overflow: make(chan struct{}),
|
||||
}
|
||||
if want == 0 {
|
||||
close(l.done)
|
||||
@@ -36,22 +40,29 @@ func newTestListener(want int) *testListener {
|
||||
// Receive a Message, store it in the messages slice, close applicable channels, and return an error
|
||||
// if instructed
|
||||
func (l *testListener) Receive(msg event.MessageMetadata) error {
|
||||
l.gotEvents++
|
||||
l.messages = append(l.messages, &msg)
|
||||
if len(l.messages) == l.wantMessages {
|
||||
if l.gotEvents == l.wantEvents {
|
||||
close(l.done)
|
||||
}
|
||||
if len(l.messages) == l.wantMessages+1 {
|
||||
if l.gotEvents == l.wantEvents+1 {
|
||||
close(l.overflow)
|
||||
}
|
||||
if l.errorAfter > 0 && len(l.messages) > l.errorAfter {
|
||||
if l.errorAfter > 0 && l.gotEvents > l.errorAfter {
|
||||
return fmt.Errorf("Too many messages")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (l *testListener) Delete(mailbox string, id string) error {
|
||||
l.gotEvents++
|
||||
l.deletes = append(l.deletes, mailbox+"/"+id)
|
||||
return nil
|
||||
}
|
||||
|
||||
// String formats the got vs wanted message counts
|
||||
func (l *testListener) String() string {
|
||||
return fmt.Sprintf("got %v messages, wanted %v", len(l.messages), l.wantMessages)
|
||||
return fmt.Sprintf("got %v messages, wanted %v", len(l.messages), l.wantEvents)
|
||||
}
|
||||
|
||||
func TestHubNew(t *testing.T) {
|
||||
@@ -198,6 +209,55 @@ func TestHubHistoryReplay(t *testing.T) {
|
||||
}
|
||||
}
|
||||
|
||||
func TestHubHistoryDelete(t *testing.T) {
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
defer cancel()
|
||||
hub := New(100, extension.NewHost())
|
||||
go hub.Start(ctx)
|
||||
l1 := newTestListener(3)
|
||||
hub.AddListener(l1)
|
||||
|
||||
// Broadcast 3 messages with no listeners
|
||||
msgs := make([]event.MessageMetadata, 3)
|
||||
for i := 0; i < len(msgs); i++ {
|
||||
msgs[i] = event.MessageMetadata{
|
||||
Mailbox: "hub",
|
||||
ID: strconv.Itoa(i),
|
||||
Subject: fmt.Sprintf("subj %v", i),
|
||||
}
|
||||
hub.Dispatch(msgs[i])
|
||||
}
|
||||
|
||||
// Wait for messages (live)
|
||||
select {
|
||||
case <-l1.done:
|
||||
case <-time.After(time.Second):
|
||||
t.Fatal("Timeout:", l1)
|
||||
}
|
||||
|
||||
hub.Delete("hub", "1") // Delete a message
|
||||
hub.Delete("zzz", "0") // Attempt to delete non-existent mailbox message
|
||||
|
||||
// Add a new listener, waits for 2 messages
|
||||
l2 := newTestListener(2)
|
||||
hub.AddListener(l2)
|
||||
|
||||
// Wait for messages (history)
|
||||
select {
|
||||
case <-l2.done:
|
||||
case <-time.After(time.Second):
|
||||
t.Fatal("Timeout:", l2)
|
||||
}
|
||||
|
||||
want := []string{"subj 0", "subj 2"}
|
||||
for i := 0; i < len(want); i++ {
|
||||
got := l2.messages[i].Subject
|
||||
if got != want[i] {
|
||||
t.Errorf("msg[%v].Subject == %q, want %q", i, got, want[i])
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestHubHistoryReplayWrap(t *testing.T) {
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
defer cancel()
|
||||
|
||||
Reference in New Issue
Block a user