Files
sigma-rules/tests/test_hunt_data.py
T
Terrance DeJesus 50e23ba242 [Hunting] Re-factor Hunting Library Code (#4085)
* updating python code for hunting library

* fixed okta queries; added MITRE search capability

* fixed hunting unit test imports

* fixed duplicate UUID; fixed duplicate index entry bug

* fixed technique finding sub-technique in search

* added more unit tests

* linted

* flake errors addressed; fixed unit test import; fixed markdown generate bug

* added description for generate-markdown command

* updated README

* adjusted YAML index, adjusted code for index changes

* adjusted relative imports; updated CODEOWNERS

* adding updates; moving to different branch for main dependencies

* finished run-query command; made some code adjustments

* removed some comments

* revised makefile; fixed unit tests; adjusted detection rules pyproject

* updated README

* updated README

* adjusted unit tests; adjusted hunt guidelines; updated makefile; adjusted several commands

* adjusted package to be more object-oriented

* removed unused variable

* Add simple breakdown stats

* addressed feedback; added keyword option for search

* Update hunting/README.md

Co-authored-by: Mika Ayenson <Mikaayenson@users.noreply.github.com>

* Update detection_rules/etc/test_hunting_cli.bash

Co-authored-by: Eric Forte <119343520+eric-forte-elastic@users.noreply.github.com>

* addressing feedback

* addressed feedback

* added message for unknown index; fixed function call

* fixed search command

* fixed flake error

---------

Co-authored-by: Mika Ayenson <Mika.ayenson@elastic.co>
Co-authored-by: Mika Ayenson <Mikaayenson@users.noreply.github.com>
Co-authored-by: Eric Forte <119343520+eric-forte-elastic@users.noreply.github.com>
2024-10-03 12:47:40 -04:00

121 lines
4.6 KiB
Python

# 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()