mirror of
https://github.com/shadow1ng/fscan.git
synced 2026-02-09 10:19:19 +08:00
## 架构重构
- 全局变量消除,迁移至 Config/State 对象
- SMB 插件融合(smb/smb2/smbghost/smbinfo)
- 服务探测重构,实现 Nmap 风格 fallback 机制
- 输出系统重构,TXT 实时刷盘 + 双写机制
- i18n 框架升级至 go-i18n
## 性能优化
- 正则表达式预编译
- 内存优化 map[string]struct{}
- 并发指纹匹配
- SOCKS5 连接复用
- 滑动窗口调度 + 自适应线程池
## 新功能
- Web 管理界面
- 多格式 POC 适配(xray/afrog)
- 增强指纹库(3139条)
- Favicon hash 指纹识别
- 插件选择性编译(Build Tags)
- fscan-lab 靶场环境
- 默认端口扩展(62→133)
## 构建系统
- 添加 no_local tag 支持排除本地插件
- 多版本构建:fscan/fscan-nolocal/fscan-web
- CI 添加 snapshot 模式支持仅测试构建
## Bug 修复
- 修复 120+ 个问题,包括 RDP panic、批量扫描漏报、
JSON 输出格式、Redis 检测、Context 超时等
## 测试增强
- 单元测试覆盖率 74-100%
- 并发安全测试
- 集成测试(Web/端口/服务/SSH/ICMP)
109 lines
3.0 KiB
Go
109 lines
3.0 KiB
Go
//go:build debug
|
|
// +build debug
|
|
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"runtime"
|
|
"runtime/pprof"
|
|
"runtime/trace"
|
|
)
|
|
|
|
var (
|
|
cpuProfile *os.File
|
|
memProfile *os.File
|
|
traceFile *os.File
|
|
profilesPath = "./profiles" // 性能分析文件保存目录
|
|
)
|
|
|
|
func startPprof() {
|
|
// 创建 profiles 目录
|
|
if err := os.MkdirAll(profilesPath, 0755); err != nil {
|
|
fmt.Printf("[DEBUG] 创建 profiles 目录失败: %v\n", err)
|
|
return
|
|
}
|
|
|
|
// 启动 CPU profiling
|
|
var err error
|
|
cpuProfile, err = os.Create(profilesPath + "/cpu.prof")
|
|
if err != nil {
|
|
fmt.Printf("[DEBUG] 创建 CPU profile 失败: %v\n", err)
|
|
} else {
|
|
if err := pprof.StartCPUProfile(cpuProfile); err != nil {
|
|
fmt.Printf("[DEBUG] 启动 CPU profile 失败: %v\n", err)
|
|
cpuProfile.Close()
|
|
cpuProfile = nil
|
|
} else {
|
|
fmt.Printf("[DEBUG] CPU profiling 已启动 -> %s/cpu.prof\n", profilesPath)
|
|
}
|
|
}
|
|
|
|
// 启动 trace
|
|
traceFile, err = os.Create(profilesPath + "/trace.out")
|
|
if err != nil {
|
|
fmt.Printf("[DEBUG] 创建 trace 文件失败: %v\n", err)
|
|
} else {
|
|
if err := trace.Start(traceFile); err != nil {
|
|
fmt.Printf("[DEBUG] 启动 trace 失败: %v\n", err)
|
|
traceFile.Close()
|
|
traceFile = nil
|
|
} else {
|
|
fmt.Printf("[DEBUG] Execution trace 已启动 -> %s/trace.out\n", profilesPath)
|
|
}
|
|
}
|
|
|
|
fmt.Printf("[DEBUG] 性能分析已启动,程序结束时自动保存到 %s/\n", profilesPath)
|
|
}
|
|
|
|
func stopPprof() {
|
|
// 停止 CPU profiling
|
|
if cpuProfile != nil {
|
|
pprof.StopCPUProfile()
|
|
cpuProfile.Close()
|
|
fmt.Printf("[DEBUG] CPU profile 已保存\n")
|
|
}
|
|
|
|
// 停止 trace
|
|
if traceFile != nil {
|
|
trace.Stop()
|
|
traceFile.Close()
|
|
fmt.Printf("[DEBUG] Trace 已保存\n")
|
|
}
|
|
|
|
// 写入内存 profile
|
|
memProfile, err := os.Create(profilesPath + "/mem.prof")
|
|
if err != nil {
|
|
fmt.Printf("[DEBUG] 创建内存 profile 失败: %v\n", err)
|
|
} else {
|
|
runtime.GC() // 先执行 GC
|
|
if err := pprof.WriteHeapProfile(memProfile); err != nil {
|
|
fmt.Printf("[DEBUG] 写入内存 profile 失败: %v\n", err)
|
|
} else {
|
|
fmt.Printf("[DEBUG] 内存 profile 已保存 -> %s/mem.prof\n", profilesPath)
|
|
}
|
|
memProfile.Close()
|
|
}
|
|
|
|
// 写入 goroutine profile
|
|
goroutineProfile, err := os.Create(profilesPath + "/goroutine.prof")
|
|
if err != nil {
|
|
fmt.Printf("[DEBUG] 创建 goroutine profile 失败: %v\n", err)
|
|
} else {
|
|
if err := pprof.Lookup("goroutine").WriteTo(goroutineProfile, 0); err != nil {
|
|
fmt.Printf("[DEBUG] 写入 goroutine profile 失败: %v\n", err)
|
|
} else {
|
|
fmt.Printf("[DEBUG] Goroutine profile 已保存 -> %s/goroutine.prof\n", profilesPath)
|
|
}
|
|
goroutineProfile.Close()
|
|
}
|
|
|
|
fmt.Printf("\n[DEBUG] 所有性能分析文件已保存到 %s/\n", profilesPath)
|
|
fmt.Printf("[DEBUG] 查看方法:\n")
|
|
fmt.Printf(" CPU 火焰图: go tool pprof -http=:8081 %s/cpu.prof\n", profilesPath)
|
|
fmt.Printf(" 内存火焰图: go tool pprof -http=:8081 %s/mem.prof\n", profilesPath)
|
|
fmt.Printf(" 协程分析: go tool pprof -http=:8081 %s/goroutine.prof\n", profilesPath)
|
|
fmt.Printf(" 执行时间线: go tool trace %s/trace.out\n", profilesPath)
|
|
}
|