mirror of
				https://github.com/cdryzun/tg_bot_collections.git
				synced 2025-11-04 08:46:44 +08:00 
			
		
		
		
	* feat: add summary and search commands Signed-off-by: Frost Ming <me@frostming.com> * fix formats Signed-off-by: Frost Ming <me@frostming.com> * fix: clean up Signed-off-by: Frost Ming <me@frostming.com>
		
			
				
	
	
		
			133 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			133 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import gc
 | 
						|
import random
 | 
						|
import shutil
 | 
						|
from tempfile import SpooledTemporaryFile
 | 
						|
 | 
						|
import numpy as np
 | 
						|
import PIL.Image
 | 
						|
from matplotlib import figure
 | 
						|
from prettymapp.geo import get_aoi
 | 
						|
from prettymapp.osm import get_osm_geometries
 | 
						|
from prettymapp.plotting import Plot as PrettyPlot
 | 
						|
from prettymapp.settings import STYLES
 | 
						|
from telebot import TeleBot
 | 
						|
from telebot.types import Message
 | 
						|
 | 
						|
MAX_IN_MEMORY = 10 * 1024 * 1024  # 10MiB
 | 
						|
PIL.Image.MAX_IMAGE_PIXELS = 933120000
 | 
						|
 | 
						|
 | 
						|
class Plot(PrettyPlot):
 | 
						|
    # memory leak fix for Plot. thanks @higuoxing https://github.com/higuoxing
 | 
						|
    # refer to: https://www.mail-archive.com/matplotlib-users@lists.sourceforge.net/msg11809.html
 | 
						|
    def __post_init__(self):
 | 
						|
        (
 | 
						|
            self.xmin,
 | 
						|
            self.ymin,
 | 
						|
            self.xmax,
 | 
						|
            self.ymax,
 | 
						|
        ) = self.aoi_bounds
 | 
						|
        # take from aoi geometry bounds, otherwise problematic if unequal geometry distribution over plot.
 | 
						|
        self.xmid = (self.xmin + self.xmax) / 2
 | 
						|
        self.ymid = (self.ymin + self.ymax) / 2
 | 
						|
        self.xdif = self.xmax - self.xmin
 | 
						|
        self.ydif = self.ymax - self.ymin
 | 
						|
 | 
						|
        self.bg_buffer_x = (self.bg_buffer / 100) * self.xdif
 | 
						|
        self.bg_buffer_y = (self.bg_buffer / 100) * self.ydif
 | 
						|
 | 
						|
        # self.fig, self.ax = subplots(
 | 
						|
        #     1, 1, figsize=(12, 12), constrained_layout=True, dpi=1200
 | 
						|
        # )
 | 
						|
        self.fig = figure.Figure(figsize=(12, 12), constrained_layout=True, dpi=1200)
 | 
						|
        self.ax = self.fig.subplots(1, 1)
 | 
						|
        self.ax.set_aspect(1 / np.cos(self.ymid * np.pi / 180))
 | 
						|
 | 
						|
        self.ax.axis("off")
 | 
						|
        self.ax.set_xlim(self.xmin - self.bg_buffer_x, self.xmax + self.bg_buffer_x)
 | 
						|
        self.ax.set_ylim(self.ymin - self.bg_buffer_y, self.ymax + self.bg_buffer_y)
 | 
						|
 | 
						|
 | 
						|
def sizeof_image(image):
 | 
						|
    with SpooledTemporaryFile(max_size=MAX_IN_MEMORY) as f:
 | 
						|
        image.save(f, format="JPEG", quality=95)
 | 
						|
        return f.tell()
 | 
						|
 | 
						|
 | 
						|
def compress_image(input_image, output_image, target_size):
 | 
						|
    quality = 95
 | 
						|
    factor = 1.0
 | 
						|
    with PIL.Image.open(input_image) as img:
 | 
						|
        while sizeof_image(img) > target_size:
 | 
						|
            factor -= 0.05
 | 
						|
            width, height = img.size
 | 
						|
            img = img.resize(
 | 
						|
                (int(width * factor), int(height * factor)),
 | 
						|
                PIL.Image.Resampling.LANCZOS,
 | 
						|
            )
 | 
						|
        img.save(output_image, format="JPEG", quality=quality)
 | 
						|
    output_image.seek(0)
 | 
						|
 | 
						|
 | 
						|
def draw_pretty_map(location, style, output_file):
 | 
						|
    aoi = get_aoi(address=location, radius=1100, rectangular=True)
 | 
						|
    df = get_osm_geometries(aoi=aoi)
 | 
						|
    fig = Plot(df=df, aoi_bounds=aoi.bounds, draw_settings=STYLES[style]).plot_all()
 | 
						|
    with SpooledTemporaryFile(max_size=MAX_IN_MEMORY) as buffer:
 | 
						|
        fig.savefig(buffer, format="jpeg")
 | 
						|
        buffer.seek(0)
 | 
						|
        compress_image(
 | 
						|
            buffer,
 | 
						|
            output_file,
 | 
						|
            10 * 1024 * 1024,  # telegram tog need png less than 10MB
 | 
						|
        )
 | 
						|
 | 
						|
 | 
						|
def map_handler(message: Message, bot: TeleBot):
 | 
						|
    """pretty map: /map <address>"""
 | 
						|
    bot.reply_to(message, "Generating pretty map may take some time please wait:")
 | 
						|
    m = message.text.strip()
 | 
						|
    location = m.strip()
 | 
						|
    styles_list = list(STYLES.keys())
 | 
						|
    style = random.choice(styles_list)
 | 
						|
    with SpooledTemporaryFile(max_size=MAX_IN_MEMORY) as out_image:
 | 
						|
        try:
 | 
						|
            draw_pretty_map(location, style, out_image)
 | 
						|
            # tg can only send image less than 10MB
 | 
						|
            with open("map_out.jpg", "wb") as f:  # for debug
 | 
						|
                shutil.copyfileobj(out_image, f)
 | 
						|
            out_image.seek(0)
 | 
						|
            bot.send_photo(
 | 
						|
                message.chat.id, out_image, reply_to_message_id=message.message_id
 | 
						|
            )
 | 
						|
        finally:
 | 
						|
            gc.collect()
 | 
						|
 | 
						|
 | 
						|
def map_location_handler(message: Message, bot: TeleBot):
 | 
						|
    # TODO refactor the function
 | 
						|
    location = "{0}, {1}".format(message.location.latitude, message.location.longitude)
 | 
						|
    styles_list = list(STYLES.keys())
 | 
						|
    style = random.choice(styles_list)
 | 
						|
    try:
 | 
						|
        with SpooledTemporaryFile(max_size=MAX_IN_MEMORY) as out_image:
 | 
						|
            draw_pretty_map(location, style, out_image)
 | 
						|
            # tg can only send image less than 10MB
 | 
						|
            with open("map_out.jpg", "wb") as f:  # for debug
 | 
						|
                shutil.copyfileobj(out_image, f)
 | 
						|
            out_image.seek(0)
 | 
						|
            bot.send_photo(
 | 
						|
                message.chat.id, out_image, reply_to_message_id=message.message_id
 | 
						|
            )
 | 
						|
 | 
						|
    finally:
 | 
						|
        gc.collect()
 | 
						|
 | 
						|
 | 
						|
def register(bot: TeleBot) -> None:
 | 
						|
    bot.register_message_handler(map_handler, commands=["map"], pass_bot=True)
 | 
						|
    bot.register_message_handler(map_handler, regexp="^map:", pass_bot=True)
 | 
						|
    bot.register_message_handler(
 | 
						|
        map_location_handler, content_types=["location", "venue"], pass_bot=True
 | 
						|
    )
 |