mirror of
https://github.com/fatedier/frp.git
synced 2026-03-16 23:09:16 +08:00
* server/group: refactor group package with shared abstractions and fix concurrency issues Extract common patterns into reusable components: - groupRegistry[G]: generic concurrent map for group lifecycle management - baseGroup: shared plumbing for listener-based groups (TCP, HTTPS, TCPMux) - Listener: unified virtual listener replacing 3 identical implementations Fix concurrency issues: - Stale-pointer race: isCurrent check + errGroupStale + controller retry loops - Worker generation safety: pass realLn and acceptCh as params instead of reading mutable fields - Connection leak: close conn on worker panic recovery path - ABBA deadlock in HTTP UnRegister: consistent lock ordering (group.mu -> registry.mu) - Round-robin overflow in HTTPGroup: use unsigned modulo Add unit tests (17 tests) for registry, listener, and baseGroup. Add TCPMux group load balancing e2e test. * server/group: replace tautological assertion with require.NotPanics * server/group: remove blank line between doc comment and type declaration
103 lines
2.4 KiB
Go
103 lines
2.4 KiB
Go
package group
|
|
|
|
import (
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestGetOrCreate_New(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
called := 0
|
|
v := 42
|
|
got := r.getOrCreate("k", func() *int { called++; return &v })
|
|
assert.Equal(t, 1, called)
|
|
assert.Equal(t, &v, got)
|
|
}
|
|
|
|
func TestGetOrCreate_Existing(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
v := 42
|
|
r.getOrCreate("k", func() *int { return &v })
|
|
|
|
called := 0
|
|
got := r.getOrCreate("k", func() *int { called++; return nil })
|
|
assert.Equal(t, 0, called)
|
|
assert.Equal(t, &v, got)
|
|
}
|
|
|
|
func TestGet_ExistingAndMissing(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
v := 1
|
|
r.getOrCreate("k", func() *int { return &v })
|
|
|
|
got, ok := r.get("k")
|
|
assert.True(t, ok)
|
|
assert.Equal(t, &v, got)
|
|
|
|
_, ok = r.get("missing")
|
|
assert.False(t, ok)
|
|
}
|
|
|
|
func TestIsCurrent(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
v1 := 1
|
|
v2 := 2
|
|
r.getOrCreate("k", func() *int { return &v1 })
|
|
|
|
assert.True(t, r.isCurrent("k", func(g *int) bool { return g == &v1 }))
|
|
assert.False(t, r.isCurrent("k", func(g *int) bool { return g == &v2 }))
|
|
assert.False(t, r.isCurrent("missing", func(g *int) bool { return true }))
|
|
}
|
|
|
|
func TestRemoveIf(t *testing.T) {
|
|
t.Run("removes when fn returns true", func(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
v := 1
|
|
r.getOrCreate("k", func() *int { return &v })
|
|
r.removeIf("k", func(g *int) bool { return g == &v })
|
|
_, ok := r.get("k")
|
|
assert.False(t, ok)
|
|
})
|
|
|
|
t.Run("keeps when fn returns false", func(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
v := 1
|
|
r.getOrCreate("k", func() *int { return &v })
|
|
r.removeIf("k", func(g *int) bool { return false })
|
|
_, ok := r.get("k")
|
|
assert.True(t, ok)
|
|
})
|
|
|
|
t.Run("noop on missing key", func(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
r.removeIf("missing", func(g *int) bool { return true }) // should not panic
|
|
})
|
|
}
|
|
|
|
func TestConcurrentGetOrCreateAndRemoveIf(t *testing.T) {
|
|
r := newGroupRegistry[*int]()
|
|
const n = 100
|
|
var wg sync.WaitGroup
|
|
wg.Add(n * 2)
|
|
for i := range n {
|
|
v := i
|
|
go func() {
|
|
defer wg.Done()
|
|
r.getOrCreate("k", func() *int { return &v })
|
|
}()
|
|
go func() {
|
|
defer wg.Done()
|
|
r.removeIf("k", func(*int) bool { return true })
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
|
|
// After all goroutines finish, accessing the key must not panic.
|
|
require.NotPanics(t, func() {
|
|
_, _ = r.get("k")
|
|
})
|
|
}
|