Skip to content

Commit

Permalink
Better handling of overrides after all fr.Behaviors are added (#487)
Browse files Browse the repository at this point in the history
* Better handling of overrides after all fr.Behaviors are added

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Appease the linter

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Fix scan override behavior, use original rule's name + ID

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Fix overall risk score after downgrading

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Retain original rule, drop override, remove test rules

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Allow for severity upgrades as well

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Comments

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Address PR comments

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Appease the linter

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Remove unused struct

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Only pass in the behavior slices

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

* Fix rule that was causing the archive scan test to break

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>

---------

Signed-off-by: egibs <20933572+egibs@users.noreply.github.com>
  • Loading branch information
egibs authored Oct 4, 2024
1 parent 0acb2e0 commit ab8a15b
Show file tree
Hide file tree
Showing 3 changed files with 86 additions and 33 deletions.
5 changes: 5 additions & 0 deletions pkg/malcontent/malcontent.go
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,9 @@ type Behavior struct {

// Name is the value of m.Rule
RuleName string `json:",omitempty" yaml:",omitempty"`

// The name of the rule this behavior overrides
Override string `json:",omitempty" yaml:",omitempty"`
}

type FileReport struct {
Expand Down Expand Up @@ -88,6 +91,8 @@ type FileReport struct {
RiskLevel string `json:",omitempty" yaml:",omitempty"`

IsMalcontent bool `json:",omitempty" yaml:",omitempty"`

Overrides []*Behavior `json:",omitempty" yaml:",omitempty"`
}

type DiffReport struct {
Expand Down
112 changes: 80 additions & 32 deletions pkg/report/report.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,10 +22,17 @@ import (
"github.com/hillu/go-yara/v4"
)

const NAME string = "malcontent"

const (
NAME string = "malcontent"
HARMLESS int = iota
LOW
MEDIUM
HIGH
CRITICAL
)

// Map to handle RiskScore -> RiskLevel conversions.
var RiskLevels = map[int]string{
0: "NONE", // harmless: common to all executables, no system impact
1: "LOW", // undefined: low impact, common to good and bad executables
Expand Down Expand Up @@ -79,8 +86,10 @@ var threatHuntingKeywordRe = regexp.MustCompile(`Detection patterns for the tool

var dateRe = regexp.MustCompile(`[a-z]{3}\d{1,2}`)

// Map to handle RiskLevel -> RiskScore conversions.
var Levels = map[string]int{
"harmless": 0,
"low": 1,
"notable": 2,
"medium": 2,
"suspicious": 3,
Expand Down Expand Up @@ -321,6 +330,7 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
Size: size,
Meta: map[string]string{},
Behaviors: []*malcontent.Behavior{},
Overrides: []*malcontent.Behavior{},
}

var pledges []string
Expand All @@ -343,6 +353,12 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
}
}

// Store match rules in a map for future override operations
mrsMap := make(map[string]yara.MatchRule)
for _, m := range mrs {
mrsMap[m.Rule] = m
}

for _, m := range mrs {
override := false
if all(m.Rule == NAME, ignoreSelf) {
Expand All @@ -368,7 +384,7 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
switch {
case risk < minScore && !ignoreMalcontent:
continue
case c.Scan && risk < highestRisk && !ignoreMalcontent:
case c.Scan && risk < highestRisk && !ignoreMalcontent && !override:
continue
}
key = generateKey(m.Namespace, m.Rule)
Expand All @@ -386,12 +402,6 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
k := ""
v := ""

// Store match rules in a map for future override operations
mrsMap := make(map[string]yara.MatchRule)
for _, m := range mrs {
mrsMap[m.Rule] = m
}

for _, meta := range m.Metas {
k = meta.Identifier
v = fmt.Sprintf("%s", meta.Value)
Expand All @@ -401,10 +411,16 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
}

// If we find a match in the map for the metadata key, that's the rule to override
// Ensure that the override tag is present on the override rule
var overrideRule string
if match, exists := mrsMap[k]; exists && override {
overrideRule = match.Rule
// Store this rule (the override) in the fr.Overrides behavior slice
if _, exists := mrsMap[k]; exists && override {
var overrideSev int
if sev, ok := Levels[v]; ok {
overrideSev = sev
}
b.RiskLevel = RiskLevels[overrideSev]
b.RiskScore = overrideSev
b.Override = k
fr.Overrides = append(fr.Overrides, b)
}

switch k {
Expand Down Expand Up @@ -449,26 +465,6 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
syscalls = append(syscalls, sy...)
case "cap":
caps = append(caps, v)
// If we find a rule to override, pull in that rule's configuration and update the severity
case overrideRule:
var overrideSev int
if sev, ok := Levels[v]; ok {
overrideSev = sev
}
// Find the behavior for the overridden (original) rule
// Store its behavior in the current behavior and remove the original behavior
for i, ob := range fr.Behaviors {
if ob.RuleName == overrideRule {
b = ob
b.RuleName = m.Rule
b.Description = fmt.Sprintf("%s, [%s]", b.Description, m.Rule)
b.RiskScore = overrideSev
b.RiskLevel = RiskLevels[overrideSev]

// Remove the original rule from the behaviors slice
fr.Behaviors = slices.Delete(fr.Behaviors, i, i+1)
}
}
}
}

Expand Down Expand Up @@ -521,6 +517,20 @@ func Generate(ctx context.Context, path string, mrs yara.MatchRules, c malconten
// TODO: If we match multiple rules within a single namespace, merge matchstrings
}

// Update the behaviors to account for overrides
fr.Behaviors = handleOverrides(fr.Behaviors, fr.Overrides)

// Adjust the overall risk if we deviated from overallRiskScore
// Scans will still need to drop <= medium results
newRisk := highestBehaviorRisk(fr)
if overallRiskScore != newRisk {
overallRiskScore = newRisk
}

if c.Scan && overallRiskScore < HIGH {
return malcontent.FileReport{}, nil
}

// Check for both the full and shortened variants of malcontent
isMalBinary := (filepath.Base(path) == NAME || filepath.Base(path) == "mal")

Expand Down Expand Up @@ -614,3 +624,41 @@ func highestMatchRisk(mrs yara.MatchRules) int {
}
return highestRisk
}

// highestBehaviorRisk returns the highest risk score from a slice of FileReport Behaviors.
func highestBehaviorRisk(fr malcontent.FileReport) int {
if len(fr.Behaviors) == 0 {
return 0
}

highestRisk := 0
for _, b := range fr.Behaviors {
if b.RiskScore > highestRisk {
highestRisk = b.RiskScore
}
}

return highestRisk
}

// handleOverrides modifies the behavior slice based on the contents of the override slice.
func handleOverrides(original, override []*malcontent.Behavior) []*malcontent.Behavior {
behaviorMap := make(map[string]*malcontent.Behavior, len(original))
for _, b := range original {
behaviorMap[b.RuleName] = b
}

for _, o := range override {
if b, exists := behaviorMap[o.Override]; exists {
b.RiskLevel = o.RiskLevel
b.RiskScore = o.RiskScore
}
}

modified := make([]*malcontent.Behavior, 0, len(behaviorMap))
for _, b := range behaviorMap {
modified = append(modified, b)
}

return modified
}
2 changes: 1 addition & 1 deletion rules/net/dns-servers.yara
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
rule go_dns_refs {
rule go_dns_refs_local {
meta:
description = "Examines local DNS servers"
strings:
Expand Down

0 comments on commit ab8a15b

Please sign in to comment.