mirror of
https://github.com/projectdiscovery/nuclei.git
synced 2025-12-18 00:25:26 +00:00
consider protocolType in max host error (#5668)
* consider protocolType in max host error * add mutex when updating internal-event
This commit is contained in:
parent
e4dae52d5a
commit
1f945d6d50
@ -107,7 +107,7 @@ func (e *Engine) executeTemplateWithTargets(ctx context.Context, template *templ
|
|||||||
currentInfo.Unlock()
|
currentInfo.Unlock()
|
||||||
|
|
||||||
// Skip if the host has had errors
|
// Skip if the host has had errors
|
||||||
if e.executerOpts.HostErrorsCache != nil && e.executerOpts.HostErrorsCache.Check(contextargs.NewWithMetaInput(ctx, scannedValue)) {
|
if e.executerOpts.HostErrorsCache != nil && e.executerOpts.HostErrorsCache.Check(e.executerOpts.ProtocolType.String(), contextargs.NewWithMetaInput(ctx, scannedValue)) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -98,7 +98,7 @@ func (e *Engine) runWorkflowStep(template *workflows.WorkflowTemplate, ctx *scan
|
|||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if w.Options.HostErrorsCache != nil {
|
if w.Options.HostErrorsCache != nil {
|
||||||
w.Options.HostErrorsCache.MarkFailed(ctx.Input, err)
|
w.Options.HostErrorsCache.MarkFailed(w.Options.ProtocolType.String(), ctx.Input, err)
|
||||||
}
|
}
|
||||||
if len(template.Executers) == 1 {
|
if len(template.Executers) == 1 {
|
||||||
mainErr = err
|
mainErr = err
|
||||||
|
|||||||
@ -22,8 +22,8 @@ import (
|
|||||||
type CacheInterface interface {
|
type CacheInterface interface {
|
||||||
SetVerbose(verbose bool) // log verbosely
|
SetVerbose(verbose bool) // log verbosely
|
||||||
Close() // close the cache
|
Close() // close the cache
|
||||||
Check(ctx *contextargs.Context) bool // return true if the host should be skipped
|
Check(protoType string, ctx *contextargs.Context) bool // return true if the host should be skipped
|
||||||
MarkFailed(ctx *contextargs.Context, err error) // record a failure (and cause) for the host
|
MarkFailed(protoType string, ctx *contextargs.Context, err error) // record a failure (and cause) for the host
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -115,7 +115,7 @@ func (c *Cache) NormalizeCacheValue(value string) string {
|
|||||||
// - URL: https?:// type
|
// - URL: https?:// type
|
||||||
// - Host:port type
|
// - Host:port type
|
||||||
// - host type
|
// - host type
|
||||||
func (c *Cache) Check(ctx *contextargs.Context) bool {
|
func (c *Cache) Check(protoType string, ctx *contextargs.Context) bool {
|
||||||
finalValue := c.GetKeyFromContext(ctx, nil)
|
finalValue := c.GetKeyFromContext(ctx, nil)
|
||||||
|
|
||||||
existingCacheItem, err := c.failedTargets.GetIFPresent(finalValue)
|
existingCacheItem, err := c.failedTargets.GetIFPresent(finalValue)
|
||||||
@ -138,8 +138,8 @@ func (c *Cache) Check(ctx *contextargs.Context) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// MarkFailed marks a host as failed previously
|
// MarkFailed marks a host as failed previously
|
||||||
func (c *Cache) MarkFailed(ctx *contextargs.Context, err error) {
|
func (c *Cache) MarkFailed(protoType string, ctx *contextargs.Context, err error) {
|
||||||
if !c.checkError(err) {
|
if !c.checkError(protoType, err) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
finalValue := c.GetKeyFromContext(ctx, err)
|
finalValue := c.GetKeyFromContext(ctx, err)
|
||||||
@ -186,11 +186,13 @@ var reCheckError = regexp.MustCompile(`(no address found for host|could not reso
|
|||||||
// added to the host skipping table.
|
// added to the host skipping table.
|
||||||
// it first parses error and extracts the cause and checks for blacklisted
|
// it first parses error and extracts the cause and checks for blacklisted
|
||||||
// or common errors that should be skipped
|
// or common errors that should be skipped
|
||||||
func (c *Cache) checkError(err error) bool {
|
func (c *Cache) checkError(protoType string, err error) bool {
|
||||||
if err == nil {
|
if err == nil {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
if protoType != "http" {
|
||||||
|
return false
|
||||||
|
}
|
||||||
kind := errkit.GetErrorKind(err, nucleierr.ErrTemplateLogic)
|
kind := errkit.GetErrorKind(err, nucleierr.ErrTemplateLogic)
|
||||||
switch kind {
|
switch kind {
|
||||||
case nucleierr.ErrTemplateLogic:
|
case nucleierr.ErrTemplateLogic:
|
||||||
|
|||||||
@ -11,12 +11,16 @@ import (
|
|||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
protoType = "http"
|
||||||
|
)
|
||||||
|
|
||||||
func TestCacheCheck(t *testing.T) {
|
func TestCacheCheck(t *testing.T) {
|
||||||
cache := New(3, DefaultMaxHostsCount, nil)
|
cache := New(3, DefaultMaxHostsCount, nil)
|
||||||
|
|
||||||
for i := 0; i < 100; i++ {
|
for i := 0; i < 100; i++ {
|
||||||
cache.MarkFailed(newCtxArgs("test"), fmt.Errorf("could not resolve host"))
|
cache.MarkFailed(protoType, newCtxArgs("test"), fmt.Errorf("could not resolve host"))
|
||||||
got := cache.Check(newCtxArgs("test"))
|
got := cache.Check(protoType, newCtxArgs("test"))
|
||||||
if i < 2 {
|
if i < 2 {
|
||||||
// till 3 the host is not flagged to skip
|
// till 3 the host is not flagged to skip
|
||||||
require.False(t, got)
|
require.False(t, got)
|
||||||
@ -26,7 +30,7 @@ func TestCacheCheck(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
value := cache.Check(newCtxArgs("test"))
|
value := cache.Check(protoType, newCtxArgs("test"))
|
||||||
require.Equal(t, true, value, "could not get checked value")
|
require.Equal(t, true, value, "could not get checked value")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -34,8 +38,8 @@ func TestTrackErrors(t *testing.T) {
|
|||||||
cache := New(3, DefaultMaxHostsCount, []string{"custom error"})
|
cache := New(3, DefaultMaxHostsCount, []string{"custom error"})
|
||||||
|
|
||||||
for i := 0; i < 100; i++ {
|
for i := 0; i < 100; i++ {
|
||||||
cache.MarkFailed(newCtxArgs("custom"), fmt.Errorf("got: nested: custom error"))
|
cache.MarkFailed(protoType, newCtxArgs("custom"), fmt.Errorf("got: nested: custom error"))
|
||||||
got := cache.Check(newCtxArgs("custom"))
|
got := cache.Check(protoType, newCtxArgs("custom"))
|
||||||
if i < 2 {
|
if i < 2 {
|
||||||
// till 3 the host is not flagged to skip
|
// till 3 the host is not flagged to skip
|
||||||
require.False(t, got)
|
require.False(t, got)
|
||||||
@ -44,7 +48,7 @@ func TestTrackErrors(t *testing.T) {
|
|||||||
require.True(t, got)
|
require.True(t, got)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
value := cache.Check(newCtxArgs("custom"))
|
value := cache.Check(protoType, newCtxArgs("custom"))
|
||||||
require.Equal(t, true, value, "could not get checked value")
|
require.Equal(t, true, value, "could not get checked value")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -86,7 +90,7 @@ func TestCacheMarkFailed(t *testing.T) {
|
|||||||
|
|
||||||
for _, test := range tests {
|
for _, test := range tests {
|
||||||
normalizedCacheValue := cache.GetKeyFromContext(newCtxArgs(test.host), nil)
|
normalizedCacheValue := cache.GetKeyFromContext(newCtxArgs(test.host), nil)
|
||||||
cache.MarkFailed(newCtxArgs(test.host), fmt.Errorf("no address found for host"))
|
cache.MarkFailed(protoType, newCtxArgs(test.host), fmt.Errorf("no address found for host"))
|
||||||
failedTarget, err := cache.failedTargets.Get(normalizedCacheValue)
|
failedTarget, err := cache.failedTargets.Get(normalizedCacheValue)
|
||||||
require.Nil(t, err)
|
require.Nil(t, err)
|
||||||
require.NotNil(t, failedTarget)
|
require.NotNil(t, failedTarget)
|
||||||
@ -122,14 +126,14 @@ func TestCacheMarkFailedConcurrent(t *testing.T) {
|
|||||||
wg.Add(1)
|
wg.Add(1)
|
||||||
go func() {
|
go func() {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
cache.MarkFailed(newCtxArgs(currentTest.host), fmt.Errorf("could not resolve host"))
|
cache.MarkFailed(protoType, newCtxArgs(currentTest.host), fmt.Errorf("could not resolve host"))
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
|
|
||||||
for _, test := range tests {
|
for _, test := range tests {
|
||||||
require.True(t, cache.Check(newCtxArgs(test.host)))
|
require.True(t, cache.Check(protoType, newCtxArgs(test.host)))
|
||||||
|
|
||||||
normalizedCacheValue := cache.NormalizeCacheValue(test.host)
|
normalizedCacheValue := cache.NormalizeCacheValue(test.host)
|
||||||
failedTarget, err := cache.failedTargets.Get(normalizedCacheValue)
|
failedTarget, err := cache.failedTargets.Get(normalizedCacheValue)
|
||||||
|
|||||||
@ -1177,14 +1177,14 @@ func (request *Request) markUnresponsiveAddress(input *contextargs.Context, err
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
if request.options.HostErrorsCache != nil {
|
if request.options.HostErrorsCache != nil {
|
||||||
request.options.HostErrorsCache.MarkFailed(input, err)
|
request.options.HostErrorsCache.MarkFailed(request.options.ProtocolType.String(), input, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// isUnresponsiveAddress checks if the error is a unreponsive based on its execution history
|
// isUnresponsiveAddress checks if the error is a unreponsive based on its execution history
|
||||||
func (request *Request) isUnresponsiveAddress(input *contextargs.Context) bool {
|
func (request *Request) isUnresponsiveAddress(input *contextargs.Context) bool {
|
||||||
if request.options.HostErrorsCache != nil {
|
if request.options.HostErrorsCache != nil {
|
||||||
return request.options.HostErrorsCache.Check(input)
|
return request.options.HostErrorsCache.Check(request.options.ProtocolType.String(), input)
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|||||||
@ -161,7 +161,7 @@ func (request *Request) executeAllFuzzingRules(input *contextargs.Context, value
|
|||||||
func (request *Request) executeGeneratedFuzzingRequest(gr fuzz.GeneratedRequest, input *contextargs.Context, callback protocols.OutputEventCallback) bool {
|
func (request *Request) executeGeneratedFuzzingRequest(gr fuzz.GeneratedRequest, input *contextargs.Context, callback protocols.OutputEventCallback) bool {
|
||||||
hasInteractMatchers := interactsh.HasMatchers(request.CompiledOperators)
|
hasInteractMatchers := interactsh.HasMatchers(request.CompiledOperators)
|
||||||
hasInteractMarkers := len(gr.InteractURLs) > 0
|
hasInteractMarkers := len(gr.InteractURLs) > 0
|
||||||
if request.options.HostErrorsCache != nil && request.options.HostErrorsCache.Check(input) {
|
if request.options.HostErrorsCache != nil && request.options.HostErrorsCache.Check(request.options.ProtocolType.String(), input) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
request.options.RateLimitTake()
|
request.options.RateLimitTake()
|
||||||
@ -215,7 +215,7 @@ func (request *Request) executeGeneratedFuzzingRequest(gr fuzz.GeneratedRequest,
|
|||||||
}
|
}
|
||||||
if requestErr != nil {
|
if requestErr != nil {
|
||||||
if request.options.HostErrorsCache != nil {
|
if request.options.HostErrorsCache != nil {
|
||||||
request.options.HostErrorsCache.MarkFailed(input, requestErr)
|
request.options.HostErrorsCache.MarkFailed(request.options.ProtocolType.String(), input, requestErr)
|
||||||
}
|
}
|
||||||
gologger.Verbose().Msgf("[%s] Error occurred in request: %s\n", request.options.TemplateID, requestErr)
|
gologger.Verbose().Msgf("[%s] Error occurred in request: %s\n", request.options.TemplateID, requestErr)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -504,14 +504,14 @@ func (request *Request) markUnresponsiveAddress(input *contextargs.Context, err
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
if request.options.HostErrorsCache != nil {
|
if request.options.HostErrorsCache != nil {
|
||||||
request.options.HostErrorsCache.MarkFailed(input, err)
|
request.options.HostErrorsCache.MarkFailed(request.options.ProtocolType.String(), input, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// isUnresponsiveAddress checks if the error is a unreponsive based on its execution history
|
// isUnresponsiveAddress checks if the error is a unreponsive based on its execution history
|
||||||
func (request *Request) isUnresponsiveAddress(input *contextargs.Context) bool {
|
func (request *Request) isUnresponsiveAddress(input *contextargs.Context) bool {
|
||||||
if request.options.HostErrorsCache != nil {
|
if request.options.HostErrorsCache != nil {
|
||||||
return request.options.HostErrorsCache.Check(input)
|
return request.options.HostErrorsCache.Check(request.options.ProtocolType.String(), input)
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|||||||
@ -274,7 +274,7 @@ func (e *ClusterExecuter) Execute(ctx *scan.ScanContext) (bool, error) {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
if err != nil && e.options.HostErrorsCache != nil {
|
if err != nil && e.options.HostErrorsCache != nil {
|
||||||
e.options.HostErrorsCache.MarkFailed(ctx.Input, err)
|
e.options.HostErrorsCache.MarkFailed(e.options.ProtocolType.String(), ctx.Input, err)
|
||||||
}
|
}
|
||||||
return results, err
|
return results, err
|
||||||
}
|
}
|
||||||
@ -310,7 +310,7 @@ func (e *ClusterExecuter) ExecuteWithResults(ctx *scan.ScanContext) ([]*output.R
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err != nil && e.options.HostErrorsCache != nil {
|
if err != nil && e.options.HostErrorsCache != nil {
|
||||||
e.options.HostErrorsCache.MarkFailed(ctx.Input, err)
|
e.options.HostErrorsCache.MarkFailed(e.options.ProtocolType.String(), ctx.Input, err)
|
||||||
}
|
}
|
||||||
return scanCtx.GenerateResult(), err
|
return scanCtx.GenerateResult(), err
|
||||||
}
|
}
|
||||||
|
|||||||
@ -206,7 +206,9 @@ func (e *TemplateExecuter) Execute(ctx *scan.ScanContext) (bool, error) {
|
|||||||
ctx.LogError(errx)
|
ctx.LogError(errx)
|
||||||
|
|
||||||
if lastMatcherEvent != nil {
|
if lastMatcherEvent != nil {
|
||||||
|
lastMatcherEvent.Lock()
|
||||||
lastMatcherEvent.InternalEvent["error"] = getErrorCause(ctx.GenerateErrorMessage())
|
lastMatcherEvent.InternalEvent["error"] = getErrorCause(ctx.GenerateErrorMessage())
|
||||||
|
lastMatcherEvent.Unlock()
|
||||||
writeFailureCallback(lastMatcherEvent, e.options.Options.MatcherStatus)
|
writeFailureCallback(lastMatcherEvent, e.options.Options.MatcherStatus)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -85,7 +85,7 @@ func (g *Generic) ExecuteWithResults(ctx *scan.ScanContext) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.LogError(err)
|
ctx.LogError(err)
|
||||||
if g.options.HostErrorsCache != nil {
|
if g.options.HostErrorsCache != nil {
|
||||||
g.options.HostErrorsCache.MarkFailed(ctx.Input, err)
|
g.options.HostErrorsCache.MarkFailed(g.options.ProtocolType.String(), ctx.Input, err)
|
||||||
}
|
}
|
||||||
gologger.Warning().Msgf("[%s] Could not execute request for %s: %s\n", g.options.TemplateID, ctx.Input.MetaInput.PrettyPrint(), err)
|
gologger.Warning().Msgf("[%s] Could not execute request for %s: %s\n", g.options.TemplateID, ctx.Input.MetaInput.PrettyPrint(), err)
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user