Files
Anthropic-Cybersecurity-Skills/skills/implementing-cloud-trail-log-analysis/scripts/agent.py
T
mukul975 27c6414ca5 Add folder anatomy (scripts/agent.py + references/api-reference.md) for 648 cybersecurity skills
Complete skill folder anatomy across all cybersecurity skills:
- scripts/agent.py: 80-150 line Python agents using real libraries (impacket,
  boto3, azure-mgmt-*, kubernetes, pefile, yara, scapy, shodan, stix2, etc.)
- references/api-reference.md: real API documentation with method signatures
- LICENSE: MIT license for all skill folders
2026-03-10 21:02:12 +01:00

177 lines
7.4 KiB
Python

#!/usr/bin/env python3
"""CloudTrail log analysis agent for security monitoring and threat detection."""
import json
import sys
import argparse
from datetime import datetime, timedelta
from collections import Counter
try:
import boto3
from botocore.exceptions import ClientError
except ImportError:
print("Install boto3: pip install boto3")
sys.exit(1)
SUSPICIOUS_EVENTS = {
"ConsoleLogin": "Potential unauthorized console access",
"StopLogging": "CloudTrail logging disabled - cover tracks",
"DeleteTrail": "CloudTrail trail deleted - evidence destruction",
"CreateUser": "New IAM user created - possible persistence",
"CreateAccessKey": "New access key - potential credential theft",
"AttachUserPolicy": "Policy attached to user - privilege escalation",
"PutBucketPolicy": "S3 bucket policy changed - data exposure risk",
"AuthorizeSecurityGroupIngress": "Security group opened - lateral movement",
"RunInstances": "EC2 instances launched - cryptomining or C2",
"CreateRole": "New IAM role created - privilege escalation",
"AssumeRole": "Role assumed - potential lateral movement",
"PutUserPolicy": "Inline policy added to user",
"DeleteBucketEncryption": "Bucket encryption removed",
"DisableKey": "KMS key disabled - ransomware indicator",
"ModifyInstanceAttribute": "Instance attribute changed",
}
def get_cloudtrail_client(region="us-east-1"):
"""Create CloudTrail client."""
return boto3.client("cloudtrail", region_name=region)
def lookup_events(client, event_name=None, hours=24, max_results=50):
"""Look up CloudTrail events with optional filtering."""
start_time = datetime.utcnow() - timedelta(hours=hours)
kwargs = {"StartTime": start_time, "MaxResults": max_results,
"LookupAttributes": []}
if event_name:
kwargs["LookupAttributes"] = [{"AttributeKey": "EventName", "AttributeValue": event_name}]
try:
resp = client.lookup_events(**kwargs)
events = []
for e in resp.get("Events", []):
detail = json.loads(e.get("CloudTrailEvent", "{}"))
events.append({
"event_name": e.get("EventName"),
"event_time": str(e.get("EventTime")),
"username": e.get("Username", "unknown"),
"source_ip": detail.get("sourceIPAddress", "unknown"),
"user_agent": detail.get("userAgent", "unknown"),
"region": detail.get("awsRegion", "unknown"),
"error_code": detail.get("errorCode"),
"error_message": detail.get("errorMessage"),
"resources": [r.get("ResourceName", "") for r in e.get("Resources", [])],
})
return events
except ClientError as e:
return [{"error": str(e)}]
def detect_suspicious_activity(client, hours=24):
"""Scan CloudTrail for suspicious API calls."""
detections = []
for event_name, description in SUSPICIOUS_EVENTS.items():
events = lookup_events(client, event_name=event_name, hours=hours)
for e in events:
if e.get("error"):
continue
severity = "CRITICAL" if event_name in ["StopLogging", "DeleteTrail", "DisableKey"] \
else "HIGH" if event_name in ["CreateUser", "CreateAccessKey", "AttachUserPolicy"] \
else "MEDIUM"
detections.append({
"event": event_name, "description": description,
"severity": severity, "user": e["username"],
"source_ip": e["source_ip"], "time": e["event_time"],
"resources": e["resources"],
})
return sorted(detections, key=lambda x: {"CRITICAL": 0, "HIGH": 1, "MEDIUM": 2}.get(x["severity"], 3))
def detect_failed_auth(client, hours=24):
"""Detect failed authentication attempts."""
events = lookup_events(client, event_name="ConsoleLogin", hours=hours, max_results=100)
failed = [e for e in events if e.get("error_code")]
by_ip = Counter(e["source_ip"] for e in failed)
by_user = Counter(e["username"] for e in failed)
return {"total_failed": len(failed), "by_source_ip": dict(by_ip.most_common(10)),
"by_username": dict(by_user.most_common(10))}
def detect_unauthorized_regions(client, authorized_regions, hours=24):
"""Detect API calls from unauthorized AWS regions."""
events = lookup_events(client, hours=hours, max_results=100)
unauthorized = [e for e in events if e.get("region") and
e["region"] not in authorized_regions and not e.get("error")]
return unauthorized
def analyze_user_activity(client, username, hours=24):
"""Analyze all activity for a specific user."""
kwargs = {"StartTime": datetime.utcnow() - timedelta(hours=hours),
"MaxResults": 50,
"LookupAttributes": [{"AttributeKey": "Username", "AttributeValue": username}]}
try:
resp = client.lookup_events(**kwargs)
actions = Counter()
timeline = []
for e in resp.get("Events", []):
actions[e.get("EventName")] += 1
timeline.append({"event": e.get("EventName"), "time": str(e.get("EventTime"))})
return {"user": username, "total_events": len(timeline),
"actions": dict(actions.most_common(20)), "timeline": timeline[:20]}
except ClientError as e:
return {"error": str(e)}
def run_cloudtrail_analysis(region="us-east-1", hours=24):
"""Run comprehensive CloudTrail security analysis."""
client = get_cloudtrail_client(region)
print(f"\n{'='*60}")
print(f" CLOUDTRAIL SECURITY ANALYSIS")
print(f" Region: {region} | Lookback: {hours}h")
print(f" Generated: {datetime.utcnow().strftime('%Y-%m-%d %H:%M:%S')} UTC")
print(f"{'='*60}\n")
detections = detect_suspicious_activity(client, hours)
print(f"--- SUSPICIOUS ACTIVITY ({len(detections)} detections) ---")
for d in detections[:15]:
print(f" [{d['severity']}] {d['event']}: {d['description']}")
print(f" User: {d['user']} | IP: {d['source_ip']} | Time: {d['time']}")
auth = detect_failed_auth(client, hours)
print(f"\n--- FAILED AUTHENTICATION ---")
print(f" Total failures: {auth['total_failed']}")
print(f" Top IPs: {auth['by_source_ip']}")
print(f" Top Users: {auth['by_username']}")
print(f"\n{'='*60}\n")
return {"detections": detections, "auth_failures": auth}
def main():
parser = argparse.ArgumentParser(description="CloudTrail Log Analysis Agent")
parser.add_argument("--region", default="us-east-1")
parser.add_argument("--hours", type=int, default=24, help="Lookback period in hours")
parser.add_argument("--analyze", action="store_true", help="Run full analysis")
parser.add_argument("--user", help="Analyze specific user activity")
parser.add_argument("--output", help="Save report to JSON")
args = parser.parse_args()
if args.user:
client = get_cloudtrail_client(args.region)
result = analyze_user_activity(client, args.user, args.hours)
print(json.dumps(result, indent=2, default=str))
elif args.analyze:
report = run_cloudtrail_analysis(args.region, args.hours)
if args.output:
with open(args.output, "w") as f:
json.dump(report, f, indent=2, default=str)
print(f"[+] Report saved to {args.output}")
else:
parser.print_help()
if __name__ == "__main__":
main()