mirror of
https://github.com/mukul975/Anthropic-Cybersecurity-Skills.git
synced 2026-06-14 15:04: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
144 lines
6.7 KiB
Python
144 lines
6.7 KiB
Python
#!/usr/bin/env python3
|
|
"""Agent for managing Saviynt access recertification campaigns via REST API."""
|
|
|
|
import os
|
|
import requests
|
|
import json
|
|
import argparse
|
|
from datetime import datetime, timezone
|
|
import urllib3
|
|
|
|
urllib3.disable_warnings(urllib3.exceptions.InsecureRequestWarning)
|
|
|
|
|
|
def authenticate(base_url, username, password):
|
|
"""Authenticate to Saviynt EIC and get OAuth token."""
|
|
url = f"{base_url}/ECM/api/login"
|
|
payload = {"username": username, "password": password}
|
|
resp = requests.post(url, json=payload, verify=not os.environ.get("SKIP_TLS_VERIFY", "").lower() == "true", timeout=30) # Set SKIP_TLS_VERIFY=true for self-signed certs in lab environments
|
|
resp.raise_for_status()
|
|
token = resp.json().get("access_token")
|
|
print(f"[*] Authenticated to Saviynt EIC")
|
|
return {"Authorization": f"Bearer {token}"}
|
|
|
|
|
|
def list_campaigns(base_url, headers, status="active"):
|
|
"""List certification campaigns."""
|
|
url = f"{base_url}/ECM/api/v5/listCertification"
|
|
payload = {"certificationstatus": status, "max": 50, "offset": 0}
|
|
resp = requests.post(url, headers=headers, json=payload, verify=not os.environ.get("SKIP_TLS_VERIFY", "").lower() == "true", timeout=30) # Set SKIP_TLS_VERIFY=true for self-signed certs in lab environments
|
|
resp.raise_for_status()
|
|
campaigns = resp.json().get("certifications", [])
|
|
print(f"\n[*] Campaigns ({status}): {len(campaigns)}")
|
|
for c in campaigns:
|
|
print(f" {c.get('certificationname')} - certifier: {c.get('certifier', 'N/A')} "
|
|
f"| due: {c.get('duedate', 'N/A')}")
|
|
return campaigns
|
|
|
|
|
|
def get_campaign_details(base_url, headers, cert_key):
|
|
"""Get detailed campaign status including item counts."""
|
|
url = f"{base_url}/ECM/api/v5/getCertificationDetails"
|
|
payload = {"certkey": cert_key}
|
|
resp = requests.post(url, headers=headers, json=payload, verify=not os.environ.get("SKIP_TLS_VERIFY", "").lower() == "true", timeout=30) # Set SKIP_TLS_VERIFY=true for self-signed certs in lab environments
|
|
resp.raise_for_status()
|
|
details = resp.json()
|
|
total = details.get("totalitems", 0)
|
|
certified = details.get("certifieditems", 0)
|
|
revoked = details.get("revokeditems", 0)
|
|
pending = total - certified - revoked
|
|
print(f"\n[*] Campaign {cert_key}: total={total}, certified={certified}, "
|
|
f"revoked={revoked}, pending={pending}")
|
|
return details
|
|
|
|
|
|
def get_pending_items(base_url, headers, cert_key, max_items=100):
|
|
"""Get items pending review in a certification campaign."""
|
|
url = f"{base_url}/ECM/api/v5/getCertificationItems"
|
|
payload = {"certkey": cert_key, "status": "pending", "max": max_items, "offset": 0}
|
|
resp = requests.post(url, headers=headers, json=payload, verify=not os.environ.get("SKIP_TLS_VERIFY", "").lower() == "true", timeout=30) # Set SKIP_TLS_VERIFY=true for self-signed certs in lab environments
|
|
resp.raise_for_status()
|
|
items = resp.json().get("certificationitems", [])
|
|
print(f"\n[*] Pending items: {len(items)}")
|
|
high_risk = [i for i in items if i.get("risk_score", 0) > 7]
|
|
print(f" High-risk items (score > 7): {len(high_risk)}")
|
|
for i in high_risk[:10]:
|
|
print(f" [!] {i.get('username')} - {i.get('entitlement_value')} "
|
|
f"(risk: {i.get('risk_score')})")
|
|
return items
|
|
|
|
|
|
def certify_items(base_url, headers, cert_key, item_ids, action="certify"):
|
|
"""Certify or revoke items in a campaign."""
|
|
url = f"{base_url}/ECM/api/v5/certifyItems"
|
|
payload = {"certkey": cert_key, "itemids": item_ids, "action": action,
|
|
"comments": f"Auto-{action} by recertification agent"}
|
|
resp = requests.post(url, headers=headers, json=payload, verify=not os.environ.get("SKIP_TLS_VERIFY", "").lower() == "true", timeout=30) # Set SKIP_TLS_VERIFY=true for self-signed certs in lab environments
|
|
resp.raise_for_status()
|
|
print(f"[*] {action.capitalize()}d {len(item_ids)} items in campaign {cert_key}")
|
|
return resp.json()
|
|
|
|
|
|
def check_overdue_campaigns(base_url, headers):
|
|
"""Find campaigns past their due date."""
|
|
url = f"{base_url}/ECM/api/v5/listCertification"
|
|
payload = {"certificationstatus": "active", "max": 200, "offset": 0}
|
|
resp = requests.post(url, headers=headers, json=payload, verify=not os.environ.get("SKIP_TLS_VERIFY", "").lower() == "true", timeout=30) # Set SKIP_TLS_VERIFY=true for self-signed certs in lab environments
|
|
resp.raise_for_status()
|
|
campaigns = resp.json().get("certifications", [])
|
|
now = datetime.now(timezone.utc)
|
|
overdue = []
|
|
for c in campaigns:
|
|
due = c.get("duedate", "")
|
|
if due:
|
|
try:
|
|
due_dt = datetime.fromisoformat(due.replace("Z", "+00:00"))
|
|
if due_dt < now:
|
|
overdue.append({"name": c.get("certificationname"),
|
|
"due": due, "certifier": c.get("certifier")})
|
|
except ValueError:
|
|
pass
|
|
print(f"\n[*] Overdue campaigns: {len(overdue)}")
|
|
for o in overdue:
|
|
print(f" [!] {o['name']} (due: {o['due']}, certifier: {o['certifier']})")
|
|
return overdue
|
|
|
|
|
|
def generate_report(campaigns, overdue, output_path):
|
|
"""Generate recertification status report."""
|
|
report = {"report_date": datetime.now(timezone.utc).isoformat(),
|
|
"active_campaigns": len(campaigns), "overdue_campaigns": len(overdue),
|
|
"campaigns": campaigns[:50], "overdue": overdue}
|
|
with open(output_path, "w") as f:
|
|
json.dump(report, f, indent=2, default=str)
|
|
print(f"\n[*] Report saved to {output_path}")
|
|
|
|
|
|
def main():
|
|
parser = argparse.ArgumentParser(description="Saviynt Access Recertification Agent")
|
|
parser.add_argument("action", choices=["list", "details", "pending", "overdue", "full-audit"])
|
|
parser.add_argument("--url", required=True, help="Saviynt EIC base URL")
|
|
parser.add_argument("--username", required=True)
|
|
parser.add_argument("--password", required=True)
|
|
parser.add_argument("--cert-key", help="Certification campaign key")
|
|
parser.add_argument("-o", "--output", default="recert_report.json")
|
|
args = parser.parse_args()
|
|
|
|
headers = authenticate(args.url, args.username, args.password)
|
|
if args.action == "list":
|
|
list_campaigns(args.url, headers)
|
|
elif args.action == "details" and args.cert_key:
|
|
get_campaign_details(args.url, headers, args.cert_key)
|
|
elif args.action == "pending" and args.cert_key:
|
|
get_pending_items(args.url, headers, args.cert_key)
|
|
elif args.action == "overdue":
|
|
check_overdue_campaigns(args.url, headers)
|
|
elif args.action == "full-audit":
|
|
campaigns = list_campaigns(args.url, headers)
|
|
overdue = check_overdue_campaigns(args.url, headers)
|
|
generate_report(campaigns, overdue, args.output)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|