Files

187 lines
6.0 KiB
Python

#!/usr/bin/env python3
"""
GCP Binary Authorization Management Script
Automates attestor management, policy configuration, and attestation
creation for container image supply chain security.
"""
import json
import subprocess
import sys
def run_gcloud(args):
"""Execute gcloud command and return parsed output."""
cmd = ["gcloud"] + args + ["--format=json"]
result = subprocess.run(cmd, capture_output=True, text=True)
if result.returncode != 0:
return None, result.stderr
try:
return json.loads(result.stdout) if result.stdout.strip() else {}, None
except json.JSONDecodeError:
return result.stdout, None
def check_binauthz_status(project_id):
"""Check Binary Authorization configuration status."""
print(f"[*] Checking Binary Authorization status for project: {project_id}")
policy, err = run_gcloud([
"container", "binauthz", "policy", "export",
f"--project={project_id}"
])
if err:
print(f"[!] Error fetching policy: {err}")
return None
print(f"[+] Current policy retrieved")
return policy
def list_attestors(project_id):
"""List all configured attestors."""
attestors, err = run_gcloud([
"container", "binauthz", "attestors", "list",
f"--project={project_id}"
])
if err:
print(f"[!] Error listing attestors: {err}")
return []
if isinstance(attestors, list):
print(f"[+] Found {len(attestors)} attestors:")
for a in attestors:
print(f" - {a.get('name', 'unknown')}")
return attestors or []
def verify_attestation(project_id, image_url, attestor):
"""Verify that an image has a valid attestation."""
print(f"[*] Verifying attestation for: {image_url}")
attestations, err = run_gcloud([
"container", "binauthz", "attestations", "list",
f"--attestor={attestor}",
f"--attestor-project={project_id}",
f"--artifact-url={image_url}"
])
if err:
print(f"[!] Error: {err}")
return False
if isinstance(attestations, list) and len(attestations) > 0:
print(f"[+] Image has {len(attestations)} attestation(s)")
return True
else:
print(f"[!] No attestations found for image")
return False
def create_attestation(project_id, image_url, attestor, key_info):
"""Create an attestation for a container image."""
print(f"[*] Creating attestation for: {image_url}")
_, err = run_gcloud([
"container", "binauthz", "attestations", "sign-and-create",
f"--artifact-url={image_url}",
f"--attestor={attestor}",
f"--attestor-project={project_id}",
f"--keyversion-project={key_info['project']}",
f"--keyversion-location={key_info['location']}",
f"--keyversion-keyring={key_info['keyring']}",
f"--keyversion-key={key_info['key']}",
f"--keyversion={key_info['version']}"
])
if err:
print(f"[!] Error creating attestation: {err}")
return False
print(f"[+] Attestation created successfully")
return True
def audit_policy_compliance(project_id):
"""Audit Binary Authorization policy for security compliance."""
policy, err = run_gcloud([
"container", "binauthz", "policy", "export",
f"--project={project_id}"
])
if not policy:
print("[!] Could not retrieve policy")
return
findings = []
if isinstance(policy, str):
import yaml
policy = yaml.safe_load(policy)
default_rule = policy.get("defaultAdmissionRule", {})
if default_rule.get("evaluationMode") == "ALWAYS_ALLOW":
findings.append({
"severity": "HIGH",
"finding": "Default admission rule allows all images",
"recommendation": "Set to REQUIRE_ATTESTATION or ALWAYS_DENY"
})
if default_rule.get("enforcementMode") == "DRYRUN_AUDIT_LOG_ONLY":
findings.append({
"severity": "MEDIUM",
"finding": "Default rule is in dry-run mode (not enforcing)",
"recommendation": "Switch to ENFORCED_BLOCK_AND_AUDIT_LOG"
})
if not policy.get("globalPolicyEvaluationMode") == "ENABLE":
findings.append({
"severity": "LOW",
"finding": "Global policy evaluation not explicitly enabled",
"recommendation": "Enable globalPolicyEvaluationMode"
})
whitelist = policy.get("admissionWhitelistPatterns", [])
for pattern in whitelist:
name = pattern.get("namePattern", "")
if name == "*" or name == "**":
findings.append({
"severity": "CRITICAL",
"finding": f"Wildcard whitelist pattern: {name}",
"recommendation": "Remove overly broad whitelist patterns"
})
print(f"\n[*] Policy Compliance Audit Results:")
if findings:
for f in findings:
print(f" [{f['severity']}] {f['finding']}")
print(f" Recommendation: {f['recommendation']}")
else:
print(" [OK] No issues found")
return findings
if __name__ == "__main__":
import argparse
parser = argparse.ArgumentParser(description="GCP Binary Authorization Management")
parser.add_argument("--project", required=True, help="GCP Project ID")
parser.add_argument("--status", action="store_true", help="Check policy status")
parser.add_argument("--list-attestors", action="store_true", help="List attestors")
parser.add_argument("--verify", type=str, help="Verify attestation for image URL")
parser.add_argument("--attestor", type=str, help="Attestor name")
parser.add_argument("--audit", action="store_true", help="Audit policy compliance")
args = parser.parse_args()
if args.status:
check_binauthz_status(args.project)
if args.list_attestors:
list_attestors(args.project)
if args.verify and args.attestor:
verify_attestation(args.project, args.verify, args.attestor)
if args.audit:
audit_policy_compliance(args.project)