197 lines
9.3 KiB
Python
197 lines
9.3 KiB
Python
"""EspoCRM Webhook - Document Update
|
|
|
|
Empfängt Update-Webhooks von EspoCRM für Documents.
|
|
Loggt detailliert alle Payload-Informationen für Analyse.
|
|
"""
|
|
import json
|
|
import datetime
|
|
from typing import Any
|
|
from motia import FlowContext, http, ApiRequest, ApiResponse
|
|
|
|
|
|
config = {
|
|
"name": "VMH Webhook Document Update",
|
|
"description": "Empfängt Update-Webhooks von EspoCRM für Document Entities",
|
|
"flows": ["vmh-documents"],
|
|
"triggers": [
|
|
http("POST", "/vmh/webhook/document/update")
|
|
],
|
|
"enqueues": ["vmh.document.update"],
|
|
}
|
|
|
|
|
|
async def handler(request: ApiRequest, ctx: FlowContext[Any]) -> ApiResponse:
|
|
"""
|
|
Webhook handler for Document updates in EspoCRM.
|
|
|
|
Receives notifications when documents are updated and emits queue events
|
|
for processing (xAI sync, etc.).
|
|
|
|
Note: Loop-Prevention sollte auf EspoCRM-Seite implementiert werden.
|
|
xAI-Feld-Updates sollten keine neuen Webhooks triggern.
|
|
"""
|
|
try:
|
|
payload = request.body or []
|
|
|
|
# ═══════════════════════════════════════════════════════════════
|
|
# DETAILLIERTES LOGGING FÜR ANALYSE
|
|
# ═══════════════════════════════════════════════════════════════
|
|
|
|
ctx.logger.info("=" * 80)
|
|
ctx.logger.info("📥 EspoCRM DOCUMENT UPDATE WEBHOOK EMPFANGEN")
|
|
ctx.logger.info("=" * 80)
|
|
|
|
# Log Request Headers
|
|
ctx.logger.info("\n🔍 REQUEST HEADERS:")
|
|
if hasattr(request, 'headers'):
|
|
for key, value in request.headers.items():
|
|
ctx.logger.info(f" {key}: {value}")
|
|
else:
|
|
ctx.logger.info(" (keine Headers verfügbar)")
|
|
|
|
# Log Payload Type & Structure
|
|
ctx.logger.info(f"\n📦 PAYLOAD TYPE: {type(payload).__name__}")
|
|
ctx.logger.info(f"📦 PAYLOAD LENGTH: {len(payload) if isinstance(payload, (list, dict)) else 'N/A'}")
|
|
|
|
# Log Full Payload (pretty-printed)
|
|
ctx.logger.info("\n📄 FULL PAYLOAD:")
|
|
ctx.logger.info(json.dumps(payload, indent=2, ensure_ascii=False))
|
|
|
|
# ═══════════════════════════════════════════════════════════════
|
|
# PAYLOAD ANALYSE & ID EXTRAKTION
|
|
# ═══════════════════════════════════════════════════════════════
|
|
|
|
entity_ids = set()
|
|
|
|
if isinstance(payload, list):
|
|
ctx.logger.info(f"\n✅ Payload ist LIST mit {len(payload)} Einträgen")
|
|
for idx, entity in enumerate(payload):
|
|
if isinstance(entity, dict):
|
|
entity_id = entity.get('id')
|
|
if entity_id:
|
|
entity_ids.add(entity_id)
|
|
|
|
ctx.logger.info(f"\n 📄 Document #{idx + 1}:")
|
|
ctx.logger.info(f" ID: {entity_id}")
|
|
ctx.logger.info(f" Name: {entity.get('name', 'N/A')}")
|
|
ctx.logger.info(f" Modified At: {entity.get('modifiedAt', 'N/A')}")
|
|
ctx.logger.info(f" Modified By: {entity.get('modifiedById', 'N/A')}")
|
|
ctx.logger.info(f" Verfügbare Felder: {', '.join(entity.keys())}")
|
|
|
|
# Prüfe ob CHANGED fields mitgeliefert werden
|
|
changed_fields = entity.get('changedFields') or entity.get('changed') or entity.get('modifiedFields')
|
|
if changed_fields:
|
|
ctx.logger.info(f" 🔄 Geänderte Felder: {json.dumps(changed_fields, ensure_ascii=False)}")
|
|
|
|
# xAI-relevante Felder
|
|
xai_fields = {k: v for k, v in entity.items()
|
|
if 'xai' in k.lower() or 'collection' in k.lower()}
|
|
if xai_fields:
|
|
ctx.logger.info(f" 🤖 xAI-Felder: {json.dumps(xai_fields, ensure_ascii=False)}")
|
|
|
|
# Relationship Felder
|
|
rel_fields = {k: v for k, v in entity.items()
|
|
if 'parent' in k.lower() or 'related' in k.lower() or
|
|
'link' in k.lower() or k.endswith('Id') or k.endswith('Ids')}
|
|
if rel_fields:
|
|
ctx.logger.info(f" 🔗 Relationship-Felder: {json.dumps(rel_fields, ensure_ascii=False)}")
|
|
|
|
elif isinstance(payload, dict):
|
|
ctx.logger.info("\n✅ Payload ist SINGLE DICT")
|
|
entity_id = payload.get('id')
|
|
if entity_id:
|
|
entity_ids.add(entity_id)
|
|
|
|
ctx.logger.info(f"\n 📄 Document:")
|
|
ctx.logger.info(f" ID: {entity_id}")
|
|
ctx.logger.info(f" Name: {payload.get('name', 'N/A')}")
|
|
ctx.logger.info(f" Modified At: {payload.get('modifiedAt', 'N/A')}")
|
|
ctx.logger.info(f" Modified By: {payload.get('modifiedById', 'N/A')}")
|
|
ctx.logger.info(f" Verfügbare Felder: {', '.join(payload.keys())}")
|
|
|
|
# Geänderte Felder
|
|
changed_fields = payload.get('changedFields') or payload.get('changed') or payload.get('modifiedFields')
|
|
if changed_fields:
|
|
ctx.logger.info(f" 🔄 Geänderte Felder: {json.dumps(changed_fields, ensure_ascii=False)}")
|
|
|
|
# xAI-relevante Felder
|
|
xai_fields = {k: v for k, v in payload.items()
|
|
if 'xai' in k.lower() or 'collection' in k.lower()}
|
|
if xai_fields:
|
|
ctx.logger.info(f" 🤖 xAI-Felder: {json.dumps(xai_fields, ensure_ascii=False)}")
|
|
|
|
# Relationship Felder
|
|
rel_fields = {k: v for k, v in payload.items()
|
|
if 'parent' in k.lower() or 'related' in k.lower() or
|
|
'link' in k.lower() or k.endswith('Id') or k.endswith('Ids')}
|
|
if rel_fields:
|
|
ctx.logger.info(f" 🔗 Relationship-Felder: {json.dumps(rel_fields, ensure_ascii=False)}")
|
|
else:
|
|
ctx.logger.warning(f"⚠️ Unerwarteter Payload-Typ: {type(payload)}")
|
|
|
|
# ═══════════════════════════════════════════════════════════════
|
|
# QUEUE EVENTS EMITTIEREN
|
|
# ═══════════════════════════════════════════════════════════════
|
|
|
|
ctx.logger.info("\n" + "=" * 80)
|
|
ctx.logger.info(f"📊 ZUSAMMENFASSUNG: {len(entity_ids)} Document(s) gefunden")
|
|
ctx.logger.info("=" * 80)
|
|
|
|
if not entity_ids:
|
|
ctx.logger.warning("⚠️ Keine Document-IDs im Payload gefunden!")
|
|
return ApiResponse(
|
|
status=200,
|
|
body={
|
|
'status': 'received',
|
|
'action': 'update',
|
|
'ids_count': 0,
|
|
'warning': 'No document IDs found in payload'
|
|
}
|
|
)
|
|
|
|
# Emit events für Queue-Processing
|
|
for entity_id in entity_ids:
|
|
await ctx.enqueue({
|
|
'topic': 'vmh.document.update',
|
|
'data': {
|
|
'entity_id': entity_id,
|
|
'action': 'update',
|
|
'source': 'webhook',
|
|
'timestamp': datetime.datetime.now().isoformat()
|
|
}
|
|
})
|
|
ctx.logger.info(f"✅ Event emittiert: vmh.document.update für ID {entity_id}")
|
|
|
|
ctx.logger.info("\n" + "=" * 80)
|
|
ctx.logger.info(f"✅ WEBHOOK VERARBEITUNG ABGESCHLOSSEN")
|
|
ctx.logger.info("=" * 80)
|
|
|
|
return ApiResponse(
|
|
status=200,
|
|
body={
|
|
'status': 'received',
|
|
'action': 'update',
|
|
'ids_count': len(entity_ids),
|
|
'document_ids': list(entity_ids)
|
|
}
|
|
)
|
|
|
|
except Exception as e:
|
|
ctx.logger.error("=" * 80)
|
|
ctx.logger.error(f"❌ FEHLER beim Verarbeiten des Document Update Webhooks")
|
|
ctx.logger.error("=" * 80)
|
|
ctx.logger.error(f"Error Type: {type(e).__name__}")
|
|
ctx.logger.error(f"Error Message: {str(e)}")
|
|
|
|
import traceback
|
|
ctx.logger.error(f"Stack Trace:\n{traceback.format_exc()}")
|
|
|
|
return ApiResponse(
|
|
status=500,
|
|
body={
|
|
'error': 'Internal server error',
|
|
'error_type': type(e).__name__,
|
|
'details': str(e)
|
|
}
|
|
)
|