lolly/internal/config/config_test.go

485 lines
11 KiB
Go

// Package config 提供配置加载和管理的测试。
package config
import (
"os"
"path/filepath"
"strings"
"testing"
)
// TestLoad 测试从文件加载配置。
func TestLoad(t *testing.T) {
t.Run("有效配置文件", func(t *testing.T) {
// 创建临时配置文件
content := `
servers:
- listen: ":8080"
static:
- path: "/"
root: "/var/www"
index:
- "index.html"
logging:
access:
path: "/var/log/access.log"
format: "combined"
error:
path: "/var/log/error.log"
level: "info"
performance:
goroutine_pool:
enabled: true
max_workers: 100
file_cache:
max_entries: 1000
monitoring:
status:
path: "/status"
`
tmpDir := t.TempDir()
tmpFile := filepath.Join(tmpDir, "config.yaml")
if err := os.WriteFile(tmpFile, []byte(content), 0o644); err != nil {
t.Fatalf("创建临时配置文件失败: %v", err)
}
cfg, err := Load(tmpFile)
if err != nil {
t.Fatalf("Load() 失败: %v", err)
}
if cfg.Servers[0].Listen != ":8080" {
t.Errorf("Servers[0].Listen = %q, want %q", cfg.Servers[0].Listen, ":8080")
}
if cfg.Servers[0].Static[0].Root != "/var/www" {
t.Errorf("Servers[0].Static.Root = %q, want %q", cfg.Servers[0].Static[0].Root, "/var/www")
}
if len(cfg.Servers[0].Static[0].Index) != 1 || cfg.Servers[0].Static[0].Index[0] != "index.html" {
t.Errorf("Servers[0].Static.Index = %v, want [index.html]", cfg.Servers[0].Static[0].Index)
}
})
t.Run("文件不存在", func(t *testing.T) {
_, err := Load("/nonexistent/path/config.yaml")
if err == nil {
t.Error("Load() 期望返回错误,但返回 nil")
}
})
t.Run("无效YAML", func(t *testing.T) {
content := `
server:
listen: ":8080"
static:
root: [invalid yaml structure
`
tmpDir := t.TempDir()
tmpFile := filepath.Join(tmpDir, "invalid.yaml")
if err := os.WriteFile(tmpFile, []byte(content), 0o644); err != nil {
t.Fatalf("创建临时配置文件失败: %v", err)
}
_, err := Load(tmpFile)
if err == nil {
t.Error("Load() 期望返回错误,但返回 nil")
}
})
t.Run("缺少必填字段(无服务器配置)", func(t *testing.T) {
content := `
logging:
access:
path: "/var/log/access.log"
`
tmpDir := t.TempDir()
tmpFile := filepath.Join(tmpDir, "no_server.yaml")
if err := os.WriteFile(tmpFile, []byte(content), 0o644); err != nil {
t.Fatalf("创建临时配置文件失败: %v", err)
}
_, err := Load(tmpFile)
if err == nil {
t.Error("Load() 期望返回错误,但返回 nil")
}
})
t.Run("多虚拟主机模式", func(t *testing.T) {
content := `
servers:
- listen: ":8080"
name: "server1"
- listen: ":8081"
name: "server2"
`
tmpDir := t.TempDir()
tmpFile := filepath.Join(tmpDir, "multi.yaml")
if err := os.WriteFile(tmpFile, []byte(content), 0o644); err != nil {
t.Fatalf("创建临时配置文件失败: %v", err)
}
cfg, err := Load(tmpFile)
if err != nil {
t.Fatalf("Load() 失败: %v", err)
}
if len(cfg.Servers) != 2 {
t.Fatalf("len(Servers) = %d, want 2", len(cfg.Servers))
}
if cfg.Servers[0].Name != "server1" {
t.Errorf("Servers[0].Name = %q, want %q", cfg.Servers[0].Name, "server1")
}
if cfg.Servers[1].Name != "server2" {
t.Errorf("Servers[1].Name = %q, want %q", cfg.Servers[1].Name, "server2")
}
})
}
// TestConfigMethods 测试 Config 结构体的方法。
func TestProxyBufferingConfig_ParseBuffers(t *testing.T) {
tests := []struct {
name string
buffers string
bufferSize int
wantCount int
wantSizeEach int
}{
{
name: "empty uses buffer_size",
buffers: "",
bufferSize: 4096,
wantCount: 1,
wantSizeEach: 4096,
},
{
name: "8 16k format",
buffers: "8 16k",
wantCount: 8,
wantSizeEach: 16 * 1024,
},
{
name: "4 4k format",
buffers: "4 4k",
wantCount: 4,
wantSizeEach: 4 * 1024,
},
{
name: "2 1m format",
buffers: "2 1m",
wantCount: 2,
wantSizeEach: 1024 * 1024,
},
{
name: "bytes without unit",
buffers: "4 8192",
wantCount: 4,
wantSizeEach: 8192,
},
{
name: "uppercase K",
buffers: "8 16K",
wantCount: 8,
wantSizeEach: 16 * 1024,
},
{
name: "invalid format",
buffers: "invalid",
wantCount: 0,
wantSizeEach: 0,
},
{
name: "missing size",
buffers: "8",
wantCount: 0,
wantSizeEach: 0,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
cfg := &ProxyBufferingConfig{
Buffers: tt.buffers,
BufferSize: tt.bufferSize,
}
cfg.ParseBuffers()
if cfg.BufferCount != tt.wantCount {
t.Errorf("BufferCount = %d, want %d", cfg.BufferCount, tt.wantCount)
}
if cfg.BufferSizeEach != tt.wantSizeEach {
t.Errorf("BufferSizeEach = %d, want %d", cfg.BufferSizeEach, tt.wantSizeEach)
}
})
}
}
func TestLoad_Include(t *testing.T) {
t.Run("append servers from include", func(t *testing.T) {
tmpDir := t.TempDir()
mainCfg := `
servers:
- listen: ":8080"
name: main
include:
- path: "conf.d/*.yaml"
`
incCfg := `
servers:
- listen: ":9090"
name: included
`
confDir := filepath.Join(tmpDir, "conf.d")
if err := os.MkdirAll(confDir, 0o755); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "config.yaml"), []byte(mainCfg), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(confDir, "extra.yaml"), []byte(incCfg), 0o644); err != nil {
t.Fatal(err)
}
cfg, err := Load(filepath.Join(tmpDir, "config.yaml"))
if err != nil {
t.Fatalf("Load() failed: %v", err)
}
if len(cfg.Servers) != 2 {
t.Fatalf("expected 2 servers, got %d", len(cfg.Servers))
}
if cfg.Servers[0].Name != "main" {
t.Errorf("Servers[0].Name = %q, want %q", cfg.Servers[0].Name, "main")
}
if cfg.Servers[1].Name != "included" {
t.Errorf("Servers[1].Name = %q, want %q", cfg.Servers[1].Name, "included")
}
})
t.Run("merge variables with main priority", func(t *testing.T) {
tmpDir := t.TempDir()
mainCfg := `
servers:
- listen: ":8080"
variables:
set:
app: lolly
env: production
include:
- path: "extra.yaml"
`
incCfg := `
servers:
- listen: ":9090"
variables:
set:
app: other
debug: "true"
`
if err := os.WriteFile(filepath.Join(tmpDir, "config.yaml"), []byte(mainCfg), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "extra.yaml"), []byte(incCfg), 0o644); err != nil {
t.Fatal(err)
}
cfg, err := Load(filepath.Join(tmpDir, "config.yaml"))
if err != nil {
t.Fatalf("Load() failed: %v", err)
}
if cfg.Variables.Set["app"] != "lolly" {
t.Errorf("app = %q, want %q (main should win)", cfg.Variables.Set["app"], "lolly")
}
if cfg.Variables.Set["debug"] != "true" {
t.Errorf("debug = %q, want %q (included should fill missing)", cfg.Variables.Set["debug"], "true")
}
if cfg.Variables.Set["env"] != "production" {
t.Errorf("env = %q, want %q", cfg.Variables.Set["env"], "production")
}
})
t.Run("no matches returns error", func(t *testing.T) {
tmpDir := t.TempDir()
mainCfg := `
servers:
- listen: ":8080"
include:
- path: "nonexistent/*.yaml"
`
if err := os.WriteFile(filepath.Join(tmpDir, "config.yaml"), []byte(mainCfg), 0o644); err != nil {
t.Fatal(err)
}
_, err := Load(filepath.Join(tmpDir, "config.yaml"))
if err == nil {
t.Error("expected error for non-matching glob pattern")
}
})
t.Run("circular include detected immediately", func(t *testing.T) {
tmpDir := t.TempDir()
cfg1 := `
servers:
- listen: ":8080"
include:
- path: "b.yaml"
`
cfg2 := `
servers:
- listen: ":9090"
include:
- path: "a.yaml"
`
if err := os.WriteFile(filepath.Join(tmpDir, "a.yaml"), []byte(cfg1), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "b.yaml"), []byte(cfg2), 0o644); err != nil {
t.Fatal(err)
}
_, err := Load(filepath.Join(tmpDir, "a.yaml"))
if err == nil {
t.Error("expected error for circular include")
}
if !strings.Contains(err.Error(), "循环引入") {
t.Errorf("error should mention circular include, got: %v", err)
}
})
t.Run("self include detected", func(t *testing.T) {
tmpDir := t.TempDir()
cfg := `
servers:
- listen: ":8080"
include:
- path: "a.yaml"
`
if err := os.WriteFile(filepath.Join(tmpDir, "a.yaml"), []byte(cfg), 0o644); err != nil {
t.Fatal(err)
}
_, err := Load(filepath.Join(tmpDir, "a.yaml"))
if err == nil {
t.Error("expected error for self include")
}
})
t.Run("diamond include works", func(t *testing.T) {
tmpDir := t.TempDir()
mainCfg := `
servers:
- listen: ":8080"
include:
- path: "b.yaml"
- path: "c.yaml"
`
bCfg := `
servers:
- listen: ":9090"
include:
- path: "shared.yaml"
`
cCfg := `
servers:
- listen: ":9091"
include:
- path: "shared.yaml"
`
sharedCfg := `
variables:
set:
shared: value
`
if err := os.WriteFile(filepath.Join(tmpDir, "config.yaml"), []byte(mainCfg), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "b.yaml"), []byte(bCfg), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "c.yaml"), []byte(cCfg), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "shared.yaml"), []byte(sharedCfg), 0o644); err != nil {
t.Fatal(err)
}
cfg, err := Load(filepath.Join(tmpDir, "config.yaml"))
if err != nil {
t.Fatalf("diamond include should work: %v", err)
}
if len(cfg.Servers) != 3 {
t.Errorf("expected 3 servers, got %d", len(cfg.Servers))
}
if cfg.Variables.Set["shared"] != "value" {
t.Error("shared variable should be merged")
}
})
t.Run("empty include list is no-op", func(t *testing.T) {
tmpDir := t.TempDir()
mainCfg := `
servers:
- listen: ":8080"
`
if err := os.WriteFile(filepath.Join(tmpDir, "config.yaml"), []byte(mainCfg), 0o644); err != nil {
t.Fatal(err)
}
cfg, err := Load(filepath.Join(tmpDir, "config.yaml"))
if err != nil {
t.Fatalf("Load() failed: %v", err)
}
if len(cfg.Servers) != 1 {
t.Errorf("expected 1 server, got %d", len(cfg.Servers))
}
})
t.Run("append stream from include", func(t *testing.T) {
tmpDir := t.TempDir()
mainCfg := `
servers:
- listen: ":8080"
stream:
- listen: ":5432"
protocol: tcp
upstream:
targets:
- addr: "127.0.0.1:9000"
include:
- path: "stream.yaml"
`
incCfg := `
stream:
- listen: ":5433"
protocol: udp
upstream:
targets:
- addr: "127.0.0.1:9001"
`
if err := os.WriteFile(filepath.Join(tmpDir, "config.yaml"), []byte(mainCfg), 0o644); err != nil {
t.Fatal(err)
}
if err := os.WriteFile(filepath.Join(tmpDir, "stream.yaml"), []byte(incCfg), 0o644); err != nil {
t.Fatal(err)
}
cfg, err := Load(filepath.Join(tmpDir, "config.yaml"))
if err != nil {
t.Fatalf("Load() failed: %v", err)
}
if len(cfg.Stream) != 2 {
t.Fatalf("expected 2 streams, got %d", len(cfg.Stream))
}
})
}