2019-11-12 17:13:56 +00:00
|
|
|
import base64
|
2019-11-24 16:22:30 +00:00
|
|
|
import json
|
2019-11-12 17:13:56 +00:00
|
|
|
import os
|
|
|
|
import tempfile
|
|
|
|
from io import BytesIO
|
|
|
|
from uuid import uuid4
|
|
|
|
|
|
|
|
import requests
|
|
|
|
from PIL import Image
|
|
|
|
from django.db import models
|
2019-11-24 16:22:30 +00:00
|
|
|
from telegram import Update, Bot
|
2022-02-22 22:58:23 +00:00
|
|
|
from telegram.ext import Dispatcher, CallbackContext, MessageHandler, Filters, CommandHandler
|
2019-11-12 17:13:56 +00:00
|
|
|
from jsonrpc import Dispatcher as RPCDispatcher
|
2019-11-24 16:22:30 +00:00
|
|
|
from djconfig import config
|
2019-11-12 17:13:56 +00:00
|
|
|
|
2021-03-15 18:59:34 +00:00
|
|
|
from bots.models import TelegramBotModuleConfig, BotUser
|
2020-05-05 08:47:40 +00:00
|
|
|
|
|
|
|
|
2019-11-12 17:13:56 +00:00
|
|
|
class ChannelHelperBotModuleConfig(TelegramBotModuleConfig):
|
|
|
|
chat_id = models.CharField(max_length=32)
|
2019-11-24 16:22:30 +00:00
|
|
|
queued = models.BooleanField(default=False)
|
2021-03-15 18:59:34 +00:00
|
|
|
users = models.ManyToManyField(BotUser)
|
2019-11-12 17:13:56 +00:00
|
|
|
|
|
|
|
MODULE_NAME = 'Channel helper'
|
|
|
|
|
|
|
|
rpc_dispatcher = None
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.rpc_dispatcher = RPCDispatcher()
|
|
|
|
self.rpc_dispatcher['post_photo'] = self.rpc_post_photo
|
|
|
|
|
|
|
|
def rpc_post_photo(self, photo, is_base64=False):
|
2019-11-24 16:22:30 +00:00
|
|
|
config._reload_maybe()
|
|
|
|
bot = self.bot.get_bot()
|
2019-11-12 17:13:56 +00:00
|
|
|
try:
|
|
|
|
if is_base64:
|
|
|
|
f = BytesIO(base64.b64decode(photo))
|
|
|
|
else:
|
|
|
|
resp = requests.get(photo)
|
|
|
|
resp.raise_for_status()
|
|
|
|
f = BytesIO(resp.content)
|
|
|
|
except:
|
|
|
|
raise RuntimeError('Could not load image')
|
|
|
|
im = Image.open(f) # type: Image.Image
|
|
|
|
width, height = im.size
|
|
|
|
if width > 2000 or height > 2000:
|
|
|
|
im.thumbnail((2000, 2000))
|
2019-11-12 17:52:08 +00:00
|
|
|
im = im.convert('RGB')
|
2019-11-12 17:13:56 +00:00
|
|
|
with tempfile.TemporaryDirectory() as d:
|
|
|
|
fpath = os.path.join(d, '{}.jpg'.format(uuid4()))
|
2019-11-12 17:52:08 +00:00
|
|
|
im.save(fpath)
|
2019-11-24 16:42:28 +00:00
|
|
|
if self.queued:
|
|
|
|
m = bot.send_photo(config.tmp_uploads_chat_id, open(fpath, 'rb'))
|
|
|
|
i = QueuedItem(config=self, type='photo', args=json.dumps([m.photo[-1].file_id]))
|
|
|
|
i.save()
|
|
|
|
else:
|
|
|
|
bot.send_photo(self.chat_id, open(fpath, 'rb'))
|
2019-11-12 17:13:56 +00:00
|
|
|
return True
|
|
|
|
|
2019-11-24 16:22:30 +00:00
|
|
|
def periodic_task(self, bot: Bot):
|
|
|
|
i = self.queued_items.order_by('?').first() # type: QueuedItem
|
|
|
|
if i:
|
|
|
|
i.send(bot)
|
|
|
|
i.delete()
|
|
|
|
|
2019-11-12 17:13:56 +00:00
|
|
|
def handle_message(self, update: Update, ctx: CallbackContext):
|
2020-05-05 08:47:40 +00:00
|
|
|
if self.users.count() and not self.users.filter(user_id=update.effective_user.id).count():
|
|
|
|
update.effective_message.reply_text('GTFO')
|
|
|
|
return
|
|
|
|
|
2019-11-12 17:13:56 +00:00
|
|
|
m = update.effective_message
|
|
|
|
bot = ctx.bot
|
2022-02-22 22:58:23 +00:00
|
|
|
i = QueuedItem(config=self, message_id=m.message_id)
|
2019-11-12 17:13:56 +00:00
|
|
|
if hasattr(m, 'audio') and m.audio:
|
|
|
|
a = m.audio
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'audio'
|
|
|
|
i.args = json.dumps([a.file_id, a.duration, a.performer, a.title])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'document') and m.document:
|
|
|
|
d = m.document
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'document'
|
|
|
|
i.args = json.dumps([d.file_id, d.file_name])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'photo') and m.photo:
|
|
|
|
p = m.photo
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'photo'
|
|
|
|
i.args = json.dumps([p[-1].file_id])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'sticker') and m.sticker:
|
|
|
|
s = m.sticker
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'sticker'
|
|
|
|
i.args = json.dumps([s.file_id])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'video') and m.video:
|
|
|
|
v = m.video
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'video'
|
|
|
|
i.args = json.dumps([v.file_id, v.duration])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'voice') and m.voice:
|
|
|
|
v = m.voice
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'voice'
|
|
|
|
i.args = json.dumps([v.file_id, v.duration])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'video_note') and m.video_note:
|
|
|
|
vn = m.video_note
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'video_note'
|
|
|
|
i.args = json.dumps([vn.file_id, vn.duration, vn.length])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'contact') and m.contact:
|
|
|
|
c = m.contact
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'contact'
|
|
|
|
i.args = json.dumps([c.phone_number, c.first_name, c.last_name])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'location') and m.location:
|
|
|
|
l = m.location
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'location'
|
|
|
|
i.args = json.dumps([l.latitude, l.longitude])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'venue') and m.venue:
|
|
|
|
v = m.venue
|
2019-11-24 16:22:30 +00:00
|
|
|
i.type = 'venue'
|
|
|
|
i.args = json.dumps([v.location.latitude, v.location.longitude, v.title, v.address, v.foursquare_id])
|
2019-11-12 17:13:56 +00:00
|
|
|
elif hasattr(m, 'text') and m.text:
|
2019-12-04 15:38:14 +00:00
|
|
|
i.type = 'message'
|
2019-11-24 16:22:30 +00:00
|
|
|
i.args = json.dumps([m.text_html, 'html'])
|
|
|
|
|
|
|
|
if self.queued:
|
|
|
|
i.save()
|
|
|
|
else:
|
|
|
|
i.send(bot)
|
2019-11-12 17:13:56 +00:00
|
|
|
|
2022-02-22 22:58:23 +00:00
|
|
|
def handle_delete(self, update: Update, ctx: CallbackContext):
|
|
|
|
if update.effective_message.chat_id != self.chat_id:
|
|
|
|
return
|
|
|
|
reply_to_id = update.effective_message.reply_to_message.message_id
|
|
|
|
try:
|
2022-03-23 09:30:10 +00:00
|
|
|
msg = QueuedItem.objects.get(message_id=reply_to_id, config=self)
|
2022-02-22 22:58:23 +00:00
|
|
|
msg.delete()
|
|
|
|
except QueuedItem.DoesNotExist:
|
|
|
|
update.effective_message.reply_text('Deleted')
|
|
|
|
|
2022-03-23 09:30:10 +00:00
|
|
|
def handle_count(self, update: Update, ctx: CallbackContext):
|
|
|
|
if update.effective_message.chat_id != self.chat_id:
|
|
|
|
return
|
|
|
|
update.effective_message.reply_text(f'{QueuedItem.objects.filter(config=self).count()} items queued')
|
|
|
|
|
2019-11-12 17:13:56 +00:00
|
|
|
def build_dispatcher(self, dispatcher: Dispatcher):
|
2022-02-22 22:58:23 +00:00
|
|
|
dispatcher.add_handler(CommandHandler(['delete', 'del', 'remove', 'rem'], self.handle_delete, Filters.reply))
|
2022-03-23 09:30:10 +00:00
|
|
|
dispatcher.add_handler(CommandHandler(['count'], self.handle_count))
|
|
|
|
dispatcher.add_handler(MessageHandler(Filters.private, self.handle_message))
|
2019-11-12 17:13:56 +00:00
|
|
|
return dispatcher
|
2019-11-24 16:22:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
class QueuedItem(models.Model):
|
|
|
|
config = models.ForeignKey(ChannelHelperBotModuleConfig, on_delete=models.CASCADE, related_name='queued_items')
|
|
|
|
type = models.CharField(max_length=12)
|
|
|
|
args = models.TextField()
|
2022-02-23 10:13:42 +00:00
|
|
|
message_id = models.PositiveBigIntegerField(default=None, db_index=True, null=True, blank=True)
|
2019-11-24 16:22:30 +00:00
|
|
|
|
|
|
|
def send(self, bot: Bot):
|
|
|
|
getattr(bot, 'send_' + self.type)(self.config.chat_id, *json.loads(self.args))
|