PaiGram/service/template.py
2022-06-09 23:25:19 +08:00

105 lines
4.3 KiB
Python

import asyncio
import os
import time
from typing import Optional
from jinja2 import PackageLoader, Environment, Template
from playwright.async_api import async_playwright, Browser, ViewportSize, Playwright
from config import config
from logger import Log
class TemplateService:
def __init__(self, template_package_name: str = "resources", cache_dir_name: str = "cache", loop=None):
self._template_package_name = template_package_name
self._browser: Optional[Browser] = None
self._playwright: Optional[Playwright] = None
self._current_dir = os.getcwd()
self._output_dir = os.path.join(self._current_dir, cache_dir_name)
if not os.path.exists(self._output_dir):
os.mkdir(self._output_dir)
self._jinja2_env = {}
self._jinja2_template = {}
self._loop = loop
if self._loop is None:
self._loop = asyncio.get_event_loop()
try:
Log.info("正在尝试启动Playwright")
self._loop.run_until_complete(self._browser_init())
Log.info("启动Playwright成功")
except (KeyboardInterrupt, SystemExit):
pass
except Exception as exc:
Log.error("启动浏览器失败 \n")
raise exc
async def _browser_init(self) -> Browser:
if self._playwright is None:
self._playwright = await async_playwright().start()
try:
self._browser = await self._playwright.chromium.launch(timeout=5000)
except TimeoutError as err:
raise err
else:
if self._browser is None:
try:
self._browser = await self._playwright.chromium.launch(timeout=10000)
except TimeoutError as err:
raise err
return self._browser
async def close(self):
if self._browser is not None:
await self._browser.close()
if self._playwright is not None:
await self._playwright.stop()
async def get_browser(self) -> Browser:
if self._browser is None:
return await self._browser_init()
return self._browser
def get_template(self, package_path: str, template_name: str) -> Template:
if config.DEBUG:
# DEBUG下 禁止复用 方便查看和修改模板
loader = PackageLoader(self._template_package_name, package_path)
jinja2_env = Environment(loader=loader, enable_async=True)
jinja2_template = jinja2_env.get_template(template_name)
else:
jinja2_env: Environment = self._jinja2_env.get(package_path)
jinja2_template: Template = self._jinja2_template.get(package_path + template_name)
if jinja2_env is None:
loader = PackageLoader(self._template_package_name, package_path)
jinja2_env = Environment(loader=loader, enable_async=True)
jinja2_template = jinja2_env.get_template(template_name)
self._jinja2_env[package_path] = jinja2_env
self._jinja2_template[package_path + template_name] = jinja2_template
return jinja2_template
async def render(self, template_path: str, template_name: str, template_data: dict,
viewport: ViewportSize, full_page: bool = True) -> bytes:
"""
模板渲染成图片
:param template_path: 模板目录
:param template_name: 模板文件名
:param template_data: 模板数据
:param viewport: 截图大小
:param full_page: 是否长截图
:return:
"""
start_time = time.time()
template = self.get_template(template_path, template_name)
template_data["res_path"] = "file://" + self._current_dir
html = await template.render_async(**template_data)
Log.info(f"{template_name} 模板渲染使用了 {str(time.time() - start_time)}")
browser = await self.get_browser()
start_time = time.time()
page = await browser.new_page(viewport=viewport)
await page.goto("file://" + template.filename)
await page.set_content(html, wait_until="networkidle")
png_data = await page.screenshot(full_page=full_page)
await page.close()
Log.info(f"{template_name} 图片渲染使用了 {str(time.time() - start_time)}")
return png_data