fscan/Plugins/SMB2.go

403 lines
9.2 KiB
Go
Raw Normal View History

package Plugins
import (
"fmt"
2024-12-18 22:00:18 +08:00
"github.com/shadow1ng/fscan/Common"
"net"
"os"
"strings"
2024-12-31 20:25:54 +08:00
"sync"
"time"
"github.com/hirochachacha/go-smb2"
)
2024-12-18 15:19:47 +08:00
// SmbScan2 执行SMB2服务的认证扫描支持密码和哈希两种认证方式
2024-12-19 16:15:53 +08:00
func SmbScan2(info *Common.HostInfo) (tmperr error) {
2024-12-20 03:46:09 +08:00
if Common.DisableBrute {
return nil
}
2024-12-18 15:19:47 +08:00
// 使用哈希认证模式
2024-12-18 22:00:18 +08:00
if len(Common.HashBytes) > 0 {
2024-12-31 20:25:54 +08:00
return smbHashScan(info)
2024-12-18 15:19:47 +08:00
}
// 使用密码认证模式
2024-12-31 20:25:54 +08:00
return smbPasswordScan(info)
2024-12-18 15:19:47 +08:00
}
// smbHashScan 使用哈希进行认证扫描
2024-12-31 20:25:54 +08:00
func smbHashScan(info *Common.HostInfo) error {
maxRetries := Common.MaxRetries
threads := Common.BruteThreads
hasprint := false
// 创建任务通道
taskChan := make(chan struct {
user string
hash []byte
}, len(Common.Userdict["smb"])*len(Common.HashBytes))
resultChan := make(chan error, threads)
// 生成所有用户名和哈希组合任务
2024-12-18 22:00:18 +08:00
for _, user := range Common.Userdict["smb"] {
for _, hash := range Common.HashBytes {
2024-12-31 20:25:54 +08:00
taskChan <- struct {
user string
hash []byte
}{user, hash}
}
}
close(taskChan)
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
// 启动工作线程
var wg sync.WaitGroup
var hasPrintMutex sync.Mutex
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
for i := 0; i < threads; i++ {
wg.Add(1)
go func() {
defer wg.Done()
startTime := time.Now().Unix()
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
for task := range taskChan {
// 重试循环
for retryCount := 0; retryCount < maxRetries; retryCount++ {
// 检查是否超时
if time.Now().Unix()-startTime > int64(Common.Timeout) {
resultChan <- fmt.Errorf("扫描超时")
return
}
// 执行SMB2认证
done := make(chan struct {
success bool
err error
printed bool
})
go func(user string, hash []byte) {
hasPrintMutex.Lock()
currentHasPrint := hasprint
hasPrintMutex.Unlock()
success, err, printed := Smb2Con(info, user, "", hash, currentHasPrint)
if printed {
hasPrintMutex.Lock()
hasprint = true
hasPrintMutex.Unlock()
}
done <- struct {
success bool
err error
printed bool
}{success, err, printed}
}(task.user, task.hash)
// 等待结果或超时
select {
case result := <-done:
if result.success {
logSuccessfulAuth(info, task.user, "", task.hash)
resultChan <- nil
return
}
if result.err != nil {
logFailedAuth(info, task.user, "", task.hash, result.err)
// 检查是否需要重试
if retryErr := Common.CheckErrs(result.err); retryErr != nil {
if retryCount == maxRetries-1 {
resultChan <- result.err
return
}
continue // 继续重试
}
}
case <-time.After(time.Duration(Common.Timeout) * time.Second):
logFailedAuth(info, task.user, "", task.hash, fmt.Errorf("连接超时"))
}
break // 如果不需要重试,跳出重试循环
}
if len(Common.HashValue) > 0 {
break
}
2024-12-18 15:19:47 +08:00
}
2024-12-31 20:25:54 +08:00
resultChan <- nil
}()
}
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
// 等待所有线程完成
go func() {
wg.Wait()
close(resultChan)
}()
// 检查结果
for err := range resultChan {
if err != nil {
if retryErr := Common.CheckErrs(err); retryErr != nil {
return err
2024-08-29 09:50:32 +08:00
}
}
2024-12-18 15:19:47 +08:00
}
2024-12-31 20:25:54 +08:00
2024-12-18 15:19:47 +08:00
return nil
}
// smbPasswordScan 使用密码进行认证扫描
2024-12-31 20:25:54 +08:00
func smbPasswordScan(info *Common.HostInfo) error {
maxRetries := Common.MaxRetries
threads := Common.BruteThreads
hasprint := false
// 创建任务通道
taskChan := make(chan struct {
user string
pass string
}, len(Common.Userdict["smb"])*len(Common.Passwords))
resultChan := make(chan error, threads)
// 生成所有用户名密码组合任务
2024-12-18 22:00:18 +08:00
for _, user := range Common.Userdict["smb"] {
for _, pass := range Common.Passwords {
2024-12-18 15:19:47 +08:00
pass = strings.ReplaceAll(pass, "{user}", user)
2024-12-31 20:25:54 +08:00
taskChan <- struct {
user string
pass string
}{user, pass}
}
}
close(taskChan)
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
// 启动工作线程
var wg sync.WaitGroup
var hasPrintMutex sync.Mutex
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
for i := 0; i < threads; i++ {
wg.Add(1)
go func() {
defer wg.Done()
startTime := time.Now().Unix()
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
for task := range taskChan {
// 重试循环
for retryCount := 0; retryCount < maxRetries; retryCount++ {
// 检查是否超时
if time.Now().Unix()-startTime > int64(Common.Timeout) {
resultChan <- fmt.Errorf("扫描超时")
return
}
// 执行SMB2认证
done := make(chan struct {
success bool
err error
printed bool
})
go func(user, pass string) {
hasPrintMutex.Lock()
currentHasPrint := hasprint
hasPrintMutex.Unlock()
success, err, printed := Smb2Con(info, user, pass, []byte{}, currentHasPrint)
if printed {
hasPrintMutex.Lock()
hasprint = true
hasPrintMutex.Unlock()
}
done <- struct {
success bool
err error
printed bool
}{success, err, printed}
}(task.user, task.pass)
// 等待结果或超时
select {
case result := <-done:
if result.success {
logSuccessfulAuth(info, task.user, task.pass, []byte{})
resultChan <- nil
return
}
if result.err != nil {
logFailedAuth(info, task.user, task.pass, []byte{}, result.err)
// 检查是否需要重试
if retryErr := Common.CheckErrs(result.err); retryErr != nil {
if retryCount == maxRetries-1 {
resultChan <- result.err
return
}
continue // 继续重试
}
}
case <-time.After(time.Duration(Common.Timeout) * time.Second):
logFailedAuth(info, task.user, task.pass, []byte{}, fmt.Errorf("连接超时"))
}
break // 如果不需要重试,跳出重试循环
}
if len(Common.HashValue) > 0 {
break
}
2024-12-18 15:19:47 +08:00
}
2024-12-31 20:25:54 +08:00
resultChan <- nil
}()
}
// 等待所有线程完成
go func() {
wg.Wait()
close(resultChan)
}()
2024-12-18 15:19:47 +08:00
2024-12-31 20:25:54 +08:00
// 检查结果
for err := range resultChan {
if err != nil {
if retryErr := Common.CheckErrs(err); retryErr != nil {
return err
}
}
}
2024-12-31 20:25:54 +08:00
2024-12-18 15:19:47 +08:00
return nil
}
// logSuccessfulAuth 记录成功的认证
2024-12-19 16:15:53 +08:00
func logSuccessfulAuth(info *Common.HostInfo, user, pass string, hash []byte) {
2024-12-18 15:19:47 +08:00
var result string
2024-12-18 22:00:18 +08:00
if Common.Domain != "" {
2024-12-31 20:25:54 +08:00
result = fmt.Sprintf("[+] SMB2认证成功 %v:%v Domain:%v\\%v ",
2024-12-18 22:00:18 +08:00
info.Host, info.Ports, Common.Domain, user)
2024-12-18 15:19:47 +08:00
} else {
2024-12-31 20:25:54 +08:00
result = fmt.Sprintf("[+] SMB2认证成功 %v:%v User:%v ",
2024-12-18 15:19:47 +08:00
info.Host, info.Ports, user)
}
2024-08-29 09:50:32 +08:00
2024-12-18 15:19:47 +08:00
if len(hash) > 0 {
2024-12-20 03:46:09 +08:00
result += fmt.Sprintf("HashValue:%v", Common.HashValue)
2024-12-18 15:19:47 +08:00
} else {
result += fmt.Sprintf("Pass:%v", pass)
}
2024-12-18 22:00:18 +08:00
Common.LogSuccess(result)
}
2024-12-18 15:19:47 +08:00
// logFailedAuth 记录失败的认证
2024-12-19 16:15:53 +08:00
func logFailedAuth(info *Common.HostInfo, user, pass string, hash []byte, err error) {
2024-12-18 15:19:47 +08:00
var errlog string
if len(hash) > 0 {
2024-12-31 20:25:54 +08:00
errlog = fmt.Sprintf("[-] SMB2认证失败 %v:%v User:%v HashValue:%v Err:%v",
2024-12-20 03:46:09 +08:00
info.Host, info.Ports, user, Common.HashValue, err)
2024-12-18 15:19:47 +08:00
} else {
2024-12-31 20:25:54 +08:00
errlog = fmt.Sprintf("[-] SMB2认证失败 %v:%v User:%v Pass:%v Err:%v",
2024-12-18 15:19:47 +08:00
info.Host, info.Ports, user, pass, err)
}
errlog = strings.ReplaceAll(errlog, "\n", " ")
2024-12-18 22:00:18 +08:00
Common.LogError(errlog)
2024-12-18 15:19:47 +08:00
}
// Smb2Con 尝试SMB2连接并进行认证检查共享访问权限
2024-12-19 16:15:53 +08:00
func Smb2Con(info *Common.HostInfo, user string, pass string, hash []byte, hasprint bool) (flag bool, err error, flag2 bool) {
2024-12-18 15:19:47 +08:00
// 建立TCP连接
conn, err := net.DialTimeout("tcp", fmt.Sprintf("%s:445", info.Host),
2024-12-18 22:00:18 +08:00
time.Duration(Common.Timeout)*time.Second)
if err != nil {
2024-12-18 15:19:47 +08:00
return false, fmt.Errorf("连接失败: %v", err), false
}
2023-11-13 16:23:19 +08:00
defer conn.Close()
2024-12-18 15:19:47 +08:00
// 配置NTLM认证
initiator := smb2.NTLMInitiator{
User: user,
2024-12-18 22:00:18 +08:00
Domain: Common.Domain,
}
2024-12-18 15:19:47 +08:00
// 设置认证方式(哈希或密码)
if len(hash) > 0 {
initiator.Hash = hash
} else {
initiator.Password = pass
}
2024-12-18 15:19:47 +08:00
// 创建SMB2会话
d := &smb2.Dialer{
Initiator: &initiator,
}
2024-12-18 15:19:47 +08:00
session, err := d.Dial(conn)
if err != nil {
2024-12-18 15:19:47 +08:00
return false, fmt.Errorf("SMB2会话建立失败: %v", err), false
}
2024-12-18 15:19:47 +08:00
defer session.Logoff()
// 获取共享列表
shares, err := session.ListSharenames()
if err != nil {
2024-12-18 15:19:47 +08:00
return false, fmt.Errorf("获取共享列表失败: %v", err), false
}
2024-12-18 15:19:47 +08:00
// 打印共享信息(如果未打印过)
if !hasprint {
2024-12-18 15:19:47 +08:00
logShareInfo(info, user, pass, hash, shares)
flag2 = true
}
2024-12-18 15:19:47 +08:00
// 尝试访问C$共享以验证管理员权限
fs, err := session.Mount("C$")
if err != nil {
2024-12-18 15:19:47 +08:00
return false, fmt.Errorf("挂载C$失败: %v", err), flag2
}
defer fs.Umount()
2024-12-18 15:19:47 +08:00
// 尝试读取系统文件以验证权限
path := `Windows\win.ini`
f, err := fs.OpenFile(path, os.O_RDONLY, 0666)
if err != nil {
2024-12-18 15:19:47 +08:00
return false, fmt.Errorf("访问系统文件失败: %v", err), flag2
}
defer f.Close()
2024-12-18 15:19:47 +08:00
return true, nil, flag2
}
2023-11-13 11:24:44 +08:00
2024-12-18 15:19:47 +08:00
// logShareInfo 记录SMB共享信息
2024-12-19 16:15:53 +08:00
func logShareInfo(info *Common.HostInfo, user string, pass string, hash []byte, shares []string) {
2024-12-18 15:19:47 +08:00
var result string
// 构建基础信息
2024-12-18 22:00:18 +08:00
if Common.Domain != "" {
2024-12-18 15:19:47 +08:00
result = fmt.Sprintf("[*] SMB2共享信息 %v:%v Domain:%v\\%v ",
2024-12-18 22:00:18 +08:00
info.Host, info.Ports, Common.Domain, user)
2024-12-18 15:19:47 +08:00
} else {
result = fmt.Sprintf("[*] SMB2共享信息 %v:%v User:%v ",
info.Host, info.Ports, user)
}
// 添加认证信息
if len(hash) > 0 {
2024-12-20 03:46:09 +08:00
result += fmt.Sprintf("HashValue:%v ", Common.HashValue)
2024-12-18 15:19:47 +08:00
} else {
result += fmt.Sprintf("Pass:%v ", pass)
}
// 添加共享列表
result += fmt.Sprintf("可用共享: %v", shares)
2024-12-18 22:00:18 +08:00
Common.LogSuccess(result)
2024-12-18 15:19:47 +08:00
}