mirror of
https://github.com/projectdiscovery/nuclei.git
synced 2025-12-17 22:15:27 +00:00
* feat: global matchers Signed-off-by: Dwi Siswanto <git@dw1.io> Co-authored-by: Ice3man543 <ice3man543@users.noreply.github.com> * feat(globalmatchers): make `Callback` as type Signed-off-by: Dwi Siswanto <git@dw1.io> * feat: update `passive` term to `(matchers-)static` Signed-off-by: Dwi Siswanto <git@dw1.io> * feat(globalmatchers): add `origin-template-*` event also use `Set` method instead of `maps.Clone` Signed-off-by: Dwi Siswanto <git@dw1.io> * feat: update `matchers-static` term to `global-matchers` Signed-off-by: Dwi Siswanto <git@dw1.io> * feat(globalmatchers): clone event before `operator.Execute` Signed-off-by: Dwi Siswanto <git@dw1.io> * fix(tmplexec): don't store `matched` on `global-matchers` templ This will end up generating 2 events from the same `scan.ScanContext` if one of the templates has `global-matchers` enabled. This way, non- `global-matchers` templates can enter the `writeFailureCallback` func to log failure output. Signed-off-by: Dwi Siswanto <git@dw1.io> * feat(globalmatchers): initializes `requests` on `New` Signed-off-by: Dwi Siswanto <git@dw1.io> * feat(globalmatchers): add `hasStorage` method Signed-off-by: Dwi Siswanto <git@dw1.io> * refactor(templates): rename global matchers checks method Signed-off-by: Dwi Siswanto <git@dw1.io> * fix(loader): handle nil `templates.Template` pointer Signed-off-by: Dwi Siswanto <git@dw1.io> --------- Signed-off-by: Dwi Siswanto <git@dw1.io> Co-authored-by: Ice3man543 <ice3man543@users.noreply.github.com>
132 lines
3.4 KiB
Go
132 lines
3.4 KiB
Go
package output
|
|
|
|
import (
|
|
"bytes"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/projectdiscovery/nuclei/v3/pkg/types"
|
|
mapsutil "github.com/projectdiscovery/utils/maps"
|
|
)
|
|
|
|
// formatScreen formats the output for showing on screen.
|
|
func (w *StandardWriter) formatScreen(output *ResultEvent) []byte {
|
|
builder := &bytes.Buffer{}
|
|
|
|
if !w.noMetadata {
|
|
if w.timestamp {
|
|
builder.WriteRune('[')
|
|
builder.WriteString(w.aurora.Cyan(output.Timestamp.Format("2006-01-02 15:04:05")).String())
|
|
builder.WriteString("] ")
|
|
}
|
|
builder.WriteRune('[')
|
|
builder.WriteString(w.aurora.BrightGreen(output.TemplateID).String())
|
|
|
|
if output.MatcherName != "" {
|
|
builder.WriteString(":")
|
|
builder.WriteString(w.aurora.BrightGreen(output.MatcherName).Bold().String())
|
|
} else if output.ExtractorName != "" {
|
|
builder.WriteString(":")
|
|
builder.WriteString(w.aurora.BrightGreen(output.ExtractorName).Bold().String())
|
|
}
|
|
|
|
if w.matcherStatus {
|
|
builder.WriteString("] [")
|
|
if !output.MatcherStatus {
|
|
builder.WriteString(w.aurora.Red("failed").String())
|
|
} else {
|
|
builder.WriteString(w.aurora.Green("matched").String())
|
|
}
|
|
}
|
|
|
|
if output.GlobalMatchers {
|
|
builder.WriteString("] [")
|
|
builder.WriteString(w.aurora.BrightMagenta("global").String())
|
|
}
|
|
|
|
builder.WriteString("] [")
|
|
builder.WriteString(w.aurora.BrightBlue(output.Type).String())
|
|
builder.WriteString("] ")
|
|
|
|
builder.WriteString("[")
|
|
builder.WriteString(w.severityColors(output.Info.SeverityHolder.Severity))
|
|
builder.WriteString("] ")
|
|
}
|
|
if output.Matched != "" {
|
|
builder.WriteString(output.Matched)
|
|
} else {
|
|
builder.WriteString(output.Host)
|
|
}
|
|
|
|
// If any extractors, write the results
|
|
if len(output.ExtractedResults) > 0 {
|
|
builder.WriteString(" [")
|
|
|
|
for i, item := range output.ExtractedResults {
|
|
// trim trailing space
|
|
// quote non-ascii and non printable characters and then
|
|
// unquote quotes (`"`) for readability
|
|
item = strings.TrimSpace(item)
|
|
item = strconv.QuoteToASCII(item)
|
|
item = strings.ReplaceAll(item, `\"`, `"`)
|
|
|
|
builder.WriteString(w.aurora.BrightCyan(item).String())
|
|
|
|
if i != len(output.ExtractedResults)-1 {
|
|
builder.WriteRune(',')
|
|
}
|
|
}
|
|
builder.WriteString("]")
|
|
}
|
|
|
|
if len(output.Lines) > 0 {
|
|
builder.WriteString(" [LN: ")
|
|
|
|
for i, line := range output.Lines {
|
|
builder.WriteString(strconv.Itoa(line))
|
|
|
|
if i != len(output.Lines)-1 {
|
|
builder.WriteString(",")
|
|
}
|
|
}
|
|
builder.WriteString("]")
|
|
}
|
|
|
|
// Write meta if any
|
|
if len(output.Metadata) > 0 {
|
|
builder.WriteString(" [")
|
|
|
|
first := true
|
|
// sort to get predictable output
|
|
for _, name := range mapsutil.GetSortedKeys(output.Metadata) {
|
|
value := output.Metadata[name]
|
|
if !first {
|
|
builder.WriteRune(',')
|
|
}
|
|
first = false
|
|
|
|
builder.WriteString(w.aurora.BrightYellow(name).String())
|
|
builder.WriteRune('=')
|
|
builder.WriteString(w.aurora.BrightYellow(strconv.QuoteToASCII(types.ToString(value))).String())
|
|
}
|
|
builder.WriteString("]")
|
|
}
|
|
|
|
// If it is a fuzzing output, enrich with additional
|
|
// metadata for the match.
|
|
if output.IsFuzzingResult {
|
|
if output.FuzzingParameter != "" {
|
|
builder.WriteString(" [")
|
|
builder.WriteString(output.FuzzingPosition)
|
|
builder.WriteRune(':')
|
|
builder.WriteString(w.aurora.BrightMagenta(output.FuzzingParameter).String())
|
|
builder.WriteString("]")
|
|
}
|
|
|
|
builder.WriteString(" [")
|
|
builder.WriteString(output.FuzzingMethod)
|
|
builder.WriteString("]")
|
|
}
|
|
return builder.Bytes()
|
|
}
|