Files
claudetools/temp/parra_onboard.py
Mike Swanson fa15b03180 sync: Auto-sync from ACG-M-L5090 at 2026-03-10 19:11:00
Synced files:
- Quote wizard frontend (all components, hooks, types, config)
- API updates (config, models, routers, schemas, services)
- Client work (bg-builders, gurushow)
- Scripts (BGB Lesley termination, CIPP, Datto, migration)
- Temp files (Bardach contacts, VWP investigation, misc)
- Credentials and session logs
- Email service, PHP API, session logs

Machine: ACG-M-L5090
Timestamp: 2026-03-10 19:11:00

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-10 19:59:08 -07:00

161 lines
6.6 KiB
Python

"""Onboard drelenaparra.com - verify access, assign Exchange Admin role."""
import urllib.request, urllib.parse, json, sys, base64
CLAUDE_APP = "fabb3421-8b34-484b-bc17-e46de9703418"
CLAUDE_SECRET = "~QJ8Q~NyQSs4OcGqHZyPrA2CVnq9KBfKiimntbMO"
TENANT_ID = "f06c26c7-a314-432c-a8e4-549574b6af74"
TENANT = "drelenaparra.com"
def get_token(scope):
data = urllib.parse.urlencode({
'client_id': CLAUDE_APP, 'client_secret': CLAUDE_SECRET,
'scope': scope, 'grant_type': 'client_credentials'
}).encode()
req = urllib.request.Request(
f"https://login.microsoftonline.com/{TENANT_ID}/oauth2/v2.0/token",
data=data, method='POST')
with urllib.request.urlopen(req) as resp:
return json.loads(resp.read())['access_token']
def graph_get(token, url):
req = urllib.request.Request(url, headers={'Authorization': f'Bearer {token}'})
with urllib.request.urlopen(req) as resp:
return json.loads(resp.read())
def graph_post(token, url, body):
data = json.dumps(body).encode()
req = urllib.request.Request(url, data=data, method='POST',
headers={'Authorization': f'Bearer {token}', 'Content-Type': 'application/json'})
with urllib.request.urlopen(req) as resp:
return json.loads(resp.read())
# Step 1: Get token and check permissions
print(f"[STEP 1] Getting Graph token for {TENANT}...")
token = get_token("https://graph.microsoft.com/.default")
# Decode JWT to check roles
payload = token.split('.')[1]
payload += '=' * (4 - len(payload) % 4)
decoded = json.loads(base64.urlsafe_b64decode(payload))
roles = decoded.get('roles', [])
print(f"[OK] Token acquired - {len(roles)} permissions granted")
# Step 2: List users
print(f"\n[STEP 2] Listing users...")
users = graph_get(token, f"https://graph.microsoft.com/v1.0/users?$select=displayName,userPrincipalName,mail")
for u in users.get('value', []):
print(f" {u['displayName']} - {u['userPrincipalName']}")
# Step 3: Find Claude SP
print(f"\n[STEP 3] Finding Claude SP...")
sp_filter = urllib.parse.quote(f"appId eq '{CLAUDE_APP}'")
sp_result = graph_get(token, f"https://graph.microsoft.com/v1.0/servicePrincipals?$filter={sp_filter}&$select=id,displayName")
if sp_result.get('value'):
sp = sp_result['value'][0]
sp_id = sp['id']
print(f"[OK] SP: {sp['displayName']} (ID: {sp_id})")
else:
print("[ERROR] SP not found")
sys.exit(1)
# Step 4: Check granted permissions
print(f"\n[STEP 4] Checking granted permissions...")
try:
grants = graph_get(token, f"https://graph.microsoft.com/v1.0/servicePrincipals/{sp_id}/appRoleAssignments")
roles_granted = grants.get('value', [])
resources = {}
for r in roles_granted:
res = r.get('resourceDisplayName', '?')
resources[res] = resources.get(res, 0) + 1
for res, count in sorted(resources.items()):
print(f" {res}: {count} permissions")
print(f" Total: {len(roles_granted)}")
except urllib.error.HTTPError as e:
print(f" Cannot read appRoleAssignments: HTTP {e.code}")
# Step 5: Activate and assign Exchange Admin
print(f"\n[STEP 5] Exchange Administrator role...")
try:
roles_result = graph_get(token, "https://graph.microsoft.com/v1.0/directoryRoles?$select=id,displayName")
exo_role = None
for role in roles_result.get('value', []):
if role.get('displayName') == 'Exchange Administrator':
exo_role = role
break
if not exo_role:
print(" Activating from template...")
try:
exo_role = graph_post(token, "https://graph.microsoft.com/v1.0/directoryRoles",
{"roleTemplateId": "29232cdf-9323-42fd-ade2-1d097af3e4de"})
print(f" [OK] Activated")
except urllib.error.HTTPError as e:
body = e.read().decode()
if 'already' in body.lower():
# Re-fetch
roles_result = graph_get(token, "https://graph.microsoft.com/v1.0/directoryRoles?$select=id,displayName")
for role in roles_result.get('value', []):
if role.get('displayName') == 'Exchange Administrator':
exo_role = role
break
else:
print(f" [ERROR] {e.code}: {body[:200]}")
if exo_role:
exo_role_id = exo_role['id']
print(f" Role ID: {exo_role_id}")
# Assign
try:
graph_post(token, f"https://graph.microsoft.com/v1.0/directoryRoles/{exo_role_id}/members/$ref",
{"@odata.id": f"https://graph.microsoft.com/v1.0/servicePrincipals/{sp_id}"})
print(f" [OK] Exchange Administrator assigned to Claude SP")
except urllib.error.HTTPError as e:
body = e.read().decode()
if 'already exist' in body.lower():
print(f" [OK] Already assigned")
else:
print(f" [WARNING] {e.code}: {body[:200]}")
except urllib.error.HTTPError as e:
print(f" [ERROR] Cannot manage roles: HTTP {e.code}")
# Step 6: Test access
print(f"\n[STEP 6] Testing API access...")
tests = [
("Users", f"https://graph.microsoft.com/v1.0/users?$top=1&$select=displayName"),
("Security", "https://graph.microsoft.com/v1.0/security/alerts?$top=1"),
("AuditLogs", "https://graph.microsoft.com/v1.0/auditLogs/signIns?$top=1"),
("ConditionalAccess", "https://graph.microsoft.com/v1.0/identity/conditionalAccess/policies"),
("Devices", "https://graph.microsoft.com/v1.0/deviceManagement/managedDevices?$top=1"),
]
for name, url in tests:
try:
graph_get(token, url)
print(f" [OK] {name}")
except urllib.error.HTTPError as e:
print(f" [FAIL] {name}: HTTP {e.code}")
# Step 7: Test Exchange
print(f"\n[STEP 7] Testing Exchange Online...")
try:
exo_token = get_token("https://outlook.office365.com/.default")
invoke_url = f"https://outlook.office365.com/adminapi/beta/{TENANT_ID}/InvokeCommand"
cmd = json.dumps({"CmdletInput": {"CmdletName": "Get-Mailbox", "Parameters": {"ResultSize": "1"}}}).encode()
req = urllib.request.Request(invoke_url, data=cmd, method='POST',
headers={'Authorization': f'Bearer {exo_token}', 'Content-Type': 'application/json'})
with urllib.request.urlopen(req) as resp:
result = json.loads(resp.read())
if result.get('value'):
print(f" [OK] Exchange Online - {result['value'][0].get('DisplayName','?')}")
else:
print(f" [OK] Exchange responded (no mailboxes yet)")
except urllib.error.HTTPError as e:
print(f" [FAIL] Exchange: HTTP {e.code}")
except Exception as e:
print(f" [FAIL] Exchange: {e}")
print(f"\n{'='*50}")
print(f" ONBOARDING COMPLETE: {TENANT}")
print(f" Tenant ID: {TENANT_ID}")
print(f"{'='*50}")