mirror of
https://github.com/valitydev/SigmaHQ.git
synced 2024-11-07 17:58:52 +00:00
eb022f3908
Fixes #326
106 lines
4.4 KiB
Python
106 lines
4.4 KiB
Python
# NetWitness output backend for sigmac
|
|
# Copyright 2018 John Tuckner (@tuckner)
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Lesser General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Lesser General Public License for more details.
|
|
|
|
# You should have received a copy of the GNU Lesser General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
import re
|
|
import sigma
|
|
from .base import SingleTextQueryBackend
|
|
from .mixins import MultiRuleOutputMixin
|
|
|
|
|
|
class NetWitnessBackend(SingleTextQueryBackend):
|
|
"""Converts Sigma rule into NetWitness saved search. Contributed by @tuckner"""
|
|
identifier = "netwitness"
|
|
active = True
|
|
reEscape = re.compile('(")')
|
|
reClear = None
|
|
andToken = " && "
|
|
orToken = " || "
|
|
notToken = "NOT"
|
|
subExpression = "(%s)"
|
|
listExpression = "(%s)"
|
|
listSeparator = ", "
|
|
valueExpression = "\'%s\'"
|
|
keyExpression = "%s"
|
|
nullExpression = "%s exists"
|
|
notNullExpression = "%s exists"
|
|
mapExpression = "(%s=%s)"
|
|
mapListsSpecialHandling = True
|
|
|
|
def generateMapItemNode(self, node):
|
|
key, value = node
|
|
if self.mapListsSpecialHandling == False and type(value) in (str, int, list) or self.mapListsSpecialHandling == True and type(value) in (str, int):
|
|
if type(value) == str and "*" in value[1:-1]:
|
|
value = re.sub('([".^$]|\\\\(?![*?]))', '\\\\\g<1>', value)
|
|
value = re.sub('\\*', '.*', value)
|
|
value = re.sub('\\?', '.', value)
|
|
return "(%s regex %s)" %(key, self.generateValueNode(value))
|
|
elif type(value) == str and "*" in value:
|
|
value = re.sub("(\*\\\\)|(\*)", "", value)
|
|
return "(%s contains %s)" % (key, self.generateValueNode(value))
|
|
elif type(value) in (str, int):
|
|
return self.mapExpression % (key, self.generateValueNode(value))
|
|
else:
|
|
return self.mapExpression % (key, self.generateNode(value))
|
|
elif type(value) == list:
|
|
return self.generateMapItemListNode(key, value)
|
|
elif value is None:
|
|
return self.nullExpression % (key, )
|
|
else:
|
|
raise TypeError("Backend does not support map values of type " + str(type(value)))
|
|
|
|
def generateMapItemListNode(self, key, value):
|
|
equallist = list()
|
|
containlist = list()
|
|
regexlist = list()
|
|
for item in value:
|
|
if type(item) == str and "*" in item[1:-1]:
|
|
item = re.sub('([".^$]|\\\\(?![*?]))', '\\\\\g<1>', item)
|
|
item = re.sub('\\*', '.*', item)
|
|
item = re.sub('\\?', '.', item)
|
|
regexlist.append(self.generateValueNode(item))
|
|
elif type(item) == str and (item.endswith("*") or item.startswith("*")):
|
|
item = re.sub('([".^$]|\\\\(?![*?]))', '\\\\\g<1>', item)
|
|
item = re.sub("(\*\\\\)|(\*)", "", item)
|
|
containlist.append(self.generateValueNode(item))
|
|
else:
|
|
equallist.append(self.generateValueNode(item))
|
|
fmtitems = list()
|
|
if equallist:
|
|
fmtitems.append("%s = %s" % (key, ", ".join(equallist)))
|
|
if containlist:
|
|
fmtitems.append("%s contains %s" % (key, ", ".join(containlist)))
|
|
if regexlist:
|
|
fmtitems.append("%s regex %s" % (key, ", ".join(regexlist)))
|
|
fmtquery = "("+" || ".join(filter(None, fmtitems))+")"
|
|
return fmtquery
|
|
|
|
def generateValueNode(self, node):
|
|
return self.valueExpression % (str(node))
|
|
|
|
def generate(self, sigmaparser):
|
|
"""Method is called for each sigma rule and receives the parsed rule (SigmaParser)"""
|
|
for k in sigmaparser.parsedyaml["detection"].keys():
|
|
if k.startswith('keyword'):
|
|
raise NotImplementedError("Backend does not support keywords")
|
|
for parsed in sigmaparser.condparsed:
|
|
query = self.generateQuery(parsed, sigmaparser)
|
|
return query
|
|
|
|
def generateQuery(self, parsed, sigmaparser):
|
|
result = self.generateNode(parsed.parsedSearch)
|
|
return result
|