mail_message.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. import logging
  2. import requests
  3. from odoo import models
  4. from odoo.addons.mail.tools.discuss import Store
  5. from odoo.addons.whatsapp.tools.whatsapp_exception import WhatsAppError
  6. from odoo.exceptions import UserError
  7. _logger = logging.getLogger(__name__)
  8. class MailMessage(models.Model):
  9. _inherit = 'mail.message'
  10. def _message_reaction(self, content, action, partner, guest, store: Store = None):
  11. """Sobrescribir para usar WhatsApp Web API Gateway cuando esté configurado"""
  12. # Si es mensaje de WhatsApp, verificar si usa WhatsApp Web
  13. if self.message_type == "whatsapp_message" and self.wa_message_ids:
  14. wa_msg = self.wa_message_ids[0]
  15. # Verificar si la cuenta usa WhatsApp Web
  16. if wa_msg.wa_account_id and wa_msg.wa_account_id.whatsapp_web_url:
  17. # Usar API Gateway para WhatsApp Web
  18. self._send_whatsapp_web_reaction(wa_msg, content, action, partner, guest, store)
  19. # Actualizar UI directamente usando el método base de mail (sin pasar por enterprise)
  20. # Esto evita que el método del enterprise intente enviar de nuevo
  21. return self._update_reaction_ui(content, action, partner, guest, store)
  22. else:
  23. # Usar método original para WhatsApp Business API (enterprise)
  24. # Este llamará a super() al final para actualizar la UI
  25. return super()._message_reaction(content, action, partner, guest, store)
  26. # Para mensajes que no son de WhatsApp, usar método base
  27. return super()._message_reaction(content, action, partner, guest, store)
  28. def _update_reaction_ui(self, content, action, partner, guest, store: Store = None):
  29. """Actualizar la UI de reacciones sin intentar enviar (para WhatsApp Web)"""
  30. self.ensure_one()
  31. # Buscar reacción existente
  32. domain = [
  33. ("message_id", "=", self.id),
  34. ("partner_id", "=", partner.id),
  35. ("guest_id", "=", guest.id),
  36. ("content", "=", content),
  37. ]
  38. reaction = self.env["mail.message.reaction"].search(domain)
  39. # Crear/eliminar reacción según la acción
  40. if action == "add" and not reaction:
  41. create_values = {
  42. "message_id": self.id,
  43. "content": content,
  44. "partner_id": partner.id,
  45. "guest_id": guest.id,
  46. }
  47. self.env["mail.message.reaction"].create(create_values)
  48. if action == "remove" and reaction:
  49. reaction.unlink()
  50. if store:
  51. # Llenar el store para usuarios portal no autenticados
  52. self._reaction_group_to_store(store, content)
  53. # Enviar el grupo de reacciones al bus para usuarios autenticados
  54. self._bus_send_reaction_group(content)
  55. def _send_whatsapp_web_reaction(self, wa_msg, content, action, partner, guest, store: Store = None):
  56. """Enviar reacción usando WhatsApp Web API Gateway"""
  57. self.ensure_one()
  58. account = wa_msg.wa_account_id
  59. url = account.whatsapp_web_url
  60. session_name = account.whatsapp_web_login
  61. api_key = account.whatsapp_web_api_key
  62. if not all([url, session_name, api_key]):
  63. raise UserError("WhatsApp Web no está completamente configurado. Faltan URL, Login o API Key.")
  64. # Manejar reacciones previas (igual que el método original)
  65. if action == "add":
  66. previous_reaction = self.env["mail.message.reaction"].search([
  67. ("message_id", "=", self.id),
  68. ("partner_id", "=", partner.id),
  69. ("guest_id", "=", guest.id),
  70. ], limit=1)
  71. if previous_reaction:
  72. previous_reaction_emoji = previous_reaction.content
  73. if previous_reaction_emoji == content:
  74. return
  75. previous_reaction.unlink()
  76. self._bus_send_reaction_group(previous_reaction_emoji)
  77. # Obtener el ID del mensaje original
  78. message_id = wa_msg.msg_uid
  79. if not message_id:
  80. raise UserError("No se puede enviar reacción: el mensaje no tiene ID válido.")
  81. # Construir URL y payload para la API Gateway
  82. base_url = url.rstrip('/')
  83. endpoint = 'send-reaction'
  84. full_url = f"{base_url}/api/v1/{session_name}/{endpoint}"
  85. # Determinar emoji (vacío si es remover)
  86. emoji = content if action == "add" else ""
  87. payload = {
  88. "messageId": message_id,
  89. "emoji": emoji
  90. }
  91. headers = {
  92. "Content-Type": "application/json",
  93. "X-API-Key": api_key
  94. }
  95. try:
  96. _logger.info("Enviando reacción %s al mensaje %s", emoji or "vacía", message_id)
  97. response = requests.post(full_url, json=payload, headers=headers, timeout=30)
  98. if response.status_code == 200:
  99. _logger.info("Reacción enviada exitosamente a WhatsApp Web")
  100. # No retornar aquí, dejar que el método padre actualice la UI
  101. return
  102. else:
  103. error_text = response.text
  104. _logger.error("Error al enviar reacción. Código: %s, Respuesta: %s", response.status_code, error_text)
  105. raise UserError(f"Error al enviar reacción: {error_text}")
  106. except requests.exceptions.RequestException as e:
  107. _logger.error("Error de conexión al enviar reacción: %s", str(e))
  108. raise UserError(f"Error de conexión al enviar reacción: {str(e)}")
  109. except Exception as e:
  110. _logger.error("Error inesperado al enviar reacción: %s", str(e))
  111. raise UserError(f"Error al enviar reacción: {str(e)}")