Files
mukul975 c47eed6a64 Production hardening: security fixes, code quality, 724 skills complete
- 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
2026-03-19 13:26:49 +01:00

131 lines
4.8 KiB
Python

#!/usr/bin/env python3
"""BeyondCorp zero trust access assessment agent using Google Cloud IAP API via requests."""
import argparse
import json
import logging
import os
import subprocess
import sys
from datetime import datetime
from typing import List
try:
import requests
except ImportError:
sys.exit("requests required: pip install requests")
logging.basicConfig(level=logging.INFO, format="%(asctime)s [%(levelname)s] %(message)s")
logger = logging.getLogger(__name__)
def get_gcloud_token() -> str:
"""Get access token from gcloud CLI."""
try:
result = subprocess.run(
["gcloud", "auth", "print-access-token"], capture_output=True, text=True, timeout=10)
return result.stdout.strip()
except FileNotFoundError:
return ""
def list_iap_resources(project_id: str, token: str) -> List[dict]:
"""List IAP-protected resources in a GCP project."""
url = f"https://iap.googleapis.com/v1/projects/{project_id}/iap_tunnel/locations/-/destGroups"
resp = requests.get(url, headers={"Authorization": f"Bearer {token}"}, timeout=30)
if resp.status_code == 200:
return resp.json().get("destGroups", [])
return []
def get_iap_settings(project_id: str, resource: str, token: str) -> dict:
"""Get IAP settings for a specific resource."""
url = f"https://iap.googleapis.com/v1/projects/{project_id}/iap_web/compute/services/{resource}:iapSettings"
resp = requests.get(url, headers={"Authorization": f"Bearer {token}"}, timeout=30)
if resp.status_code == 200:
return resp.json()
return {"error": resp.status_code}
def list_access_levels(org_id: str, policy_name: str, token: str) -> List[dict]:
"""List Access Context Manager access levels."""
url = f"https://accesscontextmanager.googleapis.com/v1/accessPolicies/{policy_name}/accessLevels"
resp = requests.get(url, headers={"Authorization": f"Bearer {token}"}, timeout=30)
if resp.status_code == 200:
return resp.json().get("accessLevels", [])
return []
def audit_iap_bindings(project_id: str, token: str) -> List[dict]:
"""Audit IAM policy bindings for IAP-secured resources."""
url = f"https://cloudresourcemanager.googleapis.com/v1/projects/{project_id}:getIamPolicy"
resp = requests.post(url, headers={"Authorization": f"Bearer {token}"},
json={}, timeout=30)
if resp.status_code != 200:
return []
bindings = resp.json().get("bindings", [])
iap_bindings = [b for b in bindings if "iap" in b.get("role", "").lower()]
return iap_bindings
def assess_zero_trust_posture(project_id: str, token: str) -> dict:
"""Assess BeyondCorp zero trust posture for a project."""
iap_resources = list_iap_resources(project_id, token)
iap_bindings = audit_iap_bindings(project_id, token)
findings = []
if not iap_resources:
findings.append({"severity": "HIGH", "finding": "No IAP-protected resources found"})
if not iap_bindings:
findings.append({"severity": "HIGH", "finding": "No IAP IAM bindings configured"})
allUsers = any("allUsers" in str(b.get("members", [])) for b in iap_bindings)
if allUsers:
findings.append({"severity": "CRITICAL", "finding": "IAP binding includes allUsers"})
return {
"iap_resources": len(iap_resources),
"iap_bindings": len(iap_bindings),
"findings": findings,
}
def generate_report(project_id: str, token: str) -> dict:
"""Generate BeyondCorp zero trust assessment report."""
report = {
"analysis_date": datetime.utcnow().isoformat(),
"project": project_id,
"posture": assess_zero_trust_posture(project_id, token),
}
score = 100
for f in report["posture"]["findings"]:
if f["severity"] == "CRITICAL":
score -= 30
elif f["severity"] == "HIGH":
score -= 15
report["zero_trust_score"] = max(0, score)
return report
def main():
parser = argparse.ArgumentParser(description="BeyondCorp Zero Trust Assessment Agent")
parser.add_argument("--project", required=True, help="GCP project ID")
parser.add_argument("--token", default="", help="Access token (or uses gcloud)")
parser.add_argument("--output-dir", default=".")
parser.add_argument("--output", default="beyondcorp_report.json")
args = parser.parse_args()
token = args.token or get_gcloud_token()
if not token:
logger.error("No access token. Run: gcloud auth print-access-token")
sys.exit(1)
os.makedirs(args.output_dir, exist_ok=True)
report = generate_report(args.project, token)
out_path = os.path.join(args.output_dir, args.output)
with open(out_path, "w") as f:
json.dump(report, f, indent=2)
logger.info("Report saved to %s", out_path)
print(json.dumps(report, indent=2))
if __name__ == "__main__":
main()