mirror of
https://github.com/projectdiscovery/nuclei.git
synced 2026-02-01 08:13:09 +08:00
to avoid redundant ECDSA checks.
Add `protocols.TemplateVerification` & callback
mechanism to `protocols.ExecutorOptions` to enable
reusing cached verification data from the metadata
index. Also updating internal
`templates.parseTemplate` func to skip ECDSA
verification when cached data is any, and wire the
callback in `loader.New` for metadata-backed
lookups.
Proof:
```
$ go tool pprof -list "signer\..*" -base 3.6.2.cpu patch.cpu
Total: 34.78s
ROUTINE ======================== github.com/projectdiscovery/nuclei/v3/pkg/templates/signer.(*TemplateSigner).Verify in /home/dw1/Development/PD/nuclei/pkg/templates/signer/tmpl_signer.go
0 -1.75s (flat, cum) 5.03% of Total
. . 131:func (t *TemplateSigner) Verify(data []byte, tmpl SignableTemplate) (bool, error) {
. -70ms 132: signature, content := ExtractSignatureAndContent(data)
. . 133: if len(signature) == 0 {
. . 134: return false, errors.New("no signature found")
. . 135: }
. . 136:
. . 137: if !bytes.HasPrefix(signature, []byte(SignaturePattern)) {
. . 138: return false, errors.New("signature must be at the end of the template")
. . 139: }
. . 140:
. . 141: digestData := bytes.TrimSpace(bytes.TrimPrefix(signature, []byte(SignaturePattern)))
. . 142: // remove fragment from digest as it is used for re-signing purposes only
. . 143: digestString := strings.TrimSuffix(string(digestData), ":"+t.GetUserFragment())
. -20ms 144: digest, err := hex.DecodeString(digestString)
. . 145: if err != nil {
. . 146: return false, err
. . 147: }
. . 148:
. . 149: // normalize content by removing \r\n everywhere since this only done for verification
. . 150: // it does not affect the actual template
. -40ms 151: content = bytes.ReplaceAll(content, []byte("\r\n"), []byte("\n"))
. . 152:
. . 153: buff := bytes.NewBuffer(content)
. . 154: // if file has any imports process them
. . 155: for _, file := range tmpl.GetFileImports() {
. . 156: bin, err := os.ReadFile(file)
. . 157: if err != nil {
. . 158: return false, err
. . 159: }
. . 160: buff.WriteRune('\n')
. . 161: buff.Write(bin)
. . 162: }
. . 163:
. -1.62s 164: return t.verify(buff.Bytes(), digest)
. . 165:}
. . 166:
. . 167:// Verify verifies the given data with the template signer
. . 168:// Note: this should not be used for verifying templates as file references
. . 169:// in templates are not processed
ROUTINE ======================== github.com/projectdiscovery/nuclei/v3/pkg/templates/signer.(*TemplateSigner).verify in /home/dw1/Development/PD/nuclei/pkg/templates/signer/tmpl_signer.go
0 -1.62s (flat, cum) 4.66% of Total
. . 170:func (t *TemplateSigner) verify(data, signatureData []byte) (bool, error) {
. -50ms 171: dataHash := sha256.Sum256(data)
. . 172:
. . 173: var signature []byte
. -70ms 174: if err := gob.NewDecoder(bytes.NewReader(signatureData)).Decode(&signature); err != nil {
. . 175: return false, err
. . 176: }
. -1.50s 177: return ecdsa.VerifyASN1(t.handler.ecdsaPubKey, dataHash[:], signature), nil
. . 178:}
. . 179:
. . 180:// NewTemplateSigner creates a new signer for signing templates
. . 181:func NewTemplateSigner(cert, privateKey []byte) (*TemplateSigner, error) {
. . 182: handler := &KeyHandler{}
ROUTINE ======================== github.com/projectdiscovery/nuclei/v3/pkg/templates/signer.ExtractSignatureAndContent in /home/dw1/Development/PD/nuclei/pkg/templates/signer/tmpl_signer.go
0 -70ms (flat, cum) 0.2% of Total
. . 29:func ExtractSignatureAndContent(data []byte) (signature, content []byte) {
. -50ms 30: dataStr := string(data)
. -20ms 31: if idx := strings.LastIndex(dataStr, SignaturePattern); idx != -1 {
. . 32: signature = []byte(strings.TrimSpace(dataStr[idx:]))
. . 33: content = bytes.TrimSpace(data[:idx])
. . 34: } else {
. . 35: content = data
. . 36: }
$ go tool pprof -list "crypto/ecdsa\.VerifyASN1" 3.6.2.cpu patch.cpu
Total: 34.80s
ROUTINE ======================== crypto/ecdsa.VerifyASN1 in /usr/local/go/src/crypto/ecdsa/ecdsa.go
0 1.50s (flat, cum) 4.31% of Total
. . 500:func VerifyASN1(pub *PublicKey, hash, sig []byte) bool {
. . 501: if boring.Enabled {
. . 502: key, err := boringPublicKey(pub)
. . 503: if err != nil {
. . 504: return false
. . 505: }
. . 506: return boring.VerifyECDSA(key, hash, sig)
. . 507: }
. . 508: boring.UnreachableExceptTests()
. . 509:
. . 510: switch pub.Curve.Params() {
. . 511: case elliptic.P224().Params():
. . 512: return verifyFIPS(ecdsa.P224(), pub, hash, sig)
. . 513: case elliptic.P256().Params():
. 1.50s 514: return verifyFIPS(ecdsa.P256(), pub, hash, sig)
. . 515: case elliptic.P384().Params():
. . 516: return verifyFIPS(ecdsa.P384(), pub, hash, sig)
. . 517: case elliptic.P521().Params():
. . 518: return verifyFIPS(ecdsa.P521(), pub, hash, sig)
. . 519: default:
```
This eliminates `TemplateSigner.Verify` (~1.75s)
and `crypto/ecdsa.VerifyASN1` (~1.50s) from the
hot path (read: reduces startup time).
Signed-off-by: Dwi Siswanto <git@dw1.io>
486 lines
19 KiB
Go
486 lines
19 KiB
Go
package protocols
|
|
|
|
import (
|
|
"context"
|
|
"encoding/base64"
|
|
"sync/atomic"
|
|
|
|
"github.com/projectdiscovery/fastdialer/fastdialer"
|
|
"github.com/projectdiscovery/gologger"
|
|
"github.com/projectdiscovery/ratelimit"
|
|
mapsutil "github.com/projectdiscovery/utils/maps"
|
|
stringsutil "github.com/projectdiscovery/utils/strings"
|
|
|
|
"github.com/logrusorgru/aurora"
|
|
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/authprovider"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/catalog"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/fuzz/frequency"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/fuzz/stats"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/input"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/js/compiler"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/loader/parser"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/model"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/operators"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/operators/extractors"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/operators/matchers"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/output"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/progress"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/projectfile"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/common/contextargs"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/common/globalmatchers"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/common/hosterrorscache"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/common/interactsh"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/common/utils/excludematchers"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/common/variables"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/protocols/headless/engine"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/reporting"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/scan"
|
|
templateTypes "github.com/projectdiscovery/nuclei/v3/pkg/templates/types"
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/types"
|
|
unitutils "github.com/projectdiscovery/utils/unit"
|
|
)
|
|
|
|
var (
|
|
MaxTemplateFileSizeForEncoding = unitutils.Mega
|
|
)
|
|
|
|
// Executer is an interface implemented any protocol based request executer.
|
|
type Executer interface {
|
|
// Compile compiles the execution generators preparing any requests possible.
|
|
Compile() error
|
|
// Requests returns the total number of requests the rule will perform
|
|
Requests() int
|
|
// Execute executes the protocol group and returns true or false if results were found.
|
|
Execute(ctx *scan.ScanContext) (bool, error)
|
|
// ExecuteWithResults executes the protocol requests and returns results instead of writing them.
|
|
ExecuteWithResults(ctx *scan.ScanContext) ([]*output.ResultEvent, error)
|
|
}
|
|
|
|
// TemplateVerification holds cached verification information for a template.
|
|
type TemplateVerification struct {
|
|
Verified bool
|
|
Verifier string
|
|
}
|
|
|
|
// ExecutorOptions contains the configuration options for executer clients
|
|
type ExecutorOptions struct {
|
|
// TemplateID is the ID of the template for the request
|
|
TemplateID string
|
|
// TemplatePath is the path of the template for the request
|
|
TemplatePath string
|
|
// TemplateInfo contains information block of the template request
|
|
TemplateInfo model.Info
|
|
// TemplateVerifier is the verifier for the template
|
|
TemplateVerifier string
|
|
// TemplateVerificationCallback returns cached verification info for a template path.
|
|
// If it returns nil, verification should be computed normally.
|
|
TemplateVerificationCallback func(templatePath string) *TemplateVerification
|
|
// RawTemplate is the raw template for the request
|
|
RawTemplate []byte
|
|
// Output is a writer interface for writing output events from executer.
|
|
Output output.Writer
|
|
// Options contains configuration options for the executer.
|
|
Options *types.Options
|
|
// IssuesClient is a client for nuclei issue tracker reporting
|
|
IssuesClient reporting.Client
|
|
// Progress is a progress client for scan reporting
|
|
Progress progress.Progress
|
|
// RateLimiter is a rate-limiter for limiting sent number of requests.
|
|
RateLimiter *ratelimit.Limiter
|
|
// Catalog is a template catalog implementation for nuclei
|
|
Catalog catalog.Catalog
|
|
// ProjectFile is the project file for nuclei
|
|
ProjectFile *projectfile.ProjectFile
|
|
// Browser is a browser engine for running headless templates
|
|
Browser *engine.Browser
|
|
// Interactsh is a client for interactsh oob polling server
|
|
Interactsh *interactsh.Client
|
|
// HostErrorsCache is an optional cache for handling host errors
|
|
HostErrorsCache hosterrorscache.CacheInterface
|
|
// Stop execution once first match is found (Assigned while parsing templates)
|
|
// Note: this is different from Options.StopAtFirstMatch (Assigned from CLI option)
|
|
StopAtFirstMatch bool
|
|
// Variables is a list of variables from template
|
|
Variables variables.Variable
|
|
// Constants is a list of constants from template
|
|
Constants map[string]interface{}
|
|
// ExcludeMatchers is the list of matchers to exclude
|
|
ExcludeMatchers *excludematchers.ExcludeMatchers
|
|
// InputHelper is a helper for input normalization
|
|
InputHelper *input.Helper
|
|
// FuzzParamsFrequency is a cache for parameter frequency
|
|
FuzzParamsFrequency *frequency.Tracker
|
|
// FuzzStatsDB is a database for fuzzing stats
|
|
FuzzStatsDB *stats.Tracker
|
|
|
|
Operators []*operators.Operators // only used by offlinehttp module
|
|
|
|
// DoNotCache bool disables optional caching of the templates structure
|
|
DoNotCache bool
|
|
|
|
Colorizer aurora.Aurora
|
|
WorkflowLoader model.WorkflowLoader
|
|
ResumeCfg *types.ResumeCfg
|
|
// ProtocolType is the type of the template
|
|
ProtocolType templateTypes.ProtocolType
|
|
// Flow is execution flow for the template (written in javascript)
|
|
Flow string
|
|
// IsMultiProtocol is true if template has more than one protocol
|
|
IsMultiProtocol bool
|
|
// templateStore is a map which contains template context for each scan (i.e input * template-id pair)
|
|
templateCtxStore *mapsutil.SyncLockMap[string, *contextargs.Context]
|
|
// JsCompiler is abstracted javascript compiler which adds node modules and provides execution
|
|
// environment for javascript templates
|
|
JsCompiler *compiler.Compiler
|
|
// AuthProvider is a provider for auth strategies
|
|
AuthProvider authprovider.AuthProvider
|
|
//TemporaryDirectory is the directory to store temporary files
|
|
TemporaryDirectory string
|
|
Parser parser.Parser
|
|
// ExportReqURLPattern exports the request URL pattern
|
|
// in ResultEvent it contains the exact url pattern (ex: {{BaseURL}}/{{randstr}}/xyz) used in the request
|
|
ExportReqURLPattern bool
|
|
// GlobalMatchers is the storage for global matchers with http passive templates
|
|
GlobalMatchers *globalmatchers.Storage
|
|
// Logger is the shared logging instance
|
|
Logger *gologger.Logger
|
|
// CustomFastdialer is a fastdialer dialer instance
|
|
CustomFastdialer *fastdialer.Dialer
|
|
}
|
|
|
|
// todo: centralizing components is not feasible with current clogged architecture
|
|
// a possible approach could be an internal event bus with pub-subs? This would be less invasive than
|
|
// reworking dep injection from scratch
|
|
func (e *ExecutorOptions) RateLimitTake() {
|
|
// The code below can race and there isn't a great way to fix this without adding an idempotent
|
|
// function to the rate limiter implementation. For now, stick with whatever rate is already set.
|
|
/*
|
|
if e.RateLimiter.GetLimit() != uint(e.Options.RateLimit) {
|
|
e.RateLimiter.SetLimit(uint(e.Options.RateLimit))
|
|
e.RateLimiter.SetDuration(e.Options.RateLimitDuration)
|
|
}
|
|
*/
|
|
if e.RateLimiter != nil {
|
|
e.RateLimiter.Take()
|
|
}
|
|
}
|
|
|
|
// GetThreadsForNPayloadRequests returns the number of threads to use as default for
|
|
// given max-request of payloads
|
|
func (e *ExecutorOptions) GetThreadsForNPayloadRequests(totalRequests int, currentThreads int) int {
|
|
if currentThreads > 0 {
|
|
return currentThreads
|
|
}
|
|
|
|
return e.Options.PayloadConcurrency
|
|
}
|
|
|
|
// CreateTemplateCtxStore creates template context store (which contains templateCtx for every scan)
|
|
func (e *ExecutorOptions) CreateTemplateCtxStore() {
|
|
e.templateCtxStore = &mapsutil.SyncLockMap[string, *contextargs.Context]{
|
|
Map: make(map[string]*contextargs.Context),
|
|
ReadOnly: atomic.Bool{},
|
|
}
|
|
}
|
|
|
|
// RemoveTemplateCtx removes template context of given scan from store
|
|
func (e *ExecutorOptions) RemoveTemplateCtx(input *contextargs.MetaInput) {
|
|
scanId := input.GetScanHash(e.TemplateID)
|
|
if e.templateCtxStore != nil {
|
|
e.templateCtxStore.Delete(scanId)
|
|
}
|
|
}
|
|
|
|
// HasTemplateCtx returns true if template context exists for given input
|
|
func (e *ExecutorOptions) HasTemplateCtx(input *contextargs.MetaInput) bool {
|
|
scanId := input.GetScanHash(e.TemplateID)
|
|
if e.templateCtxStore != nil {
|
|
return e.templateCtxStore.Has(scanId)
|
|
}
|
|
return false
|
|
}
|
|
|
|
// GetTemplateCtx returns template context for given input
|
|
func (e *ExecutorOptions) GetTemplateCtx(input *contextargs.MetaInput) *contextargs.Context {
|
|
scanId := input.GetScanHash(e.TemplateID)
|
|
if e.templateCtxStore == nil {
|
|
// if template context store is not initialized create it
|
|
e.CreateTemplateCtxStore()
|
|
}
|
|
// get template context from store
|
|
templateCtx, ok := e.templateCtxStore.Get(scanId)
|
|
if !ok {
|
|
// if template context does not exist create new and add it to store and return it
|
|
templateCtx = contextargs.New(context.Background())
|
|
templateCtx.MetaInput = input
|
|
_ = e.templateCtxStore.Set(scanId, templateCtx)
|
|
}
|
|
return templateCtx
|
|
}
|
|
|
|
// AddTemplateVars adds vars to template context with given template type as prefix
|
|
// this method is no-op if template is not multi protocol
|
|
func (e *ExecutorOptions) AddTemplateVars(input *contextargs.MetaInput, reqType templateTypes.ProtocolType, reqID string, vars map[string]interface{}) {
|
|
// if we want to disable adding response variables and other variables to template context
|
|
// this is the statement that does it . template context is currently only enabled for
|
|
// multiprotocol and flow templates
|
|
if !e.IsMultiProtocol && e.Flow == "" {
|
|
// no-op if not multi protocol template or flow template
|
|
return
|
|
}
|
|
templateCtx := e.GetTemplateCtx(input)
|
|
for k, v := range vars {
|
|
if stringsutil.HasPrefixAny(k, templateTypes.SupportedProtocolsStrings()...) {
|
|
// this was inherited from previous protocols no need to modify it we can directly set it or omit
|
|
templateCtx.Set(k, v)
|
|
continue
|
|
}
|
|
if !stringsutil.EqualFoldAny(k, "template-id", "template-info", "template-path") {
|
|
if reqID != "" {
|
|
k = reqID + "_" + k
|
|
} else if reqType < templateTypes.InvalidProtocol {
|
|
k = reqType.String() + "_" + k
|
|
}
|
|
templateCtx.Set(k, v)
|
|
}
|
|
}
|
|
}
|
|
|
|
// AddTemplateVar adds given var to template context with given template type as prefix
|
|
// this method is no-op if template is not multi protocol
|
|
func (e *ExecutorOptions) AddTemplateVar(input *contextargs.MetaInput, templateType templateTypes.ProtocolType, reqID string, key string, value interface{}) {
|
|
if !e.IsMultiProtocol && e.Flow == "" {
|
|
// no-op if not multi protocol template or flow template
|
|
return
|
|
}
|
|
templateCtx := e.GetTemplateCtx(input)
|
|
if stringsutil.HasPrefixAny(key, templateTypes.SupportedProtocolsStrings()...) {
|
|
// this was inherited from previous protocols no need to modify it we can directly set it or omit
|
|
templateCtx.Set(key, value)
|
|
return
|
|
}
|
|
if reqID != "" {
|
|
key = reqID + "_" + key
|
|
} else if templateType < templateTypes.InvalidProtocol {
|
|
key = templateType.String() + "_" + key
|
|
}
|
|
templateCtx.Set(key, value)
|
|
}
|
|
|
|
// Copy returns a copy of the executeroptions structure
|
|
func (e *ExecutorOptions) Copy() *ExecutorOptions {
|
|
copy := &ExecutorOptions{
|
|
TemplateID: e.TemplateID,
|
|
TemplatePath: e.TemplatePath,
|
|
TemplateInfo: e.TemplateInfo,
|
|
TemplateVerifier: e.TemplateVerifier,
|
|
TemplateVerificationCallback: e.TemplateVerificationCallback,
|
|
RawTemplate: e.RawTemplate,
|
|
Output: e.Output,
|
|
Options: e.Options,
|
|
IssuesClient: e.IssuesClient,
|
|
Progress: e.Progress,
|
|
RateLimiter: e.RateLimiter,
|
|
Catalog: e.Catalog,
|
|
ProjectFile: e.ProjectFile,
|
|
Browser: e.Browser,
|
|
Interactsh: e.Interactsh,
|
|
HostErrorsCache: e.HostErrorsCache,
|
|
StopAtFirstMatch: e.StopAtFirstMatch,
|
|
Variables: e.Variables,
|
|
Constants: e.Constants,
|
|
ExcludeMatchers: e.ExcludeMatchers,
|
|
InputHelper: e.InputHelper,
|
|
FuzzParamsFrequency: e.FuzzParamsFrequency,
|
|
FuzzStatsDB: e.FuzzStatsDB,
|
|
Operators: e.Operators,
|
|
DoNotCache: e.DoNotCache,
|
|
Colorizer: e.Colorizer,
|
|
WorkflowLoader: e.WorkflowLoader,
|
|
ResumeCfg: e.ResumeCfg,
|
|
ProtocolType: e.ProtocolType,
|
|
Flow: e.Flow,
|
|
IsMultiProtocol: e.IsMultiProtocol,
|
|
JsCompiler: e.JsCompiler,
|
|
AuthProvider: e.AuthProvider,
|
|
TemporaryDirectory: e.TemporaryDirectory,
|
|
Parser: e.Parser,
|
|
ExportReqURLPattern: e.ExportReqURLPattern,
|
|
GlobalMatchers: e.GlobalMatchers,
|
|
Logger: e.Logger,
|
|
}
|
|
copy.CreateTemplateCtxStore()
|
|
return copy
|
|
}
|
|
|
|
// Request is an interface implemented any protocol based request generator.
|
|
type Request interface {
|
|
// Compile compiles the request generators preparing any requests possible.
|
|
Compile(options *ExecutorOptions) error
|
|
// Requests returns the total number of requests the rule will perform
|
|
Requests() int
|
|
// GetID returns the ID for the request if any. IDs are used for multi-request
|
|
// condition matching. So, two requests can be sent and their match can
|
|
// be evaluated from the third request by using the IDs for both requests.
|
|
GetID() string
|
|
// Match performs matching operation for a matcher on model and returns:
|
|
// true and a list of matched snippets if the matcher type is supports it
|
|
// otherwise false and an empty string slice
|
|
Match(data map[string]interface{}, matcher *matchers.Matcher) (bool, []string)
|
|
// Extract performs extracting operation for an extractor on model and returns true or false.
|
|
Extract(data map[string]interface{}, matcher *extractors.Extractor) map[string]struct{}
|
|
// ExecuteWithResults executes the protocol requests and returns results instead of writing them.
|
|
ExecuteWithResults(input *contextargs.Context, dynamicValues, previous output.InternalEvent, callback OutputEventCallback) error
|
|
// MakeResultEventItem creates a result event from internal wrapped event. Intended to be used by MakeResultEventItem internally
|
|
MakeResultEventItem(wrapped *output.InternalWrappedEvent) *output.ResultEvent
|
|
// MakeResultEvent creates a flat list of result events from an internal wrapped event, based on successful matchers and extracted data
|
|
MakeResultEvent(wrapped *output.InternalWrappedEvent) []*output.ResultEvent
|
|
// GetCompiledOperators returns a list of the compiled operators
|
|
GetCompiledOperators() []*operators.Operators
|
|
// Type returns the type of the protocol request
|
|
Type() templateTypes.ProtocolType
|
|
}
|
|
|
|
// OutputEventCallback is a callback event for any results found during scanning.
|
|
type OutputEventCallback func(result *output.InternalWrappedEvent)
|
|
|
|
func MakeDefaultResultEvent(request Request, wrapped *output.InternalWrappedEvent) []*output.ResultEvent {
|
|
// Note: operator result is generated if something was successful match/extract/dynamic-extract
|
|
// but results should not be generated if
|
|
// 1. no match was found and some dynamic values were extracted
|
|
// 2. if something was extracted (matchers exist but no match was found)
|
|
if len(wrapped.OperatorsResult.DynamicValues) > 0 && !wrapped.OperatorsResult.Matched {
|
|
return nil
|
|
}
|
|
// check if something was extracted (except dynamic values)
|
|
extracted := len(wrapped.OperatorsResult.Extracts) > 0 || len(wrapped.OperatorsResult.OutputExtracts) > 0
|
|
if extracted && len(wrapped.OperatorsResult.Operators.Matchers) > 0 && !wrapped.OperatorsResult.Matched {
|
|
// if extracted and matchers exist but no match was found then don't generate result
|
|
return nil
|
|
}
|
|
|
|
results := make([]*output.ResultEvent, 0, len(wrapped.OperatorsResult.Matches)+1)
|
|
|
|
// If we have multiple matchers with names, write each of them separately.
|
|
if len(wrapped.OperatorsResult.Matches) > 0 {
|
|
for matcherNames := range wrapped.OperatorsResult.Matches {
|
|
data := request.MakeResultEventItem(wrapped)
|
|
data.MatcherName = matcherNames
|
|
results = append(results, data)
|
|
}
|
|
} else if len(wrapped.OperatorsResult.Extracts) > 0 {
|
|
for k, v := range wrapped.OperatorsResult.Extracts {
|
|
data := request.MakeResultEventItem(wrapped)
|
|
data.ExtractorName = k
|
|
data.ExtractedResults = v
|
|
results = append(results, data)
|
|
}
|
|
} else {
|
|
data := request.MakeResultEventItem(wrapped)
|
|
results = append(results, data)
|
|
}
|
|
return results
|
|
}
|
|
|
|
// MakeDefaultExtractFunc performs extracting operation for an extractor on model and returns true or false.
|
|
func MakeDefaultExtractFunc(data map[string]interface{}, extractor *extractors.Extractor) map[string]struct{} {
|
|
part := extractor.Part
|
|
if part == "" {
|
|
part = "response"
|
|
}
|
|
|
|
item, ok := data[part]
|
|
if !ok && !extractors.SupportsMap(extractor) {
|
|
return nil
|
|
}
|
|
itemStr := types.ToString(item)
|
|
|
|
switch extractor.GetType() {
|
|
case extractors.RegexExtractor:
|
|
return extractor.ExtractRegex(itemStr)
|
|
case extractors.KValExtractor:
|
|
return extractor.ExtractKval(data)
|
|
case extractors.JSONExtractor:
|
|
return extractor.ExtractJSON(itemStr)
|
|
case extractors.XPathExtractor:
|
|
return extractor.ExtractXPath(itemStr)
|
|
case extractors.DSLExtractor:
|
|
return extractor.ExtractDSL(data)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// MakeDefaultMatchFunc performs matching operation for a matcher on model and returns true or false.
|
|
func MakeDefaultMatchFunc(data map[string]interface{}, matcher *matchers.Matcher) (bool, []string) {
|
|
part := matcher.Part
|
|
if part == "" {
|
|
part = "response"
|
|
}
|
|
|
|
partItem, ok := data[part]
|
|
if !ok && matcher.Type.MatcherType != matchers.DSLMatcher {
|
|
return false, nil
|
|
}
|
|
item := types.ToString(partItem)
|
|
|
|
switch matcher.GetType() {
|
|
case matchers.SizeMatcher:
|
|
result := matcher.Result(matcher.MatchSize(len(item)))
|
|
return result, nil
|
|
case matchers.WordsMatcher:
|
|
return matcher.ResultWithMatchedSnippet(matcher.MatchWords(item, nil))
|
|
case matchers.RegexMatcher:
|
|
return matcher.ResultWithMatchedSnippet(matcher.MatchRegex(item))
|
|
case matchers.BinaryMatcher:
|
|
return matcher.ResultWithMatchedSnippet(matcher.MatchBinary(item))
|
|
case matchers.DSLMatcher:
|
|
return matcher.Result(matcher.MatchDSL(data)), nil
|
|
case matchers.XPathMatcher:
|
|
return matcher.Result(matcher.MatchXPath(item)), []string{}
|
|
}
|
|
return false, nil
|
|
}
|
|
|
|
func (e *ExecutorOptions) EncodeTemplate() string {
|
|
if !e.Options.OmitTemplate && len(e.RawTemplate) <= MaxTemplateFileSizeForEncoding {
|
|
return base64.StdEncoding.EncodeToString(e.RawTemplate)
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// ApplyNewEngineOptions updates an existing ExecutorOptions with options from a new engine. This
|
|
// handles things like the ExecutionID that need to be updated.
|
|
func (e *ExecutorOptions) ApplyNewEngineOptions(n *ExecutorOptions) {
|
|
// TODO: cached code|headless templates have nil ExecuterOptions if -code or -headless are not enabled
|
|
if e == nil || n == nil || n.Options == nil {
|
|
return
|
|
}
|
|
|
|
e.Options = n.Options.Copy()
|
|
e.Output = n.Output
|
|
e.IssuesClient = n.IssuesClient
|
|
e.Progress = n.Progress
|
|
e.RateLimiter = n.RateLimiter
|
|
e.Catalog = n.Catalog
|
|
e.ProjectFile = n.ProjectFile
|
|
e.Browser = n.Browser
|
|
e.Interactsh = n.Interactsh
|
|
e.HostErrorsCache = n.HostErrorsCache
|
|
e.InputHelper = n.InputHelper
|
|
e.FuzzParamsFrequency = n.FuzzParamsFrequency
|
|
e.FuzzStatsDB = n.FuzzStatsDB
|
|
e.DoNotCache = n.DoNotCache
|
|
e.Colorizer = n.Colorizer
|
|
e.WorkflowLoader = n.WorkflowLoader
|
|
e.ResumeCfg = n.ResumeCfg
|
|
e.JsCompiler = n.JsCompiler
|
|
e.AuthProvider = n.AuthProvider
|
|
e.TemporaryDirectory = n.TemporaryDirectory
|
|
e.Parser = n.Parser
|
|
e.ExportReqURLPattern = n.ExportReqURLPattern
|
|
e.GlobalMatchers = n.GlobalMatchers
|
|
e.Logger = n.Logger
|
|
e.CustomFastdialer = n.CustomFastdialer
|
|
}
|