mirror of
https://github.com/yusing/godoxy.git
synced 2026-03-27 11:31:06 +01:00
305 lines
6.8 KiB
Markdown
305 lines
6.8 KiB
Markdown
# Load Balancer
|
|
|
|
Load balancing package providing multiple distribution algorithms, sticky sessions, and server health management.
|
|
|
|
## Overview
|
|
|
|
This package implements a flexible load balancer for distributing HTTP requests across multiple backend servers. It supports multiple balancing algorithms and integrates with GoDoxy's task management and health monitoring systems.
|
|
|
|
## Architecture
|
|
|
|
```mermaid
|
|
graph TD
|
|
A[HTTP Request] --> B[LoadBalancer]
|
|
B --> C{Algorithm}
|
|
C -->|Round Robin| D[RoundRobin]
|
|
C -->|Least Connections| E[LeastConn]
|
|
C -->|IP Hash| F[IPHash]
|
|
|
|
D --> G[Available Servers]
|
|
E --> G
|
|
F --> G
|
|
|
|
G --> H[Server Selection]
|
|
H --> I{Sticky Session?}
|
|
I -->|Yes| J[Set Cookie]
|
|
I -->|No| K[Continue]
|
|
|
|
J --> L[ServeHTTP]
|
|
K --> L
|
|
```
|
|
|
|
## Algorithms
|
|
|
|
### Round Robin
|
|
|
|
Distributes requests evenly across all available servers in sequence.
|
|
|
|
```mermaid
|
|
sequenceDiagram
|
|
participant C as Client
|
|
participant LB as LoadBalancer
|
|
participant S1 as Server 1
|
|
participant S2 as Server 2
|
|
participant S3 as Server 3
|
|
|
|
C->>LB: Request 1
|
|
LB->>S1: Route to Server 1
|
|
C->>LB: Request 2
|
|
LB->>S2: Route to Server 2
|
|
C->>LB: Request 3
|
|
LB->>S3: Route to Server 3
|
|
C->>LB: Request 4
|
|
LB->>S1: Route to Server 1
|
|
```
|
|
|
|
### Least Connections
|
|
|
|
Routes requests to the server with the fewest active connections.
|
|
|
|
```mermaid
|
|
flowchart LR
|
|
subgraph LB["Load Balancer"]
|
|
direction TB
|
|
A["Server A<br/>3 connections"]
|
|
B["Server B<br/>1 connection"]
|
|
C["Server C<br/>5 connections"]
|
|
end
|
|
|
|
New["New Request"] --> B
|
|
```
|
|
|
|
### IP Hash
|
|
|
|
Consistently routes requests from the same client IP to the same server using hash-based distribution.
|
|
|
|
```mermaid
|
|
graph LR
|
|
Client1["Client IP: 192.168.1.10"] -->|Hash| ServerA
|
|
Client2["Client IP: 192.168.1.20"] -->|Hash| ServerB
|
|
Client3["Client IP: 192.168.1.30"] -->|Hash| ServerA
|
|
```
|
|
|
|
## Core Components
|
|
|
|
### LoadBalancer
|
|
|
|
```go
|
|
type LoadBalancer struct {
|
|
*types.LoadBalancerConfig
|
|
task *task.Task
|
|
pool pool.Pool[types.LoadBalancerServer]
|
|
poolMu sync.Mutex
|
|
sumWeight int
|
|
startTime time.Time
|
|
}
|
|
```
|
|
|
|
**Key Methods:**
|
|
|
|
```go
|
|
// Create a new load balancer from configuration
|
|
func New(cfg *types.LoadBalancerConfig) *LoadBalancer
|
|
|
|
// Start the load balancer as a background task
|
|
func (lb *LoadBalancer) Start(parent task.Parent) gperr.Error
|
|
|
|
// Update configuration dynamically
|
|
func (lb *LoadBalancer) UpdateConfigIfNeeded(cfg *types.LoadBalancerConfig)
|
|
|
|
// Add a backend server
|
|
func (lb *LoadBalancer) AddServer(srv types.LoadBalancerServer)
|
|
|
|
// Remove a backend server
|
|
func (lb *LoadBalancer) RemoveServer(srv types.LoadBalancerServer)
|
|
|
|
// ServeHTTP implements http.Handler
|
|
func (lb *LoadBalancer) ServeHTTP(rw http.ResponseWriter, r *http.Request)
|
|
```
|
|
|
|
### Server
|
|
|
|
```go
|
|
type server struct {
|
|
name string
|
|
url *nettypes.URL
|
|
weight int
|
|
http.Handler
|
|
types.HealthMonitor
|
|
}
|
|
|
|
// Create a new backend server
|
|
func NewServer(name string, url *nettypes.URL, weight int, handler http.Handler, healthMon types.HealthMonitor) types.LoadBalancerServer
|
|
```
|
|
|
|
**Server Interface:**
|
|
|
|
```go
|
|
type LoadBalancerServer interface {
|
|
Name() string
|
|
URL() *nettypes.URL
|
|
Key() string
|
|
Weight() int
|
|
SetWeight(weight int)
|
|
Status() types.HealthStatus
|
|
Latency() time.Duration
|
|
ServeHTTP(rw http.ResponseWriter, r *http.Request)
|
|
TryWake() error
|
|
}
|
|
```
|
|
|
|
### Sticky Sessions
|
|
|
|
The load balancer supports sticky sessions via cookies:
|
|
|
|
```mermaid
|
|
flowchart TD
|
|
A[Client Request] --> B{Cookie exists?}
|
|
B -->|No| C[Select Server]
|
|
B -->|Yes| D[Extract Server Hash]
|
|
D --> E[Find Matching Server]
|
|
C --> F[Set Cookie<br/>godoxy_lb_sticky]
|
|
E --> G[Route to Server]
|
|
F --> G
|
|
```
|
|
|
|
```go
|
|
// Cookie settings
|
|
Name: "godoxy_lb_sticky"
|
|
MaxAge: Configurable (default: 24 hours)
|
|
HttpOnly: true
|
|
SameSite: Lax
|
|
Secure: Based on TLS/Forwarded-Proto
|
|
```
|
|
|
|
## Balancing Modes
|
|
|
|
```go
|
|
const (
|
|
LoadbalanceModeUnset = ""
|
|
LoadbalanceModeRoundRobin = "round_robin"
|
|
LoadbalanceModeLeastConn = "least_conn"
|
|
LoadbalanceModeIPHash = "ip_hash"
|
|
)
|
|
```
|
|
|
|
## Configuration
|
|
|
|
```go
|
|
type LoadBalancerConfig struct {
|
|
Link string // Link name
|
|
Mode LoadbalanceMode // Balancing algorithm
|
|
Sticky bool // Enable sticky sessions
|
|
StickyMaxAge time.Duration // Cookie max age
|
|
Options map[string]any // Algorithm-specific options
|
|
}
|
|
```
|
|
|
|
## Usage Examples
|
|
|
|
### Basic Round Robin Load Balancer
|
|
|
|
```go
|
|
config := &types.LoadBalancerConfig{
|
|
Link: "my-service",
|
|
Mode: types.LoadbalanceModeRoundRobin,
|
|
}
|
|
|
|
lb := loadbalancer.New(config)
|
|
lb.Start(parentTask)
|
|
|
|
// Add backend servers
|
|
lb.AddServer(loadbalancer.NewServer("backend-1", url1, 10, handler1, health1))
|
|
lb.AddServer(loadbalancer.NewServer("backend-2", url2, 10, handler2, health2))
|
|
|
|
// Use as HTTP handler
|
|
http.Handle("/", lb)
|
|
```
|
|
|
|
### Least Connections with Sticky Sessions
|
|
|
|
```go
|
|
config := &types.LoadBalancerConfig{
|
|
Link: "api-service",
|
|
Mode: types.LoadbalanceModeLeastConn,
|
|
Sticky: true,
|
|
StickyMaxAge: 1 * time.Hour,
|
|
}
|
|
|
|
lb := loadbalancer.New(config)
|
|
lb.Start(parentTask)
|
|
|
|
for _, srv := range backends {
|
|
lb.AddServer(srv)
|
|
}
|
|
```
|
|
|
|
### IP Hash Load Balancer with Real IP
|
|
|
|
```go
|
|
config := &types.LoadBalancerConfig{
|
|
Link: "user-service",
|
|
Mode: types.LoadbalanceModeIPHash,
|
|
Options: map[string]any{
|
|
"header": "X-Real-IP",
|
|
"from": []string{"10.0.0.0/8", "172.16.0.0/12"},
|
|
"recursive": true,
|
|
},
|
|
}
|
|
|
|
lb := loadbalancer.New(config)
|
|
```
|
|
|
|
### Server Weight Management
|
|
|
|
```go
|
|
// Servers are balanced based on weight (max total: 100)
|
|
lb.AddServer(NewServer("server1", url1, 30, handler, health))
|
|
lb.AddServer(NewServer("server2", url2, 50, handler, health))
|
|
lb.AddServer(NewServer("server3", url3, 20, handler, health))
|
|
|
|
// Weights are auto-rebalanced if total != 100
|
|
```
|
|
|
|
## Idlewatcher Integration
|
|
|
|
The load balancer integrates with the idlewatcher system:
|
|
|
|
- Wake events path (`/api/wake`): Wakes all idle servers
|
|
- Favicon and loading page paths: Bypassed for sticky session handling
|
|
- Server wake support via `TryWake()` interface
|
|
|
|
## Health Monitoring
|
|
|
|
The load balancer implements `types.HealthMonitor`:
|
|
|
|
```go
|
|
func (lb *LoadBalancer) Status() types.HealthStatus
|
|
func (lb *LoadBalancer) Detail() string
|
|
func (lb *LoadBalancer) Uptime() time.Duration
|
|
func (lb *LoadBalancer) Latency() time.Duration
|
|
```
|
|
|
|
Health JSON representation:
|
|
|
|
```json
|
|
{
|
|
"name": "my-service",
|
|
"status": "healthy",
|
|
"detail": "3/3 servers are healthy",
|
|
"started": "2024-01-01T00:00:00Z",
|
|
"uptime": "1h2m3s",
|
|
"latency": "10ms",
|
|
"extra": {
|
|
"config": {...},
|
|
"pool": {...}
|
|
}
|
|
}
|
|
```
|
|
|
|
## Thread Safety
|
|
|
|
- Server pool operations are protected by `poolMu` mutex
|
|
- Algorithm-specific state uses atomic operations or dedicated synchronization
|
|
- Least connections uses `xsync.Map` for thread-safe connection counting
|