fscan/plugins/services/postgresql.go
ZacharyZcR ed117a14fd fix: 修复PostgreSQL插件在禁用暴力破解模式下无法识别服务的问题
问题描述:
- 在使用-nobr参数时,PostgreSQL插件的identifyService方法
  无法正确识别PostgreSQL服务
- 原识别逻辑过于严格,只检查错误信息是否包含'postgres'字符串

修复方案:
- 扩展识别关键词列表,包含PostgreSQL常见错误信息
- 添加: authentication, database, password, role, user, pq: 等关键词
- 改进错误处理,提供更详细的调试信息
- 保持向后兼容性,不影响暴力破解功能

测试验证:
- 禁用暴力破解模式: 能正确识别PostgreSQL服务
- 启用暴力破解模式: 能正常破解弱密码
- 两种模式均工作正常
2025-09-02 02:30:07 +00:00

155 lines
3.6 KiB
Go
Raw 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 services
import (
"context"
"database/sql"
"fmt"
"strings"
"time"
_ "github.com/lib/pq"
"github.com/shadow1ng/fscan/common"
"github.com/shadow1ng/fscan/plugins"
)
type PostgreSQLPlugin struct {
plugins.BasePlugin
}
func NewPostgreSQLPlugin() *PostgreSQLPlugin {
return &PostgreSQLPlugin{
BasePlugin: plugins.NewBasePlugin("postgresql"),
}
}
func (p *PostgreSQLPlugin) Scan(ctx context.Context, info *common.HostInfo) *ScanResult {
target := fmt.Sprintf("%s:%s", info.Host, info.Ports)
if common.DisableBrute {
return p.identifyService(ctx, info)
}
credentials := GenerateCredentials("postgresql")
if len(credentials) == 0 {
return &ScanResult{
Success: false,
Service: "postgresql",
Error: fmt.Errorf("没有可用的测试凭据"),
}
}
for _, cred := range credentials {
if db := p.testCredential(ctx, info, cred); db != nil {
db.Close()
common.LogSuccess(fmt.Sprintf("PostgreSQL %s %s:%s", target, cred.Username, cred.Password))
return &ScanResult{
Success: true,
Service: "postgresql",
Username: cred.Username,
Password: cred.Password,
}
}
}
return &ScanResult{
Success: false,
Service: "postgresql",
Error: fmt.Errorf("未发现弱密码"),
}
}
func (p *PostgreSQLPlugin) testCredential(ctx context.Context, info *common.HostInfo, cred Credential) *sql.DB {
connStr := fmt.Sprintf("postgres://%s:%s@%s:%s/postgres?sslmode=disable&connect_timeout=%d",
cred.Username, cred.Password, info.Host, info.Ports, common.Timeout)
db, err := sql.Open("postgres", connStr)
if err != nil {
return nil
}
db.SetConnMaxLifetime(time.Duration(common.Timeout) * time.Second)
db.SetMaxOpenConns(1)
db.SetMaxIdleConns(0)
pingCtx, cancel := context.WithTimeout(ctx, time.Duration(common.Timeout)*time.Second)
defer cancel()
err = db.PingContext(pingCtx)
if err != nil {
db.Close()
return nil
}
return db
}
func (p *PostgreSQLPlugin) identifyService(ctx context.Context, info *common.HostInfo) *ScanResult {
target := fmt.Sprintf("%s:%s", info.Host, info.Ports)
connStr := fmt.Sprintf("postgres://invalid:invalid@%s:%s/postgres?sslmode=disable&connect_timeout=%d",
info.Host, info.Ports, common.Timeout)
db, err := sql.Open("postgres", connStr)
if err != nil {
return &ScanResult{
Success: false,
Service: "postgresql",
Error: err,
}
}
defer db.Close()
pingCtx, cancel := context.WithTimeout(ctx, time.Duration(common.Timeout)*time.Second)
defer cancel()
err = db.PingContext(pingCtx)
// 改进识别逻辑任何PostgreSQL相关的响应都认为是有效服务
var banner string
if err != nil {
errMsg := strings.ToLower(err.Error())
// PostgreSQL常见错误关键词
if strings.Contains(errMsg, "postgres") ||
strings.Contains(errMsg, "authentication") ||
strings.Contains(errMsg, "database") ||
strings.Contains(errMsg, "password") ||
strings.Contains(errMsg, "role") ||
strings.Contains(errMsg, "user") ||
strings.Contains(errMsg, "pq:") {
banner = "PostgreSQL"
} else {
return &ScanResult{
Success: false,
Service: "postgresql",
Error: fmt.Errorf("无法识别为PostgreSQL服务: %s", err.Error()),
}
}
} else {
banner = "PostgreSQL (连接成功)"
}
common.LogSuccess(fmt.Sprintf("PostgreSQL %s %s", target, banner))
return &ScanResult{
Success: true,
Service: "postgresql",
Banner: banner,
}
}
func init() {
// 使用高效注册方式:直接传递端口信息,避免实例创建
RegisterPluginWithPorts("postgresql", func() Plugin {
return NewPostgreSQLPlugin()
}, []int{5432, 5433, 5434})
}