mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2026-06-13 23:36:45 +03:00
4f17c6661a
Random allocation could pick 33646 = admin.port (23646) + GrpcPortOffset. weed mini reserves that as Admin's gRPC port even when the test only overrides Master/Filer/S3/Iceberg, so the explicit Filer flag failed with "reserved for gRPC calculation" and TestRisingWaveIcebergCatalog flaked. Pre-seed the reserved set with every mini default HTTP port plus its +10000 offset so a random pick (or its own gRPC offset) cannot land on a service the caller left at its default.
211 lines
5.8 KiB
Go
211 lines
5.8 KiB
Go
// Package testutil provides shared test utilities for SeaweedFS integration tests.
|
|
package testutil
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"net"
|
|
"testing"
|
|
)
|
|
|
|
// GrpcPortOffset is the offset weed mini uses to derive gRPC ports from HTTP ports.
|
|
const GrpcPortOffset = 10000
|
|
|
|
// miniDefaultPorts are the weed mini flag defaults (see weed/command/mini.go).
|
|
// A test only overrides services it uses; unspecified services still bind
|
|
// these defaults, so allocation must avoid handing them out (or any value
|
|
// whose gRPC offset would collide with them).
|
|
var miniDefaultPorts = []int{
|
|
9333, // master.port
|
|
8888, // filer.port
|
|
9340, // volume.port
|
|
8333, // s3.port
|
|
8181, // s3.port.iceberg
|
|
7333, // webdav.port
|
|
23646, // admin.port
|
|
}
|
|
|
|
func reservedMiniPorts() map[int]bool {
|
|
r := make(map[int]bool, len(miniDefaultPorts)*2)
|
|
for _, p := range miniDefaultPorts {
|
|
r[p] = true
|
|
r[p+GrpcPortOffset] = true
|
|
}
|
|
return r
|
|
}
|
|
|
|
// AllocatePorts allocates count unique free ports atomically.
|
|
// All listeners are held open until every port is obtained, preventing
|
|
// the OS from recycling a port between successive allocations.
|
|
func AllocatePorts(count int) ([]int, error) {
|
|
listeners := make([]net.Listener, 0, count)
|
|
ports := make([]int, 0, count)
|
|
for i := 0; i < count; i++ {
|
|
l, err := net.Listen("tcp", "127.0.0.1:0")
|
|
if err != nil {
|
|
for _, ll := range listeners {
|
|
_ = ll.Close()
|
|
}
|
|
return nil, err
|
|
}
|
|
listeners = append(listeners, l)
|
|
ports = append(ports, l.Addr().(*net.TCPAddr).Port)
|
|
}
|
|
for _, l := range listeners {
|
|
_ = l.Close()
|
|
}
|
|
return ports, nil
|
|
}
|
|
|
|
// MustAllocatePorts is a testing wrapper for AllocatePorts.
|
|
func MustAllocatePorts(t *testing.T, count int) []int {
|
|
t.Helper()
|
|
ports, err := AllocatePorts(count)
|
|
if err != nil {
|
|
t.Fatalf("Failed to allocate %d free ports: %v", count, err)
|
|
}
|
|
return ports
|
|
}
|
|
|
|
// AllocateMiniPorts allocates n free ports where each port and its gRPC
|
|
// counterpart (port + GrpcPortOffset) are available and don't collide
|
|
// with any other allocated port or its gRPC counterpart. All listeners
|
|
// are held open until the entire batch is allocated, preventing the OS
|
|
// from recycling ports between allocations. Use this when ports will be
|
|
// passed to weed mini without explicit gRPC port flags, so mini will
|
|
// derive gRPC ports as HTTP + 10000.
|
|
//
|
|
// Listeners are bound on all interfaces (":port") rather than 127.0.0.1
|
|
// to match weed mini's availability check (isPortAvailable). A port can
|
|
// be free on loopback but held by another process on a different
|
|
// interface; reserving only on loopback lets mini's check fail and
|
|
// trigger gRPC port shifting, which then causes weed shell to dial the
|
|
// wrong port and hang.
|
|
func AllocateMiniPorts(count int) ([]int, error) {
|
|
const (
|
|
minPort = 10000
|
|
maxPort = 55000
|
|
)
|
|
reserved := reservedMiniPorts()
|
|
ports := make([]int, 0, count)
|
|
var listeners []net.Listener
|
|
defer func() {
|
|
for _, l := range listeners {
|
|
l.Close()
|
|
}
|
|
}()
|
|
|
|
for idx := 0; idx < count; idx++ {
|
|
found := false
|
|
for i := 0; i < 1000; i++ {
|
|
port := minPort + rand.Intn(maxPort-minPort)
|
|
grpcPort := port + GrpcPortOffset
|
|
|
|
if reserved[port] || reserved[grpcPort] {
|
|
continue
|
|
}
|
|
|
|
l1, err := net.Listen("tcp", fmt.Sprintf(":%d", port))
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
l2, err := net.Listen("tcp", fmt.Sprintf(":%d", grpcPort))
|
|
if err != nil {
|
|
l1.Close()
|
|
continue
|
|
}
|
|
|
|
listeners = append(listeners, l1, l2)
|
|
reserved[port] = true
|
|
reserved[grpcPort] = true
|
|
ports = append(ports, port)
|
|
found = true
|
|
break
|
|
}
|
|
if !found {
|
|
return nil, fmt.Errorf("failed to allocate mini port %d of %d", idx+1, count)
|
|
}
|
|
}
|
|
|
|
return ports, nil
|
|
}
|
|
|
|
// MustFreeMiniPorts allocates n ports suitable for weed mini, ensuring
|
|
// each port's gRPC offset (port + 10000) doesn't collide with any other
|
|
// allocated port. names is used only for error messages.
|
|
func MustFreeMiniPorts(t *testing.T, names []string) []int {
|
|
t.Helper()
|
|
ports, err := AllocateMiniPorts(len(names))
|
|
if err != nil {
|
|
t.Fatalf("failed to allocate mini ports for %v: %v", names, err)
|
|
}
|
|
return ports
|
|
}
|
|
|
|
// MustFreeMiniPort allocates a single weed mini port.
|
|
func MustFreeMiniPort(t *testing.T, name string) int {
|
|
t.Helper()
|
|
return MustFreeMiniPorts(t, []string{name})[0]
|
|
}
|
|
|
|
// AllocatePortSet atomically allocates miniCount master-style port pairs (each
|
|
// with port and port+GrpcPortOffset reserved) plus regularCount additional
|
|
// distinct ports. All listeners are held open until the entire batch is
|
|
// allocated, so a mini gRPC port cannot be recycled as a regular port mid-batch.
|
|
func AllocatePortSet(miniCount, regularCount int) (mini []int, regular []int, err error) {
|
|
const (
|
|
minPort = 10000
|
|
maxPort = 55000
|
|
)
|
|
reserved := reservedMiniPorts()
|
|
mini = make([]int, 0, miniCount)
|
|
var listeners []net.Listener
|
|
defer func() {
|
|
for _, l := range listeners {
|
|
l.Close()
|
|
}
|
|
}()
|
|
|
|
for idx := 0; idx < miniCount; idx++ {
|
|
found := false
|
|
for i := 0; i < 1000; i++ {
|
|
port := minPort + rand.Intn(maxPort-minPort)
|
|
grpcPort := port + GrpcPortOffset
|
|
if reserved[port] || reserved[grpcPort] {
|
|
continue
|
|
}
|
|
l1, lErr := net.Listen("tcp", fmt.Sprintf(":%d", port))
|
|
if lErr != nil {
|
|
continue
|
|
}
|
|
l2, lErr := net.Listen("tcp", fmt.Sprintf(":%d", grpcPort))
|
|
if lErr != nil {
|
|
l1.Close()
|
|
continue
|
|
}
|
|
listeners = append(listeners, l1, l2)
|
|
reserved[port] = true
|
|
reserved[grpcPort] = true
|
|
mini = append(mini, port)
|
|
found = true
|
|
break
|
|
}
|
|
if !found {
|
|
return nil, nil, fmt.Errorf("failed to allocate mini port %d of %d", idx+1, miniCount)
|
|
}
|
|
}
|
|
|
|
regular = make([]int, 0, regularCount)
|
|
for i := 0; i < regularCount; i++ {
|
|
l, lErr := net.Listen("tcp", ":0")
|
|
if lErr != nil {
|
|
return nil, nil, lErr
|
|
}
|
|
listeners = append(listeners, l)
|
|
regular = append(regular, l.Addr().(*net.TCPAddr).Port)
|
|
}
|
|
|
|
return mini, regular, nil
|
|
}
|