mirror of
https://github.com/mukul975/Anthropic-Cybersecurity-Skills.git
synced 2026-06-11 21:54: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
193 lines
7.4 KiB
Python
193 lines
7.4 KiB
Python
#!/usr/bin/env python3
|
|
"""Cloud Incident Response Agent - Automates AWS/Azure cloud IR containment and evidence collection."""
|
|
|
|
import json
|
|
import logging
|
|
import argparse
|
|
import subprocess
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
logging.basicConfig(level=logging.INFO, format="%(asctime)s [%(levelname)s] %(message)s")
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def aws_disable_access_key(username, access_key_id):
|
|
"""Disable a compromised IAM access key via AWS CLI."""
|
|
cmd = [
|
|
"aws", "iam", "update-access-key",
|
|
"--user-name", username,
|
|
"--access-key-id", access_key_id,
|
|
"--status", "Inactive",
|
|
]
|
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
if result.returncode == 0:
|
|
logger.info("Disabled access key %s for user %s", access_key_id, username)
|
|
else:
|
|
logger.error("Failed to disable key: %s", result.stderr)
|
|
return result.returncode == 0
|
|
|
|
|
|
def aws_attach_deny_all(username):
|
|
"""Attach AWSDenyAll policy to a compromised IAM user."""
|
|
cmd = [
|
|
"aws", "iam", "attach-user-policy",
|
|
"--user-name", username,
|
|
"--policy-arn", "arn:aws:iam::aws:policy/AWSDenyAll",
|
|
]
|
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
if result.returncode == 0:
|
|
logger.info("Attached AWSDenyAll to user %s", username)
|
|
return result.returncode == 0
|
|
|
|
|
|
def aws_isolate_ec2(instance_id, forensic_sg):
|
|
"""Isolate an EC2 instance by changing its security group to forensic isolation SG."""
|
|
cmd = [
|
|
"aws", "ec2", "modify-instance-attribute",
|
|
"--instance-id", instance_id,
|
|
"--groups", forensic_sg,
|
|
]
|
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
if result.returncode == 0:
|
|
logger.info("Isolated EC2 %s with security group %s", instance_id, forensic_sg)
|
|
return result.returncode == 0
|
|
|
|
|
|
def aws_snapshot_ebs(instance_id):
|
|
"""Create EBS snapshots of all volumes attached to an EC2 instance."""
|
|
cmd = [
|
|
"aws", "ec2", "describe-volumes",
|
|
"--filters", f"Name=attachment.instance-id,Values={instance_id}",
|
|
"--query", "Volumes[*].VolumeId",
|
|
"--output", "text",
|
|
]
|
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
volume_ids = result.stdout.strip().split()
|
|
snapshots = []
|
|
for vol_id in volume_ids:
|
|
snap_cmd = [
|
|
"aws", "ec2", "create-snapshot",
|
|
"--volume-id", vol_id,
|
|
"--description", f"IR evidence - {instance_id} - {datetime.utcnow().isoformat()}",
|
|
]
|
|
snap_result = subprocess.run(snap_cmd, capture_output=True, text=True, timeout=120)
|
|
if snap_result.returncode == 0:
|
|
snap_data = json.loads(snap_result.stdout)
|
|
snapshots.append(snap_data.get("SnapshotId"))
|
|
logger.info("Created snapshot %s for volume %s", snap_data.get("SnapshotId"), vol_id)
|
|
return snapshots
|
|
|
|
|
|
def aws_query_cloudtrail(username, hours_back=24):
|
|
"""Query CloudTrail for API calls made by a specific IAM user."""
|
|
start_time = (datetime.utcnow() - timedelta(hours=hours_back)).strftime("%Y-%m-%dT%H:%M:%SZ")
|
|
cmd = [
|
|
"aws", "cloudtrail", "lookup-events",
|
|
"--lookup-attributes", f"AttributeKey=Username,AttributeValue={username}",
|
|
"--start-time", start_time,
|
|
"--output", "json",
|
|
]
|
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
if result.returncode == 0:
|
|
events = json.loads(result.stdout).get("Events", [])
|
|
logger.info("CloudTrail: %d events for user %s in last %d hours", len(events), username, hours_back)
|
|
parsed = []
|
|
for event in events:
|
|
ct_event = json.loads(event.get("CloudTrailEvent", "{}"))
|
|
parsed.append({
|
|
"time": event.get("EventTime", ""),
|
|
"event_name": event.get("EventName", ""),
|
|
"source_ip": ct_event.get("sourceIPAddress", ""),
|
|
"user_agent": ct_event.get("userAgent", ""),
|
|
"resources": event.get("Resources", []),
|
|
})
|
|
return parsed
|
|
return []
|
|
|
|
|
|
def aws_list_attacker_resources(username, events):
|
|
"""Identify resources created by the attacker from CloudTrail events."""
|
|
create_events = [
|
|
e for e in events
|
|
if e["event_name"].startswith(("Create", "Run", "Put", "Attach"))
|
|
]
|
|
logger.info("Identified %d resource creation events", len(create_events))
|
|
return create_events
|
|
|
|
|
|
def aws_check_all_regions_instances():
|
|
"""Check all AWS regions for unauthorized EC2 instances."""
|
|
cmd = ["aws", "ec2", "describe-regions", "--query", "Regions[*].RegionName", "--output", "text"]
|
|
result = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
regions = result.stdout.strip().split()
|
|
all_instances = {}
|
|
for region in regions:
|
|
cmd = [
|
|
"aws", "ec2", "describe-instances",
|
|
"--region", region,
|
|
"--query", "Reservations[*].Instances[*].[InstanceId,InstanceType,State.Name]",
|
|
"--output", "json",
|
|
]
|
|
r = subprocess.run(cmd, capture_output=True, text=True, timeout=120)
|
|
if r.returncode == 0:
|
|
instances = json.loads(r.stdout)
|
|
running = [i for reservation in instances for i in reservation if i[2] == "running"]
|
|
if running:
|
|
all_instances[region] = running
|
|
logger.info("Found instances in %d regions", len(all_instances))
|
|
return all_instances
|
|
|
|
|
|
def generate_ir_report(incident_id, username, events, snapshots, containment_actions):
|
|
"""Generate a cloud incident response report."""
|
|
report = {
|
|
"incident_id": incident_id,
|
|
"timestamp": datetime.utcnow().isoformat(),
|
|
"compromised_identity": username,
|
|
"cloudtrail_events": len(events),
|
|
"evidence_snapshots": snapshots,
|
|
"containment_actions": containment_actions,
|
|
"attacker_activity": events[:20],
|
|
}
|
|
print(json.dumps(report, indent=2))
|
|
return report
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser(description="Cloud Incident Response Agent")
|
|
parser.add_argument("--incident-id", required=True, help="Incident ID")
|
|
parser.add_argument("--username", required=True, help="Compromised IAM username")
|
|
parser.add_argument("--access-key-id", help="Compromised access key to disable")
|
|
parser.add_argument("--instance-id", help="EC2 instance to isolate")
|
|
parser.add_argument("--forensic-sg", default="sg-forensic-isolate", help="Forensic SG ID")
|
|
parser.add_argument("--output", default="cloud_ir_report.json")
|
|
args = parser.parse_args()
|
|
|
|
containment = []
|
|
|
|
if args.access_key_id:
|
|
aws_disable_access_key(args.username, args.access_key_id)
|
|
containment.append(f"Disabled access key {args.access_key_id}")
|
|
|
|
aws_attach_deny_all(args.username)
|
|
containment.append(f"Attached AWSDenyAll to {args.username}")
|
|
|
|
snapshots = []
|
|
if args.instance_id:
|
|
aws_isolate_ec2(args.instance_id, args.forensic_sg)
|
|
containment.append(f"Isolated EC2 {args.instance_id}")
|
|
snapshots = aws_snapshot_ebs(args.instance_id)
|
|
|
|
events = aws_query_cloudtrail(args.username, hours_back=72)
|
|
attacker_actions = aws_list_attacker_resources(args.username, events)
|
|
|
|
report = generate_ir_report(args.incident_id, args.username, events, snapshots, containment)
|
|
with open(args.output, "w") as f:
|
|
json.dump(report, f, indent=2)
|
|
logger.info("IR report saved to %s", args.output)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|