mirror of
https://github.com/shadow1ng/fscan.git
synced 2026-02-10 02:39:18 +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)
198 lines
5.1 KiB
Go
198 lines
5.1 KiB
Go
package lib
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"encoding/hex"
|
|
"net/url"
|
|
|
|
"github.com/google/cel-go/checker/decls"
|
|
"github.com/google/cel-go/common/types"
|
|
"github.com/google/cel-go/common/types/ref"
|
|
"github.com/google/cel-go/interpreter/functions"
|
|
exprpb "google.golang.org/genproto/googleapis/api/expr/v1alpha1"
|
|
)
|
|
|
|
// registerEncodingDeclarations 注册编码相关的CEL函数声明
|
|
func registerEncodingDeclarations() []*exprpb.Decl {
|
|
return []*exprpb.Decl{
|
|
// base64
|
|
decls.NewFunction("base64",
|
|
decls.NewOverload("base64_string",
|
|
[]*exprpb.Type{decls.String},
|
|
decls.String)),
|
|
decls.NewFunction("base64",
|
|
decls.NewOverload("base64_bytes",
|
|
[]*exprpb.Type{decls.Bytes},
|
|
decls.String)),
|
|
|
|
// base64Decode
|
|
decls.NewFunction("base64Decode",
|
|
decls.NewOverload("base64Decode_string",
|
|
[]*exprpb.Type{decls.String},
|
|
decls.String)),
|
|
decls.NewFunction("base64Decode",
|
|
decls.NewOverload("base64Decode_bytes",
|
|
[]*exprpb.Type{decls.Bytes},
|
|
decls.String)),
|
|
|
|
// urlencode
|
|
decls.NewFunction("urlencode",
|
|
decls.NewOverload("urlencode_string",
|
|
[]*exprpb.Type{decls.String},
|
|
decls.String)),
|
|
decls.NewFunction("urlencode",
|
|
decls.NewOverload("urlencode_bytes",
|
|
[]*exprpb.Type{decls.Bytes},
|
|
decls.String)),
|
|
|
|
// urldecode
|
|
decls.NewFunction("urldecode",
|
|
decls.NewOverload("urldecode_string",
|
|
[]*exprpb.Type{decls.String},
|
|
decls.String)),
|
|
decls.NewFunction("urldecode",
|
|
decls.NewOverload("urldecode_bytes",
|
|
[]*exprpb.Type{decls.Bytes},
|
|
decls.String)),
|
|
|
|
// hexdecode
|
|
decls.NewFunction("hexdecode",
|
|
decls.NewInstanceOverload("hexdecode",
|
|
[]*exprpb.Type{decls.String},
|
|
decls.Bytes)),
|
|
}
|
|
}
|
|
|
|
// registerEncodingImplementations 注册编码相关的CEL函数实现
|
|
func registerEncodingImplementations() []*functions.Overload {
|
|
return []*functions.Overload{
|
|
// base64_string
|
|
{
|
|
Operator: "base64_string",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.String)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to base64_string", value.Type())
|
|
}
|
|
return types.String(base64.StdEncoding.EncodeToString([]byte(v)))
|
|
},
|
|
},
|
|
|
|
// base64_bytes
|
|
{
|
|
Operator: "base64_bytes",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.Bytes)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to base64_bytes", value.Type())
|
|
}
|
|
return types.String(base64.StdEncoding.EncodeToString(v))
|
|
},
|
|
},
|
|
|
|
// base64Decode_string
|
|
{
|
|
Operator: "base64Decode_string",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.String)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to base64Decode_string", value.Type())
|
|
}
|
|
decodeBytes, err := base64.StdEncoding.DecodeString(string(v))
|
|
if err != nil {
|
|
return types.NewErr("%v", err)
|
|
}
|
|
return types.String(decodeBytes)
|
|
},
|
|
},
|
|
|
|
// base64Decode_bytes
|
|
{
|
|
Operator: "base64Decode_bytes",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.Bytes)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to base64Decode_bytes", value.Type())
|
|
}
|
|
decodeBytes, err := base64.StdEncoding.DecodeString(string(v))
|
|
if err != nil {
|
|
return types.NewErr("%v", err)
|
|
}
|
|
return types.String(decodeBytes)
|
|
},
|
|
},
|
|
|
|
// urlencode_string
|
|
{
|
|
Operator: "urlencode_string",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.String)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to urlencode_string", value.Type())
|
|
}
|
|
return types.String(url.QueryEscape(string(v)))
|
|
},
|
|
},
|
|
|
|
// urlencode_bytes
|
|
{
|
|
Operator: "urlencode_bytes",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.Bytes)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to urlencode_bytes", value.Type())
|
|
}
|
|
return types.String(url.QueryEscape(string(v)))
|
|
},
|
|
},
|
|
|
|
// urldecode_string
|
|
{
|
|
Operator: "urldecode_string",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.String)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to urldecode_string", value.Type())
|
|
}
|
|
decodeString, err := url.QueryUnescape(string(v))
|
|
if err != nil {
|
|
return types.NewErr("%v", err)
|
|
}
|
|
return types.String(decodeString)
|
|
},
|
|
},
|
|
|
|
// urldecode_bytes
|
|
{
|
|
Operator: "urldecode_bytes",
|
|
Unary: func(value ref.Val) ref.Val {
|
|
v, ok := value.(types.Bytes)
|
|
if !ok {
|
|
return types.ValOrErr(value, "unexpected type '%v' passed to urldecode_bytes", value.Type())
|
|
}
|
|
decodeString, err := url.QueryUnescape(string(v))
|
|
if err != nil {
|
|
return types.NewErr("%v", err)
|
|
}
|
|
return types.String(decodeString)
|
|
},
|
|
},
|
|
|
|
// hexdecode
|
|
{
|
|
Operator: "hexdecode",
|
|
Unary: func(lhs ref.Val) ref.Val {
|
|
v1, ok := lhs.(types.String)
|
|
if !ok {
|
|
return types.ValOrErr(lhs, "unexpected type '%v' passed to hexdecode", lhs.Type())
|
|
}
|
|
out, err := hex.DecodeString(string(v1))
|
|
if err != nil {
|
|
return types.ValOrErr(lhs, "hexdecode error: %v", err)
|
|
}
|
|
return types.Bytes(out)
|
|
},
|
|
},
|
|
}
|
|
}
|