-
Notifications
You must be signed in to change notification settings - Fork 500
/
test_hunt_data.py
120 lines (98 loc) · 4.63 KB
/
test_hunt_data.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
# Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
# or more contributor license agreements. Licensed under the Elastic License
# 2.0; you may not use this file except in compliance with the Elastic License
# 2.0.
"""Test for hunt toml files."""
import unittest
from hunting.definitions import HUNTING_DIR
from hunting.markdown import load_toml
from hunting.utils import load_index_file, load_all_toml
class TestHunt(unittest.TestCase):
"""Test hunt toml files."""
def test_toml_loading(self):
"""Test loading a hunt toml file content."""
example_toml = """
[hunt]
author = "Elastic"
description = "Detects denial of service or resource exhaustion attacks."
integration = "aws_bedrock.invocation"
uuid = "dc181967-c32c-46c9-b84b-ec4c8811c6a0"
name = "Denial of Service or Resource Exhaustion Attacks Detection"
language = "ES|QL"
license = "Elastic License v2"
query = ['SELECT * FROM logs']
notes = ["High token usage can strain system resources."]
mitre = ["AML.T0034"]
references = ["https://www.elastic.co"]
"""
config = load_toml(example_toml)
self.assertEqual(config.author, "Elastic")
self.assertEqual(config.integration, "aws_bedrock.invocation")
self.assertEqual(
config.name, "Denial of Service or Resource Exhaustion Attacks Detection"
)
self.assertEqual(config.language, "ES|QL")
def test_load_toml_files(self):
"""Test loading and validating all Hunt TOML files in the hunting directory."""
for toml_path in HUNTING_DIR.rglob("*.toml"):
hunt = load_toml(toml_path)
self.assertTrue(hunt.author)
self.assertTrue(hunt.description)
self.assertTrue(hunt.integration)
self.assertTrue(hunt.name)
self.assertTrue(hunt.language)
self.assertTrue(hunt.query)
def test_markdown_existence(self):
"""Ensure each TOML file has a corresponding Markdown file in the docs directory."""
for toml_file in HUNTING_DIR.rglob("*.toml"):
expected_markdown_path = (
toml_file.parent.parent / "docs" / toml_file.with_suffix(".md").name
)
self.assertTrue(
expected_markdown_path.exists(),
f"Markdown file not found for {toml_file} at expected location {expected_markdown_path}",
)
def test_toml_existence(self):
"""Ensure each Markdown file has a corresponding TOML file in the queries directory."""
for markdown_file in HUNTING_DIR.rglob("*/docs/*.md"):
expected_toml_path = (
markdown_file.parent.parent / "queries" / markdown_file.with_suffix(".toml").name
)
self.assertTrue(
expected_toml_path.exists(),
f"TOML file not found for {markdown_file} at expected location {expected_toml_path}",
)
class TestHuntIndex(unittest.TestCase):
"""Test the hunting index.yml file."""
@classmethod
def setUpClass(cls):
"""Load the index once for all tests."""
cls.hunting_index = load_index_file()
def test_mitre_techniques_present(self):
"""Ensure each query has at least one MITRE technique."""
for folder, queries in self.hunting_index.items():
for query_uuid, query_data in queries.items():
self.assertTrue(query_data.get('mitre'),
f"No MITRE techniques found for query: {query_data.get('name', query_uuid)}")
def test_valid_structure(self):
"""Ensure each query entry has a valid structure."""
required_fields = ['name', 'path', 'mitre']
for folder, queries in self.hunting_index.items():
for query_uuid, query_data in queries.items():
for field in required_fields:
self.assertIn(field, query_data, f"Missing field '{field}' in query: {query_data}")
def test_all_files_in_index(self):
"""Ensure all TOML files are included in the index."""
missing_index_entries = []
all_toml_data = load_all_toml(HUNTING_DIR)
uuids = [hunt.uuid for hunt, path in all_toml_data]
for folder, queries in self.hunting_index.items():
for query_uuid in queries:
if query_uuid not in uuids:
missing_index_entries.append(query_uuid)
self.assertFalse(
missing_index_entries,
f"Missing index entries for the following queries: {missing_index_entries}"
)
if __name__ == "__main__":
unittest.main()