mirror of
				https://github.com/TeaOSLab/EdgeNode.git
				synced 2025-11-04 07:40:56 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			197 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package waf
 | 
						||
 | 
						||
import (
 | 
						||
	"github.com/TeaOSLab/EdgeCommon/pkg/serverconfigs/firewallconfigs"
 | 
						||
	"github.com/TeaOSLab/EdgeNode/internal/remotelogs"
 | 
						||
	"github.com/TeaOSLab/EdgeNode/internal/utils"
 | 
						||
	"github.com/TeaOSLab/EdgeNode/internal/utils/fasttime"
 | 
						||
	"github.com/TeaOSLab/EdgeNode/internal/waf/requests"
 | 
						||
	wafutils "github.com/TeaOSLab/EdgeNode/internal/waf/utils"
 | 
						||
	"github.com/iwind/TeaGo/types"
 | 
						||
	"net/http"
 | 
						||
	"net/url"
 | 
						||
	"strings"
 | 
						||
	"time"
 | 
						||
)
 | 
						||
 | 
						||
const (
 | 
						||
	CaptchaSeconds = 600 // 10 minutes
 | 
						||
	CaptchaPath    = "/WAF/VERIFY/CAPTCHA"
 | 
						||
)
 | 
						||
 | 
						||
type CaptchaAction struct {
 | 
						||
	BaseAction
 | 
						||
 | 
						||
	Life              int32 `yaml:"life" json:"life"`
 | 
						||
	MaxFails          int   `yaml:"maxFails" json:"maxFails"`                   // 最大失败次数
 | 
						||
	FailBlockTimeout  int   `yaml:"failBlockTimeout" json:"failBlockTimeout"`   // 失败拦截时间
 | 
						||
	FailBlockScopeAll bool  `yaml:"failBlockScopeAll" json:"failBlockScopeAll"` // 是否全局有效
 | 
						||
 | 
						||
	CountLetters int8 `yaml:"countLetters" json:"countLetters"`
 | 
						||
 | 
						||
	CaptchaType firewallconfigs.CaptchaType `yaml:"captchaType" json:"captchaType"`
 | 
						||
 | 
						||
	UIIsOn          bool   `yaml:"uiIsOn" json:"uiIsOn"`                   // 是否使用自定义UI
 | 
						||
	UITitle         string `yaml:"uiTitle" json:"uiTitle"`                 // 消息标题
 | 
						||
	UIPrompt        string `yaml:"uiPrompt" json:"uiPrompt"`               // 消息提示
 | 
						||
	UIButtonTitle   string `yaml:"uiButtonTitle" json:"uiButtonTitle"`     // 按钮标题
 | 
						||
	UIShowRequestId bool   `yaml:"uiShowRequestId" json:"uiShowRequestId"` // 是否显示请求ID
 | 
						||
	UICss           string `yaml:"uiCss" json:"uiCss"`                     // CSS样式
 | 
						||
	UIFooter        string `yaml:"uiFooter" json:"uiFooter"`               // 页脚
 | 
						||
	UIBody          string `yaml:"uiBody" json:"uiBody"`                   // 内容轮廓
 | 
						||
 | 
						||
	OneClickUIIsOn          bool   `yaml:"oneClickUIIsOn" json:"oneClickUIIsOn"`                   // 是否使用自定义UI
 | 
						||
	OneClickUITitle         string `yaml:"oneClickUITitle" json:"oneClickUITitle"`                 // 消息标题
 | 
						||
	OneClickUIPrompt        string `yaml:"oneClickUIPrompt" json:"oneClickUIPrompt"`               // 消息提示
 | 
						||
	OneClickUIShowRequestId bool   `yaml:"oneClickUIShowRequestId" json:"oneClickUIShowRequestId"` // 是否显示请求ID
 | 
						||
	OneClickUICss           string `yaml:"oneClickUICss" json:"oneClickUICss"`                     // CSS样式
 | 
						||
	OneClickUIFooter        string `yaml:"oneClickUIFooter" json:"oneClickUIFooter"`               // 页脚
 | 
						||
	OneClickUIBody          string `yaml:"oneClickUIBody" json:"oneClickUIBody"`                   // 内容轮廓
 | 
						||
 | 
						||
	SlideUIIsOn          bool   `yaml:"sliceUIIsOn" json:"sliceUIIsOn"`                   // 是否使用自定义UI
 | 
						||
	SlideUITitle         string `yaml:"slideUITitle" json:"slideUITitle"`                 // 消息标题
 | 
						||
	SlideUIPrompt        string `yaml:"slideUIPrompt" json:"slideUIPrompt"`               // 消息提示
 | 
						||
	SlideUIShowRequestId bool   `yaml:"SlideUIShowRequestId" json:"SlideUIShowRequestId"` // 是否显示请求ID
 | 
						||
	SlideUICss           string `yaml:"slideUICss" json:"slideUICss"`                     // CSS样式
 | 
						||
	SlideUIFooter        string `yaml:"slideUIFooter" json:"slideUIFooter"`               // 页脚
 | 
						||
	SlideUIBody          string `yaml:"slideUIBody" json:"slideUIBody"`                   // 内容轮廓
 | 
						||
 | 
						||
	GeeTestConfig *firewallconfigs.GeeTestConfig `yaml:"geeTestConfig" json:"geeTestConfig"` // 极验设置 MUST be struct
 | 
						||
 | 
						||
	Lang           string `yaml:"lang" json:"lang"`                     // 语言,zh-CN, en-US ...
 | 
						||
	AddToWhiteList bool   `yaml:"addToWhiteList" json:"addToWhiteList"` // 是否加入到白名单
 | 
						||
	Scope          string `yaml:"scope" json:"scope"`
 | 
						||
}
 | 
						||
 | 
						||
func (this *CaptchaAction) Init(waf *WAF) error {
 | 
						||
	if waf.DefaultCaptchaAction != nil {
 | 
						||
		if this.Life <= 0 {
 | 
						||
			this.Life = waf.DefaultCaptchaAction.Life
 | 
						||
		}
 | 
						||
		if this.MaxFails <= 0 {
 | 
						||
			this.MaxFails = waf.DefaultCaptchaAction.MaxFails
 | 
						||
		}
 | 
						||
		if this.FailBlockTimeout <= 0 {
 | 
						||
			this.FailBlockTimeout = waf.DefaultCaptchaAction.FailBlockTimeout
 | 
						||
		}
 | 
						||
		this.FailBlockScopeAll = waf.DefaultCaptchaAction.FailBlockScopeAll
 | 
						||
 | 
						||
		if this.CountLetters <= 0 {
 | 
						||
			this.CountLetters = waf.DefaultCaptchaAction.CountLetters
 | 
						||
		}
 | 
						||
 | 
						||
		this.UIIsOn = waf.DefaultCaptchaAction.UIIsOn
 | 
						||
		if len(this.UITitle) == 0 {
 | 
						||
			this.UITitle = waf.DefaultCaptchaAction.UITitle
 | 
						||
		}
 | 
						||
		if len(this.UIPrompt) == 0 {
 | 
						||
			this.UIPrompt = waf.DefaultCaptchaAction.UIPrompt
 | 
						||
		}
 | 
						||
		if len(this.UIButtonTitle) == 0 {
 | 
						||
			this.UIButtonTitle = waf.DefaultCaptchaAction.UIButtonTitle
 | 
						||
		}
 | 
						||
		this.UIShowRequestId = waf.DefaultCaptchaAction.UIShowRequestId
 | 
						||
		if len(this.UICss) == 0 {
 | 
						||
			this.UICss = waf.DefaultCaptchaAction.UICss
 | 
						||
		}
 | 
						||
		if len(this.UIFooter) == 0 {
 | 
						||
			this.UIFooter = waf.DefaultCaptchaAction.UIFooter
 | 
						||
		}
 | 
						||
		if len(this.UIBody) == 0 {
 | 
						||
			this.UIBody = waf.DefaultCaptchaAction.UIBody
 | 
						||
		}
 | 
						||
		if len(this.Lang) == 0 {
 | 
						||
			this.Lang = waf.DefaultCaptchaAction.Lang
 | 
						||
		}
 | 
						||
 | 
						||
		if len(this.CaptchaType) == 0 {
 | 
						||
			this.CaptchaType = waf.DefaultCaptchaAction.CaptchaType
 | 
						||
		}
 | 
						||
	}
 | 
						||
 | 
						||
	return nil
 | 
						||
}
 | 
						||
 | 
						||
func (this *CaptchaAction) Code() string {
 | 
						||
	return ActionCaptcha
 | 
						||
}
 | 
						||
 | 
						||
func (this *CaptchaAction) IsAttack() bool {
 | 
						||
	return false
 | 
						||
}
 | 
						||
 | 
						||
func (this *CaptchaAction) WillChange() bool {
 | 
						||
	return true
 | 
						||
}
 | 
						||
 | 
						||
func (this *CaptchaAction) Perform(waf *WAF, group *RuleGroup, set *RuleSet, req requests.Request, writer http.ResponseWriter) PerformResult {
 | 
						||
	// 是否在白名单中
 | 
						||
	if SharedIPWhiteList.Contains(wafutils.ComposeIPType(set.Id, req), this.Scope, req.WAFServerId(), req.WAFRemoteIP()) {
 | 
						||
		return PerformResult{
 | 
						||
			ContinueRequest: true,
 | 
						||
		}
 | 
						||
	}
 | 
						||
 | 
						||
	// 检查Cookie值
 | 
						||
	var fullCookieName = captchaCookiePrefix + "_" + types.String(set.Id)
 | 
						||
	cookie, err := req.WAFRaw().Cookie(fullCookieName)
 | 
						||
	if err == nil && cookie != nil && len(cookie.Value) > 0 {
 | 
						||
		var info = &AllowCookieInfo{}
 | 
						||
		err = info.Decode(cookie.Value)
 | 
						||
		if err == nil && set.Id == info.SetId && info.ExpiresAt > fasttime.Now().Unix() {
 | 
						||
			// 重新记录到白名单
 | 
						||
			SharedIPWhiteList.RecordIP(wafutils.ComposeIPType(set.Id, req), this.Scope, req.WAFServerId(), req.WAFRemoteIP(), info.ExpiresAt, waf.Id, false, group.Id, set.Id, "")
 | 
						||
 | 
						||
			return PerformResult{
 | 
						||
				ContinueRequest: true,
 | 
						||
			}
 | 
						||
		}
 | 
						||
	}
 | 
						||
 | 
						||
	var refURL = req.WAFRaw().URL.String()
 | 
						||
 | 
						||
	// 覆盖配置
 | 
						||
	if strings.HasPrefix(refURL, CaptchaPath) {
 | 
						||
		var info = req.WAFRaw().URL.Query().Get("info")
 | 
						||
		if len(info) > 0 {
 | 
						||
			var oldArg = &InfoArg{}
 | 
						||
			decodeErr := oldArg.Decode(info)
 | 
						||
			if decodeErr == nil && oldArg.IsValid() {
 | 
						||
				refURL = oldArg.URL
 | 
						||
			} else {
 | 
						||
				// 兼容老版本
 | 
						||
				m, err := utils.SimpleDecryptMap(info)
 | 
						||
				if err == nil && m != nil {
 | 
						||
					refURL = m.GetString("url")
 | 
						||
				}
 | 
						||
			}
 | 
						||
		}
 | 
						||
	}
 | 
						||
 | 
						||
	var captchaConfig = &InfoArg{
 | 
						||
		ActionId:         this.ActionId(),
 | 
						||
		Timestamp:        time.Now().Unix(),
 | 
						||
		URL:              refURL,
 | 
						||
		PolicyId:         waf.Id,
 | 
						||
		GroupId:          group.Id,
 | 
						||
		SetId:            set.Id,
 | 
						||
		UseLocalFirewall: waf.UseLocalFirewall && (this.FailBlockScopeAll || this.Scope == firewallconfigs.AllowScopeGlobal),
 | 
						||
	}
 | 
						||
	info, err := utils.SimpleEncryptObject(captchaConfig)
 | 
						||
	if err != nil {
 | 
						||
		remotelogs.Error("WAF_CAPTCHA_ACTION", "encode captcha config failed: "+err.Error())
 | 
						||
		return PerformResult{
 | 
						||
			ContinueRequest: true,
 | 
						||
		}
 | 
						||
	}
 | 
						||
 | 
						||
	// 占用一次失败次数
 | 
						||
	CaptchaIncreaseFails(req, this, waf.Id, group.Id, set.Id, CaptchaPageCodeInit, waf.UseLocalFirewall && (this.FailBlockScopeAll || this.Scope == firewallconfigs.FirewallScopeGlobal))
 | 
						||
 | 
						||
	req.DisableStat()
 | 
						||
	req.ProcessResponseHeaders(writer.Header(), http.StatusTemporaryRedirect)
 | 
						||
	http.Redirect(writer, req.WAFRaw(), CaptchaPath+"?info="+url.QueryEscape(info)+"&from="+url.QueryEscape(refURL), http.StatusTemporaryRedirect)
 | 
						||
 | 
						||
	return PerformResult{}
 | 
						||
}
 |