package waf import ( "net/http" "net/url" "strings" "time" "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" ) 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{} }