2022-02-21 23:41:36 +00:00
from secrets import token_urlsafe
2022-10-25 18:04:12 +00:00
from decouple import config
2022-02-21 23:41:36 +00:00
from api . models import Order
2022-08-08 15:58:06 +00:00
from api . utils import get_session
2022-02-21 23:41:36 +00:00
2022-10-20 09:56:10 +00:00
class Telegram :
2023-03-27 09:37:36 +00:00
""" Simple telegram messages using TG ' s API """
2022-02-21 23:41:36 +00:00
2022-08-08 15:58:06 +00:00
session = get_session ( )
2022-10-20 09:56:10 +00:00
site = config ( " HOST_NAME " )
2022-02-22 00:55:31 +00:00
2022-02-21 23:41:36 +00:00
def get_context ( user ) :
""" returns context needed to enable TG notifications """
context = { }
2023-05-01 10:30:53 +00:00
if user . robot . telegram_enabled :
2022-10-20 09:56:10 +00:00
context [ " tg_enabled " ] = True
2022-02-21 23:41:36 +00:00
else :
2022-10-20 09:56:10 +00:00
context [ " tg_enabled " ] = False
2023-05-01 10:30:53 +00:00
if user . robot . telegram_token is None :
2023-11-06 14:01:56 +00:00
user . robot . telegram_token = token_urlsafe ( ) [ : 15 ]
2023-05-08 18:10:37 +00:00
user . robot . save ( update_fields = [ " telegram_token " ] )
2022-02-21 23:41:36 +00:00
2023-05-01 10:30:53 +00:00
context [ " tg_token " ] = user . robot . telegram_token
2022-10-20 09:56:10 +00:00
context [ " tg_bot_name " ] = config ( " TELEGRAM_BOT_NAME " )
2022-02-21 23:41:36 +00:00
return context
2023-02-13 12:41:06 +00:00
def send_message ( self , chat_id , text ) :
2022-10-20 09:56:10 +00:00
""" sends a message to a user with telegram notifications enabled """
2022-02-21 23:41:36 +00:00
2022-10-20 09:56:10 +00:00
bot_token = config ( " TELEGRAM_TOKEN " )
2022-02-21 23:41:36 +00:00
2022-10-20 09:56:10 +00:00
message_url = f " https://api.telegram.org/bot { bot_token } /sendMessage?chat_id= { chat_id } &text= { text } "
2022-03-01 21:34:25 +00:00
# if it fails, it should keep trying
while True :
try :
self . session . get ( message_url ) . json ( )
return
2022-10-20 20:53:51 +00:00
except Exception :
2022-03-01 21:34:25 +00:00
pass
2022-10-20 09:56:10 +00:00
2022-02-22 01:50:32 +00:00
def welcome ( self , user ) :
2022-10-20 09:56:10 +00:00
""" User enabled Telegram Notifications """
2023-05-01 10:30:53 +00:00
lang = user . robot . telegram_lang_code
2022-02-24 21:59:16 +00:00
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " 🔔 Hola { user . username } , te enviaré notificaciones sobre tus órdenes en RoboSats. "
2022-02-21 23:41:36 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " 🔔 Hey { user . username } , I will send you notifications about your RoboSats orders. "
2023-05-01 10:30:53 +00:00
self . send_message ( user . robot . telegram_chat_id , text )
user . robot . telegram_welcomed = True
2023-05-08 18:10:37 +00:00
user . robot . save ( update_fields = [ " telegram_welcomed " ] )
2022-02-21 23:41:36 +00:00
return
2022-03-11 15:24:39 +00:00
def order_taken_confirmed ( self , order ) :
2023-05-01 10:30:53 +00:00
if order . maker . robot . telegram_enabled :
lang = order . maker . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { order . maker . username } ¡Tu orden con ID { order . id } ha sido tomada por { order . taker . username } !🥳 Visita http:// { self . site } /order/ { order . id } para continuar. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { order . maker . username } , your order was taken by { order . taker . username } !🥳 Visit http:// { self . site } /order/ { order . id } to proceed with the trade. "
2023-05-01 10:30:53 +00:00
self . send_message ( order . maker . robot . telegram_chat_id , text )
2022-09-15 15:42:33 +00:00
2023-05-01 10:30:53 +00:00
if order . taker . robot . telegram_enabled :
lang = order . taker . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { order . taker . username } , acabas de tomar la orden con ID { order . id } . "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { order . taker . username } , you just took the order with ID { order . id } . "
2023-05-01 10:30:53 +00:00
self . send_message ( order . taker . robot . telegram_chat_id , text )
2022-03-11 15:24:39 +00:00
return
2022-04-29 18:54:20 +00:00
def fiat_exchange_starts ( self , order ) :
2022-09-15 15:42:33 +00:00
for user in [ order . maker , order . taker ] :
2023-05-01 10:30:53 +00:00
if user . robot . telegram_enabled :
lang = user . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { user . username } , el depósito de garantía y el recibo del comprador han sido recibidos. Es hora de enviar el dinero fiat. Visita http:// { self . site } /order/ { order . id } para hablar con tu contraparte. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { user . username } , the escrow and invoice have been submitted. The fiat exchange starts now via the platform chat. Visit http:// { self . site } /order/ { order . id } to talk with your counterpart. "
2023-05-01 10:30:53 +00:00
self . send_message ( user . robot . telegram_chat_id , text )
2022-04-29 18:54:20 +00:00
return
2022-02-22 17:50:01 +00:00
def order_expired_untaken ( self , order ) :
2023-05-01 10:30:53 +00:00
if order . maker . robot . telegram_enabled :
lang = order . maker . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " 😪 Hey { order . maker . username } , tu orden con ID { order . id } ha expirado sin ser tomada por ningún robot. Visita http:// { self . site } /order/ { order . id } para renovarla. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " 😪 Hey { order . maker . username } , your order with ID { order . id } has expired without a taker. Visit http:// { self . site } /order/ { order . id } to renew it. "
2023-05-01 10:30:53 +00:00
self . send_message ( order . maker . robot . telegram_chat_id , text )
2022-02-22 17:50:01 +00:00
return
def trade_successful ( self , order ) :
2022-09-15 15:42:33 +00:00
for user in [ order . maker , order . taker ] :
2023-05-01 10:30:53 +00:00
if user . robot . telegram_enabled :
lang = user . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " 🥳 ¡Tu orden con ID { order . id } ha finalizado exitosamente!⚡ Únete a nosotros en @robosats_es y ayúdanos a mejorar. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " 🥳 Your order with ID { order . id } has finished successfully!⚡ Join us @robosats and help us improve. "
2023-05-01 10:30:53 +00:00
self . send_message ( user . robot . telegram_chat_id , text )
2022-02-23 16:15:48 +00:00
return
def public_order_cancelled ( self , order ) :
2023-05-01 10:30:53 +00:00
if order . maker . robot . telegram_enabled :
lang = order . maker . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ❌ Hey { order . maker . username } , has cancelado tu orden pública con ID { order . id } . "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ❌ Hey { order . maker . username } , you have cancelled your public order with ID { order . id } . "
2023-05-01 10:30:53 +00:00
self . send_message ( order . maker . robot . telegram_chat_id , text )
2022-03-01 13:38:04 +00:00
return
2022-10-20 09:56:10 +00:00
2022-06-02 22:32:01 +00:00
def collaborative_cancelled ( self , order ) :
2022-09-15 15:42:33 +00:00
for user in [ order . maker , order . taker ] :
2023-05-01 10:30:53 +00:00
if user . robot . telegram_enabled :
lang = user . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ❌ Hey { user . username } , tu orden con ID { str ( order . id ) } fue cancelada colaborativamente. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ❌ Hey { user . username } , your order with ID { str ( order . id ) } has been collaboratively cancelled. "
2023-05-01 10:30:53 +00:00
self . send_message ( user . robot . telegram_chat_id , text )
2022-06-02 22:32:01 +00:00
return
2022-10-20 09:56:10 +00:00
2022-06-02 22:32:01 +00:00
def dispute_opened ( self , order ) :
2022-09-15 15:42:33 +00:00
for user in [ order . maker , order . taker ] :
2023-05-01 10:30:53 +00:00
if user . robot . telegram_enabled :
lang = user . robot . telegram_lang_code
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ⚖️ Hey { user . username } , la orden con ID { str ( order . id ) } ha entrado en disputa. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ⚖️ Hey { user . username } , a dispute has been opened on your order with ID { str ( order . id ) } . "
2023-05-01 10:30:53 +00:00
self . send_message ( user . robot . telegram_chat_id , text )
2024-03-28 01:23:24 +00:00
admin_chat_id = config ( " TELEGRAM_ADMIN_CHAT_ID " )
if len ( admin_chat_id ) == 0 :
return
coordinator_text = f " There is a new dispute opened for the order with ID { str ( order . id ) } . Visit http:// { self . site } /coordinator/api/order/ { str ( order . id ) } /change to proceed. "
self . send_message ( admin_chat_id , coordinator_text )
2022-06-02 22:32:01 +00:00
return
2022-03-01 13:38:04 +00:00
def order_published ( self , order ) :
2023-05-01 10:30:53 +00:00
if order . maker . robot . telegram_enabled :
lang = order . maker . robot . telegram_lang_code
2022-09-15 15:42:33 +00:00
# In weird cases the order cannot be found (e.g. it is cancelled)
queryset = Order . objects . filter ( maker = order . maker )
if len ( queryset ) == 0 :
return
order = queryset . last ( )
2022-10-20 09:56:10 +00:00
if lang == " es " :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { order . maker . username } , tu orden con ID { str ( order . id ) } es pública en el libro de ordenes. "
2022-09-15 15:42:33 +00:00
else :
2023-03-27 09:37:36 +00:00
text = f " ✅ Hey { order . maker . username } , your order with ID { str ( order . id ) } is public in the order book. "
2023-05-01 10:30:53 +00:00
self . send_message ( order . maker . robot . telegram_chat_id , text )
2022-06-22 13:09:41 +00:00
return
2023-03-27 09:37:36 +00:00
def new_chat_message ( self , order , chat_message ) :
"""
Sends a TG notification for a new in - app chat message if
the last chat was at least CHAT_NOTIFICATION_TIMEGAP minutes ago .
"""
from datetime import timedelta
from django . utils import timezone
from chat . models import Message
TIMEGAP = config ( " CHAT_NOTIFICATION_TIMEGAP " , cast = int , default = 5 )
if chat_message . index > 1 :
previous_message = Message . objects . get (
chatroom = chat_message . chatroom , index = ( chat_message . index - 1 )
)
notification_reason = f " (You receive this notification only because more than { TIMEGAP } minutes have passed since the last in-chat message) "
if previous_message . created_at > timezone . now ( ) - timedelta (
minutes = TIMEGAP
) :
return
else :
notification_reason = f " (You receive this notification because this was the first in-chat message. You will only be notified again if there is a gap bigger than { TIMEGAP } minutes between messages) "
user = chat_message . receiver
2023-05-01 10:30:53 +00:00
if user . robot . telegram_enabled :
2023-03-27 09:37:36 +00:00
text = f " 💬 Hey { user . username } , a new chat message in-app was sent to you by { chat_message . sender . username } for order ID { str ( order . id ) } . { notification_reason } "
2023-05-01 10:30:53 +00:00
self . send_message ( user . robot . telegram_chat_id , text )
2023-03-27 09:37:36 +00:00
return
2024-01-28 13:52:30 +00:00
def coordinator_cancelled ( self , order ) :
if order . maker . robot . telegram_enabled :
text = f " 🛠️ Your order with ID { order . id } has been cancelled by the coordinator { config ( ' COORDINATOR_ALIAS ' , cast = str , default = ' NoAlias ' ) } for the upcoming maintenance stop. "
self . send_message ( order . maker . robot . telegram_chat_id , text )
return