fscan/WebScan/lib/Client.go

319 lines
8.0 KiB
Go
Raw Normal View History

2021-05-31 10:03:01 +08:00
package lib
import (
2022-05-07 23:46:22 +08:00
"context"
2021-05-31 10:03:01 +08:00
"crypto/tls"
2023-05-05 21:02:31 +08:00
"embed"
2022-05-07 23:46:22 +08:00
"errors"
2023-05-05 21:02:31 +08:00
"fmt"
2024-12-18 22:00:18 +08:00
"github.com/shadow1ng/fscan/Common"
2022-05-07 23:46:22 +08:00
"golang.org/x/net/proxy"
2023-05-05 21:02:31 +08:00
"gopkg.in/yaml.v2"
2021-05-31 10:03:01 +08:00
"net"
"net/http"
"net/url"
2023-12-25 17:57:28 +08:00
"os"
2021-05-31 10:03:01 +08:00
"strings"
"time"
)
// 全局HTTP客户端变量
2021-05-31 10:03:01 +08:00
var (
Client *http.Client // 标准HTTP客户端
ClientNoRedirect *http.Client // 不自动跟随重定向的HTTP客户端
dialTimout = 5 * time.Second // 连接超时时间
keepAlive = 5 * time.Second // 连接保持时间
2021-05-31 10:03:01 +08:00
)
// Inithttp 初始化HTTP客户端配置
2023-11-15 10:40:17 +08:00
func Inithttp() {
// 设置默认并发数
2024-12-18 22:00:18 +08:00
if Common.PocNum == 0 {
Common.PocNum = 20
2023-11-15 10:40:17 +08:00
}
// 设置默认超时时间
2024-12-18 22:00:18 +08:00
if Common.WebTimeout == 0 {
Common.WebTimeout = 5
2023-11-15 10:40:17 +08:00
}
// 初始化HTTP客户端
2024-12-20 03:46:09 +08:00
err := InitHttpClient(Common.PocNum, Common.HttpProxy, time.Duration(Common.WebTimeout)*time.Second)
2021-05-31 10:03:01 +08:00
if err != nil {
2023-12-25 17:57:28 +08:00
panic(err)
2021-05-31 10:03:01 +08:00
}
}
// InitHttpClient 创建HTTP客户端
2021-05-31 10:03:01 +08:00
func InitHttpClient(ThreadsNum int, DownProxy string, Timeout time.Duration) error {
2022-05-07 23:46:22 +08:00
type DialContext = func(ctx context.Context, network, addr string) (net.Conn, error)
// 配置基础连接参数
2021-05-31 10:03:01 +08:00
dialer := &net.Dialer{
Timeout: dialTimout,
KeepAlive: keepAlive,
}
// 配置Transport参数
2021-05-31 10:03:01 +08:00
tr := &http.Transport{
2022-05-09 12:08:29 +08:00
DialContext: dialer.DialContext,
2022-01-07 17:45:13 +08:00
MaxConnsPerHost: 5,
2021-05-31 10:03:01 +08:00
MaxIdleConns: 0,
2022-04-27 11:48:22 +08:00
MaxIdleConnsPerHost: ThreadsNum * 2,
2021-05-31 10:03:01 +08:00
IdleConnTimeout: keepAlive,
2023-11-13 09:45:39 +08:00
TLSClientConfig: &tls.Config{MinVersion: tls.VersionTLS10, InsecureSkipVerify: true},
2021-05-31 10:03:01 +08:00
TLSHandshakeTimeout: 5 * time.Second,
2022-01-07 17:45:13 +08:00
DisableKeepAlives: false,
2021-05-31 10:03:01 +08:00
}
2022-05-09 12:08:29 +08:00
// 配置Socks5代理
2024-12-18 22:00:18 +08:00
if Common.Socks5Proxy != "" {
2024-12-18 23:41:01 +08:00
dialSocksProxy, err := Common.Socks5Dialer(dialer)
2022-05-09 12:08:29 +08:00
if err != nil {
return err
}
if contextDialer, ok := dialSocksProxy.(proxy.ContextDialer); ok {
tr.DialContext = contextDialer.DialContext
} else {
return errors.New("无法转换为DialContext类型")
2022-05-09 12:08:29 +08:00
}
2022-07-03 23:41:39 +08:00
} else if DownProxy != "" {
// 处理其他代理配置
2021-05-31 10:03:01 +08:00
if DownProxy == "1" {
DownProxy = "http://127.0.0.1:8080"
} else if DownProxy == "2" {
DownProxy = "socks5://127.0.0.1:1080"
2021-05-31 10:03:01 +08:00
} else if !strings.Contains(DownProxy, "://") {
DownProxy = "http://127.0.0.1:" + DownProxy
}
// 验证代理类型
2022-07-03 23:41:39 +08:00
if !strings.HasPrefix(DownProxy, "socks") && !strings.HasPrefix(DownProxy, "http") {
return errors.New("不支持的代理类型")
2022-05-09 12:08:29 +08:00
}
// 解析代理URL
2021-05-31 10:03:01 +08:00
u, err := url.Parse(DownProxy)
if err != nil {
return err
}
tr.Proxy = http.ProxyURL(u)
}
// 创建标准HTTP客户端
2021-05-31 10:03:01 +08:00
Client = &http.Client{
Transport: tr,
Timeout: Timeout,
}
// 创建不跟随重定向的HTTP客户端
2021-05-31 10:03:01 +08:00
ClientNoRedirect = &http.Client{
2021-06-09 11:16:23 +08:00
Transport: tr,
Timeout: Timeout,
CheckRedirect: func(req *http.Request, via []*http.Request) error { return http.ErrUseLastResponse },
2021-05-31 10:03:01 +08:00
}
2021-05-31 10:03:01 +08:00
return nil
}
2023-05-05 21:02:31 +08:00
// Poc 定义漏洞检测配置结构
2023-05-05 21:02:31 +08:00
type Poc struct {
Name string `yaml:"name"` // POC名称
Set StrMap `yaml:"set"` // 单值配置映射
Sets ListMap `yaml:"sets"` // 列表值配置映射
Rules []Rules `yaml:"rules"` // 检测规则列表
Groups RuleMap `yaml:"groups"` // 规则组映射
Detail Detail `yaml:"detail"` // 漏洞详情
2023-05-05 21:02:31 +08:00
}
// MapSlice 用于解析YAML的通用映射类型
2023-05-05 21:02:31 +08:00
type MapSlice = yaml.MapSlice
// 自定义映射类型
type (
StrMap []StrItem // 字符串键值对映射
ListMap []ListItem // 字符串键列表值映射
RuleMap []RuleItem // 字符串键规则列表映射
)
2023-05-05 21:02:31 +08:00
// 映射项结构定义
type (
// StrItem 字符串键值对
StrItem struct {
Key string // 键名
Value string // 值
}
2023-05-05 21:02:31 +08:00
// ListItem 字符串键列表值对
ListItem struct {
Key string // 键名
Value []string // 值列表
}
2023-05-05 21:02:31 +08:00
// RuleItem 字符串键规则列表对
RuleItem struct {
Key string // 键名
Value []Rules // 规则列表
}
)
2023-05-05 21:02:31 +08:00
// UnmarshalYAML 实现StrMap的YAML解析接口
2023-05-05 21:02:31 +08:00
func (r *StrMap) UnmarshalYAML(unmarshal func(interface{}) error) error {
// 临时使用MapSlice存储解析结果
2023-05-05 21:02:31 +08:00
var tmp yaml.MapSlice
if err := unmarshal(&tmp); err != nil {
return err
}
// 转换为StrMap结构
2023-05-05 21:02:31 +08:00
for _, one := range tmp {
key, value := one.Key.(string), one.Value.(string)
*r = append(*r, StrItem{key, value})
}
2023-05-05 21:02:31 +08:00
return nil
}
// UnmarshalYAML 实现RuleMap的YAML解析接口
// 参数:
// - unmarshal: YAML解析函数
//
// 返回:
// - error: 解析错误
2023-05-05 21:02:31 +08:00
func (r *RuleMap) UnmarshalYAML(unmarshal func(interface{}) error) error {
// 使用MapSlice保持键的顺序
2023-05-05 21:02:31 +08:00
var tmp1 yaml.MapSlice
if err := unmarshal(&tmp1); err != nil {
return err
}
// 解析规则内容
2023-05-05 21:02:31 +08:00
var tmp = make(map[string][]Rules)
if err := unmarshal(&tmp); err != nil {
return err
}
// 按顺序转换为RuleMap结构
2023-05-05 21:02:31 +08:00
for _, one := range tmp1 {
key := one.Key.(string)
value := tmp[key]
*r = append(*r, RuleItem{key, value})
}
return nil
}
// UnmarshalYAML 实现ListMap的YAML解析接口
// 参数:
// - unmarshal: YAML解析函数
//
// 返回:
// - error: 解析错误
2023-05-05 21:02:31 +08:00
func (r *ListMap) UnmarshalYAML(unmarshal func(interface{}) error) error {
// 解析YAML映射
2023-05-05 21:02:31 +08:00
var tmp yaml.MapSlice
if err := unmarshal(&tmp); err != nil {
return err
}
// 转换为ListMap结构
2023-05-05 21:02:31 +08:00
for _, one := range tmp {
key := one.Key.(string)
var value []string
// 将接口类型转换为字符串
2023-05-05 21:02:31 +08:00
for _, val := range one.Value.([]interface{}) {
v := fmt.Sprintf("%v", val)
value = append(value, v)
}
*r = append(*r, ListItem{key, value})
}
return nil
}
// Rules 定义POC检测规则结构
2023-05-05 21:02:31 +08:00
type Rules struct {
Method string `yaml:"method"` // HTTP请求方法
Path string `yaml:"path"` // 请求路径
Headers map[string]string `yaml:"headers"` // 请求头
Body string `yaml:"body"` // 请求体
Search string `yaml:"search"` // 搜索模式
FollowRedirects bool `yaml:"follow_redirects"` // 是否跟随重定向
Expression string `yaml:"expression"` // 匹配表达式
Continue bool `yaml:"continue"` // 是否继续执行
2023-05-05 21:02:31 +08:00
}
// Detail 定义POC详情结构
2023-05-05 21:02:31 +08:00
type Detail struct {
Author string `yaml:"author"` // POC作者
Links []string `yaml:"links"` // 相关链接
Description string `yaml:"description"` // POC描述
Version string `yaml:"version"` // POC版本
2023-05-05 21:02:31 +08:00
}
// LoadMultiPoc 加载多个POC文件
2023-05-05 21:02:31 +08:00
func LoadMultiPoc(Pocs embed.FS, pocname string) []*Poc {
var pocs []*Poc
// 遍历选中的POC文件
2023-05-05 21:02:31 +08:00
for _, f := range SelectPoc(Pocs, pocname) {
if p, err := LoadPoc(f, Pocs); err == nil {
pocs = append(pocs, p)
} else {
fmt.Printf("POC加载失败 %s: %v\n", f, err)
2023-05-05 21:02:31 +08:00
}
}
return pocs
}
// LoadPoc 从内嵌文件系统加载单个POC
2023-05-05 21:02:31 +08:00
func LoadPoc(fileName string, Pocs embed.FS) (*Poc, error) {
p := &Poc{}
// 读取POC文件内容
2023-05-05 21:02:31 +08:00
yamlFile, err := Pocs.ReadFile("pocs/" + fileName)
if err != nil {
fmt.Printf("POC文件读取失败 %s: %v\n", fileName, err)
2023-05-05 21:02:31 +08:00
return nil, err
}
// 解析YAML内容
2023-05-05 21:02:31 +08:00
err = yaml.Unmarshal(yamlFile, p)
if err != nil {
fmt.Printf("POC解析失败 %s: %v\n", fileName, err)
2023-05-05 21:02:31 +08:00
return nil, err
}
return p, err
}
// SelectPoc 根据名称关键字选择POC文件
2023-05-05 21:02:31 +08:00
func SelectPoc(Pocs embed.FS, pocname string) []string {
entries, err := Pocs.ReadDir("pocs")
if err != nil {
fmt.Printf("读取POC目录失败: %v\n", err)
2023-05-05 21:02:31 +08:00
}
2023-05-05 21:02:31 +08:00
var foundFiles []string
// 查找匹配关键字的POC文件
2023-05-05 21:02:31 +08:00
for _, entry := range entries {
if strings.Contains(entry.Name(), pocname) {
foundFiles = append(foundFiles, entry.Name())
}
}
return foundFiles
}
// LoadPocbyPath 从文件系统路径加载POC
2023-05-05 21:02:31 +08:00
func LoadPocbyPath(fileName string) (*Poc, error) {
p := &Poc{}
// 读取POC文件内容
2023-12-25 17:57:28 +08:00
data, err := os.ReadFile(fileName)
2023-05-05 21:02:31 +08:00
if err != nil {
fmt.Printf("POC文件读取失败 %s: %v\n", fileName, err)
2023-05-05 21:02:31 +08:00
return nil, err
}
// 解析YAML内容
2023-05-05 21:02:31 +08:00
err = yaml.Unmarshal(data, p)
if err != nil {
fmt.Printf("POC解析失败 %s: %v\n", fileName, err)
2023-05-05 21:02:31 +08:00
return nil, err
}
return p, err
}