mirror of
https://github.com/mukul975/Anthropic-Cybersecurity-Skills.git
synced 2026-06-12 22:24:56 +03:00
c47eed6a64
- Fix 25 shell=True subprocess calls with list-based commands - Fix 49 verify=False in defensive skills (env-var override) - Add timeout to 231 HTTP/subprocess/socket calls - Fix 6 SQL injection patterns with whitelist validation - Replace 8 __import__() with standard imports - Remove 701 unused imports across 442 files - Add authorized-testing disclaimers to all offensive skills - Complete 11 incomplete skill directories - Expand 10 stub SKILL.md files with full content - Fix 2 YAML parse errors in frontmatter - Fix 5 pre-existing syntax errors - Convert 22 hardcoded paths/ports to environment variables - Back up 21 redundant skill pairs to .bak - Fix 2 global declaration errors - 724/724 skills with full folder anatomy (SKILL.md + agent.py + api-reference.md + LICENSE) - 0 compile errors across all 724 agent.py files
139 lines
5.6 KiB
Python
139 lines
5.6 KiB
Python
#!/usr/bin/env python3
|
|
"""Agent for Active Directory attack path analysis using BloodHound data collection."""
|
|
|
|
import argparse
|
|
import json
|
|
import os
|
|
import subprocess
|
|
from datetime import datetime, timezone
|
|
|
|
|
|
def run_sharphound(domain, username=None, password=None, collection="All"):
|
|
"""Execute SharpHound data collection."""
|
|
cmd = ["SharpHound.exe", "-c", collection, "-d", domain]
|
|
if username:
|
|
cmd.extend(["--ldapusername", username])
|
|
if password:
|
|
cmd.extend(["--ldappassword", password])
|
|
try:
|
|
result = subprocess.check_output(cmd, text=True, errors="replace", timeout=120)
|
|
return {"status": "success", "output": result[:500]}
|
|
except (subprocess.SubprocessError, FileNotFoundError):
|
|
return {"status": "failed", "note": "SharpHound.exe not found or execution failed"}
|
|
|
|
|
|
def run_bloodhound_python(domain, username, password, dc_ip, collection="all"):
|
|
"""Execute bloodhound-python for cross-platform collection."""
|
|
cmd = [
|
|
"bloodhound-python", "-d", domain, "-u", username, "-p", password,
|
|
"-c", collection, "--zip", "-ns", dc_ip,
|
|
]
|
|
try:
|
|
result = subprocess.check_output(cmd, text=True, errors="replace", timeout=120)
|
|
return {"status": "success", "output": result[:500]}
|
|
except (subprocess.SubprocessError, FileNotFoundError):
|
|
return {"status": "failed", "note": "bloodhound-python not found"}
|
|
|
|
|
|
def analyze_bloodhound_json(data_dir):
|
|
"""Parse BloodHound JSON output for high-value findings."""
|
|
findings = {"users": 0, "computers": 0, "groups": 0, "domains": 0, "attack_paths": []}
|
|
for fname in os.listdir(data_dir):
|
|
fpath = os.path.join(data_dir, fname)
|
|
if not fname.endswith(".json"):
|
|
continue
|
|
try:
|
|
with open(fpath, "r") as f:
|
|
data = json.load(f)
|
|
if "users" in fname.lower():
|
|
users = data.get("data", [])
|
|
findings["users"] = len(users)
|
|
for u in users:
|
|
props = u.get("Properties", {})
|
|
if props.get("admincount"):
|
|
findings["attack_paths"].append({
|
|
"type": "privileged_user",
|
|
"name": props.get("name", ""),
|
|
"enabled": props.get("enabled", False),
|
|
})
|
|
elif "computers" in fname.lower():
|
|
findings["computers"] = len(data.get("data", []))
|
|
elif "groups" in fname.lower():
|
|
findings["groups"] = len(data.get("data", []))
|
|
except (json.JSONDecodeError, KeyError):
|
|
pass
|
|
return findings
|
|
|
|
|
|
def query_neo4j(query, uri="bolt://localhost:7687", user="neo4j", password="bloodhound"):
|
|
"""Execute Cypher query against BloodHound Neo4j database."""
|
|
try:
|
|
from neo4j import GraphDatabase
|
|
driver = GraphDatabase.driver(uri, auth=(user, password))
|
|
with driver.session() as session:
|
|
result = session.run(query)
|
|
records = [dict(r) for r in result]
|
|
driver.close()
|
|
return records
|
|
except ImportError:
|
|
return [{"error": "neo4j driver not installed: pip install neo4j"}]
|
|
except Exception as e:
|
|
return [{"error": str(e)}]
|
|
|
|
|
|
ATTACK_PATH_QUERIES = {
|
|
"shortest_to_da": "MATCH p=shortestPath((u:User {owned:true})-[*1..]->(g:Group {name:'DOMAIN ADMINS@DOMAIN.LOCAL'})) RETURN p",
|
|
"kerberoastable": "MATCH (u:User) WHERE u.hasspn=true AND u.enabled=true RETURN u.name, u.serviceprincipalnames",
|
|
"unconstrained_delegation": "MATCH (c:Computer {unconstraineddelegation:true}) RETURN c.name",
|
|
"dcsync_rights": "MATCH p=(u)-[:GetChanges|GetChangesAll]->(d:Domain) RETURN u.name, d.name",
|
|
}
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser(
|
|
description="AD attack path analysis with BloodHound (authorized testing only)"
|
|
)
|
|
parser.add_argument("--collect", choices=["sharphound", "bloodhound-python"])
|
|
parser.add_argument("--domain", help="AD domain")
|
|
parser.add_argument("--username", help="Domain username")
|
|
parser.add_argument("--password", help="Domain password")
|
|
parser.add_argument("--dc-ip", help="Domain controller IP")
|
|
parser.add_argument("--analyze-dir", help="Directory with BloodHound JSON files")
|
|
parser.add_argument("--cypher-query", help="Custom Cypher query for Neo4j")
|
|
parser.add_argument("--output", "-o", help="Output JSON report")
|
|
args = parser.parse_args()
|
|
|
|
print("[*] BloodHound AD Attack Path Agent")
|
|
print("[!] For authorized security testing only")
|
|
report = {"timestamp": datetime.now(timezone.utc).isoformat(), "findings": {}}
|
|
|
|
if args.collect == "sharphound":
|
|
result = run_sharphound(args.domain or "")
|
|
report["findings"]["collection"] = result
|
|
elif args.collect == "bloodhound-python":
|
|
result = run_bloodhound_python(
|
|
args.domain or "", args.username or "", args.password or "", args.dc_ip or ""
|
|
)
|
|
report["findings"]["collection"] = result
|
|
|
|
if args.analyze_dir:
|
|
analysis = analyze_bloodhound_json(args.analyze_dir)
|
|
report["findings"]["analysis"] = analysis
|
|
print(f"[*] Users: {analysis['users']}, Computers: {analysis['computers']}")
|
|
print(f"[*] Attack paths found: {len(analysis['attack_paths'])}")
|
|
|
|
if args.cypher_query:
|
|
results = query_neo4j(args.cypher_query)
|
|
report["findings"]["cypher_results"] = results
|
|
|
|
if args.output:
|
|
with open(args.output, "w") as f:
|
|
json.dump(report, f, indent=2)
|
|
print(f"[*] Report saved to {args.output}")
|
|
else:
|
|
print(json.dumps(report, indent=2))
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|