Files
aqi/logger/zap_test.go
2025-09-04 14:20:51 +08:00

252 lines
8.8 KiB
Go
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

package logger
import (
"fmt"
"log"
"regexp"
"strconv"
"strings"
"testing"
)
func TestZap(t *testing.T) {
testCases := []struct {
name string
logFilters map[string]string
messages []string
description string
}{
{
name: "基础截断测试",
logFilters: map[string]string{
"article.list": "data:116",
"res.uploadImg": "params:64",
},
messages: []string{
`{"action":"res.uploadImg","params":""}`,
`{"code":0,"action":"article.list","id":"92","data":[{"id":83,"status":1,"parentId":0,"position":"ai-diagnosis","icon":"https://cheguanjia-test.oss-cn-chengdu.aliyuncs.com/files/2011/cce5e12eb9e403b.png?rid=4384\u0026mime_type=image/png","searchIcon":"","tagType":2,"orderSort":5,"name":"AI诊断","questions":{"data":[],"rule":"random","random":5},"cards":{"steps":[]},"notes":"","createdTime":"2025-07-11T11:49:24+08:00","updatedTime":"2025-07-11T13:50:04+08:00"}]}`,
},
description: "测试基本的字符串和数组截断功能",
},
{
name: "中文字符截断测试",
logFilters: map[string]string{
"user.info": "message:20",
},
messages: []string{
`{"action":"user.info","message":"这是一个包含中文字符的很长的消息内容,用来测试中文字符的正确截断功能,确保不会从中文字符中间截断"}`,
`{"action":"user.info","message":"Hello World! 这里混合了英文和中文字符,测试混合字符的截断效果"}`,
},
description: "测试中文字符安全截断,确保不会从中文字符中间截断",
},
{
name: "边界条件测试",
logFilters: map[string]string{
"test.boundary": "content:0", // 长度为0
"test.small": "content:5", // 很小的长度
"test.exact": "content:10", // 刚好等于内容长度
},
messages: []string{
`{"action":"test.boundary","content":"任何内容都会被截断"}`,
`{"action":"test.small","content":"短内容测试"}`,
`{"action":"test.exact","content":"1234567890"}`, // 刚好10个字符
`{"action":"test.exact","content":"12345678901"}`, // 11个字符应该被截断
},
description: "测试边界条件长度为0、很小长度、刚好等于内容长度等情况",
},
{
name: "特殊字符测试",
logFilters: map[string]string{
"special.chars": "data:30",
},
messages: []string{
`{"action":"special.chars","data":"包含特殊字符: \"引号\" \n换行 \t制表符 \\反斜杠 和其他特殊符号 @#$%^&*()"}`,
`{"action":"special.chars","data":"JSON转义字符测试: \u0026 \u003c \u003e 以及 emoji 😀🎉🚀"}`,
`{"action":"special.chars","data":"URL编码字符: %20 %3A %2F %3F 和HTML实体: & < >"}`,
},
description: "测试包含特殊字符、转义字符、emoji等的内容截断",
},
{
name: "复杂JSON结构测试",
logFilters: map[string]string{
"complex.nested": "payload:50",
"complex.array": "items:40",
},
messages: []string{
`{"action":"complex.nested","payload":{"user":{"id":123,"name":"张三","profile":{"age":25,"city":"北京","hobbies":["读书","游泳","编程"]}},"timestamp":"2024-01-01T00:00:00Z"}}`,
`{"action":"complex.array","items":[{"id":1,"name":"项目A","tags":["重要","紧急"]},{"id":2,"name":"项目B","tags":["普通"]},{"id":3,"name":"项目C","tags":["低优先级"]}]}`,
},
description: "测试复杂嵌套JSON结构的截断处理",
},
{
name: "无匹配规则测试",
logFilters: map[string]string{
"match.rule": "content:20",
},
messages: []string{
`{"action":"no.match","content":"这条消息不应该被截断因为action不匹配任何规则"}`,
`{"action":"match.rule","other_field":"这个字段不会被截断,因为字段名不匹配"}`,
`{"different_action":"match.rule","content":"action字段名不同不会匹配规则"}`,
},
description: "测试不匹配过滤规则的消息应该保持原样",
},
{
name: "空值和null测试",
logFilters: map[string]string{
"empty.test": "value:10",
},
messages: []string{
`{"action":"empty.test","value":""}`,
`{"action":"empty.test","value":null}`,
`{"action":"empty.test","value":" "}`, // 只有空格
`{"action":"empty.test"}`, // 缺少字段
},
description: "测试空字符串、null值、纯空格和缺失字段的处理",
},
{
name: "多字段规则测试",
logFilters: map[string]string{
"multi.field1": "title:15",
"multi.field2": "description:25",
},
messages: []string{
`{"action":"multi.field1","title":"这是一个很长的标题内容需要被截断","description":"这个描述不会被截断因为规则不匹配"}`,
`{"action":"multi.field2","title":"这个标题不会被截断","description":"这是一个更长的描述内容,包含更多的详细信息和说明文字,应该被截断"}`,
},
description: "测试不同action对应不同字段的截断规则",
},
{
name: "性能压力测试",
logFilters: map[string]string{
"perf.test": "large_data:100",
},
messages: []string{
`{"action":"perf.test","large_data":"` + strings.Repeat("这是重复的内容用于测试大数据量的截断性能。", 100) + `"}`,
},
description: "测试大数据量的截断性能",
},
{
name: "格式异常测试",
logFilters: map[string]string{
"format.test": "data:20",
},
messages: []string{
`{"action":"format.test","data":"正常数据"}`, // 正常JSON
`{"action":"format.test","data":"包含\"引号\"的数据"}`, // 包含转义引号
`{"action":"format.test","data":"包含,逗号的数据"}`, // 包含逗号
`{"action":"format.test","data":"包含}花括号的数据"}`, // 包含花括号
`{"action":"format.test","data":"包含]方括号的数据"}`, // 包含方括号
},
description: "测试包含JSON特殊字符的数据处理",
},
{
name: "负数和零长度测试",
logFilters: map[string]string{
"negative.test": "data:-5", // 负数长度
"zero.test": "data:0", // 零长度
},
messages: []string{
`{"action":"negative.test","data":"负数长度测试数据"}`,
`{"action":"zero.test","data":"零长度测试数据"}`,
},
description: "测试负数和零长度的边界情况处理",
},
}
for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) {
log.Printf("\n=== %s ===\n%s\n", tc.name, tc.description)
// 构建过滤规则
filters := make([]FilterRule, 0, len(tc.logFilters))
for actionPattern, fieldRule := range tc.logFilters {
// 解析字段路径和长度格式field.path:maxLen
ruleParts := strings.SplitN(fieldRule, ":", 2)
if len(ruleParts) != 2 {
continue
}
fieldPath := strings.TrimSpace(ruleParts[0])
maxLen, _ := strconv.Atoi(strings.TrimSpace(ruleParts[1]))
if maxLen < 0 {
maxLen = 0
}
patternStr := fmt.Sprintf(`"action":"%s"`, regexp.QuoteMeta(actionPattern))
pattern := regexp.MustCompile(patternStr)
filters = append(filters, FilterRule{
Action: actionPattern,
Field: fieldPath,
MaxLen: maxLen,
Pattern: pattern,
})
}
fs := fileStyleEncoder{}
// 处理每条消息
for i, msg := range tc.messages {
originalLen := len(msg)
processedMsg := fs.processMessage(msg, filters)
processedLen := len(processedMsg)
log.Printf("消息 %d (原长度: %d, 处理后长度: %d):\n", i+1, originalLen, processedLen)
log.Printf("处理后: %s\n", processedMsg)
// 验证截断是否正确
if processedLen < originalLen {
log.Printf("✓ 消息被成功截断\n")
} else if processedLen == originalLen {
log.Printf("- 消息长度未变化(可能未匹配规则或无需截断)\n")
}
log.Println()
}
})
}
}
// 辅助函数:生成大量测试数据
func generateLargeTestData(size int) string {
return strings.Repeat("测试数据", size)
}
// 基准测试:测试截断性能
func BenchmarkProcessMessage(b *testing.B) {
logFilters := map[string]string{
"perf.test": "data:100",
}
filters := make([]FilterRule, 0, len(logFilters))
for actionPattern, fieldRule := range logFilters {
ruleParts := strings.SplitN(fieldRule, ":", 2)
if len(ruleParts) != 2 {
continue
}
fieldPath := strings.TrimSpace(ruleParts[0])
maxLen, _ := strconv.Atoi(strings.TrimSpace(ruleParts[1]))
if maxLen < 0 {
maxLen = 0
}
patternStr := fmt.Sprintf(`"action":"%s"`, regexp.QuoteMeta(actionPattern))
pattern := regexp.MustCompile(patternStr)
filters = append(filters, FilterRule{
Action: actionPattern,
Field: fieldPath,
MaxLen: maxLen,
Pattern: pattern,
})
}
fs := fileStyleEncoder{}
testMsg := `{"action":"perf.test","data":"` + generateLargeTestData(1000) + `"}`
b.ResetTimer()
for i := 0; i < b.N; i++ {
_ = fs.processMessage(testMsg, filters)
}
}