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>
This commit is contained in:
122
temp/bardach_dedup_step1_backup.py
Normal file
122
temp/bardach_dedup_step1_backup.py
Normal file
@@ -0,0 +1,122 @@
|
||||
#!/usr/bin/env python3
|
||||
"""Step 1: Pull all Bardach Temp contacts and save as backup."""
|
||||
|
||||
import json
|
||||
import subprocess
|
||||
import sys
|
||||
import os
|
||||
|
||||
TENANT_ID = "dd4a82e8-85a3-44ac-8800-07945ab4d95f"
|
||||
CLIENT_ID = "fabb3421-8b34-484b-bc17-e46de9703418"
|
||||
CLIENT_SECRET = "~QJ8Q~NyQSs4OcGqHZyPrA2CVnq9KBfKiimntbMO"
|
||||
SCOPE = "https://graph.microsoft.com/.default"
|
||||
USER = "barbara@bardach.net"
|
||||
BACKUP_FILE = "D:/ClaudeTools/temp/bardach_temp_backup_prededup.json"
|
||||
|
||||
SELECT_FIELDS = "id,displayName,givenName,surname,emailAddresses,homePhones,businessPhones,companyName,jobTitle,personalNotes,homeAddress,businessAddress,otherAddress,birthday,nickName,categories,lastModifiedDateTime"
|
||||
|
||||
|
||||
def get_token():
|
||||
"""Get OAuth2 token via client credentials."""
|
||||
url = f"https://login.microsoftonline.com/{TENANT_ID}/oauth2/v2.0/token"
|
||||
result = subprocess.run(
|
||||
[
|
||||
"curl", "-s", "-X", "POST", url,
|
||||
"-H", "Content-Type: application/x-www-form-urlencoded",
|
||||
"-d", f"client_id={CLIENT_ID}&scope={SCOPE}&client_secret={CLIENT_SECRET}&grant_type=client_credentials"
|
||||
],
|
||||
capture_output=True, text=True
|
||||
)
|
||||
data = json.loads(result.stdout)
|
||||
if "access_token" not in data:
|
||||
print(f"[ERROR] Failed to get token: {data}")
|
||||
sys.exit(1)
|
||||
print("[OK] Token acquired")
|
||||
return data["access_token"]
|
||||
|
||||
|
||||
def graph_get(token, url):
|
||||
"""Make a GET request to Graph API."""
|
||||
result = subprocess.run(
|
||||
[
|
||||
"curl", "-s", "-X", "GET", url,
|
||||
"-H", f"Authorization: Bearer {token}",
|
||||
"-H", "Content-Type: application/json"
|
||||
],
|
||||
capture_output=True, text=True
|
||||
)
|
||||
return json.loads(result.stdout)
|
||||
|
||||
|
||||
def find_temp_folder(token):
|
||||
"""Find the Temp contact folder ID."""
|
||||
url = f"https://graph.microsoft.com/v1.0/users/{USER}/contactFolders"
|
||||
data = graph_get(token, url)
|
||||
if "value" not in data:
|
||||
print(f"[ERROR] Failed to get contact folders: {data}")
|
||||
sys.exit(1)
|
||||
for folder in data["value"]:
|
||||
print(f" Found folder: {folder['displayName']} (id: {folder['id']})")
|
||||
if folder["displayName"].lower() == "temp":
|
||||
return folder["id"]
|
||||
# Check for child folders
|
||||
for folder in data["value"]:
|
||||
child_url = f"https://graph.microsoft.com/v1.0/users/{USER}/contactFolders/{folder['id']}/childFolders"
|
||||
child_data = graph_get(token, child_url)
|
||||
if "value" in child_data:
|
||||
for child in child_data["value"]:
|
||||
print(f" Found subfolder: {folder['displayName']}/{child['displayName']} (id: {child['id']})")
|
||||
if child["displayName"].lower() == "temp":
|
||||
return child["id"]
|
||||
print("[ERROR] Temp folder not found")
|
||||
sys.exit(1)
|
||||
|
||||
|
||||
def pull_all_contacts(token, folder_id):
|
||||
"""Pull all contacts from the Temp folder with pagination."""
|
||||
contacts = []
|
||||
url = f"https://graph.microsoft.com/v1.0/users/{USER}/contactFolders/{folder_id}/contacts?$top=100&$select={SELECT_FIELDS}"
|
||||
page = 1
|
||||
while url:
|
||||
print(f" Fetching page {page}...")
|
||||
data = graph_get(token, url)
|
||||
if "value" not in data:
|
||||
print(f"[ERROR] Failed to get contacts: {data}")
|
||||
break
|
||||
contacts.extend(data["value"])
|
||||
print(f" Got {len(data['value'])} contacts (total: {len(contacts)})")
|
||||
url = data.get("@odata.nextLink")
|
||||
page += 1
|
||||
# Re-acquire token every 50 pages to be safe
|
||||
if page % 50 == 0:
|
||||
print(" Re-acquiring token...")
|
||||
token = get_token()
|
||||
return contacts, token
|
||||
|
||||
|
||||
def main():
|
||||
print("=" * 60)
|
||||
print("STEP 1: Pull all Temp contacts and save backup")
|
||||
print("=" * 60)
|
||||
|
||||
token = get_token()
|
||||
|
||||
print("\n[INFO] Finding Temp folder...")
|
||||
folder_id = find_temp_folder(token)
|
||||
print(f"[OK] Temp folder ID: {folder_id}")
|
||||
|
||||
print("\n[INFO] Pulling all contacts...")
|
||||
contacts, token = pull_all_contacts(token, folder_id)
|
||||
|
||||
print(f"\n[OK] Total contacts pulled: {len(contacts)}")
|
||||
|
||||
# Save backup
|
||||
os.makedirs(os.path.dirname(BACKUP_FILE), exist_ok=True)
|
||||
with open(BACKUP_FILE, "w", encoding="utf-8") as f:
|
||||
json.dump({"total": len(contacts), "contacts": contacts}, f, indent=2, ensure_ascii=False)
|
||||
print(f"[OK] Backup saved to {BACKUP_FILE}")
|
||||
print(f"[OK] File size: {os.path.getsize(BACKUP_FILE) / 1024 / 1024:.1f} MB")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
Reference in New Issue
Block a user