MarseyWorld/files/helpers/media.py

147 lines
4.1 KiB
Python
Raw Normal View History

2022-05-04 23:09:46 +00:00
from PIL import Image, ImageOps
from PIL.ImageSequence import Iterator
from webptools import gifwebp
import subprocess
2022-05-07 06:04:14 +00:00
import os
2022-06-18 20:41:00 +00:00
from flask import abort, g
2022-05-22 16:13:19 +00:00
import requests
import time
from .const import *
2022-06-19 14:49:44 +00:00
import gevent
import imagehash
2022-05-22 22:15:29 +00:00
def process_files():
body = ''
if request.files.get("file") and request.headers.get("cf-ipcountry") != "T1":
files = request.files.getlist('file')[:4]
for file in files:
if file.content_type.startswith('image/'):
name = f'/images/{time.time()}'.replace('.','') + '.webp'
file.save(name)
2022-06-18 20:41:00 +00:00
url = process_image(name)
body += f"\n\n![]({url})"
elif file.content_type.startswith('video/'):
body += f"\n\n{process_video(file)}"
elif file.content_type.startswith('audio/'):
body += f"\n\n{process_audio(file)}"
else:
abort(415)
return body
2022-05-23 19:00:14 +00:00
def process_audio(file):
2022-06-22 17:44:43 +00:00
name = f'/audio/{time.time()}'.replace('.','')
extension = file.filename.split('.')[-1].lower()
if extension not in ['aac', 'amr', 'flac', 'm4a', 'm4b', 'mp3', 'ogg', 'wav']:
extension = 'mp3'
name = name + '.' + extension
2022-05-22 22:15:29 +00:00
file.save(name)
2022-05-23 18:03:59 +00:00
2022-10-06 04:31:08 +00:00
size = os.stat(name).st_size
if size > 16 * 1024 * 1024 or not g.v.patron and size > 8 * 1024 * 1024:
os.remove(name)
abort(413)
2022-05-23 18:03:59 +00:00
2022-05-22 22:15:29 +00:00
return f'{SITE_FULL}{name}'
2022-10-06 04:54:05 +00:00
def webm_to_mp4(old, new):
2022-10-06 05:05:03 +00:00
tmp = f"{new}.tmp"
subprocess.run(["ffmpeg", "-y", "-loglevel", "warning", "-i", old, "-map_metadata", "-1", tmp, "-threads:v", "1"], check=True, stderr=subprocess.STDOUT)
2022-10-06 05:07:24 +00:00
os.replace(tmp, new)
2022-10-06 04:54:05 +00:00
os.remove(old)
requests.post(f'https://api.cloudflare.com/client/v4/zones/{CF_ZONE}/purge_cache', headers=CF_HEADERS,
data=f'{{"files": ["{SITE_FULL}{new}"]}}', timeout=5)
2022-05-22 16:13:19 +00:00
def process_video(file):
2022-05-24 20:07:04 +00:00
old = f'/videos/{time.time()}'.replace('.','')
2022-06-19 13:03:14 +00:00
file.save(old)
2022-06-22 17:44:43 +00:00
2022-10-06 04:31:08 +00:00
size = os.stat(old).st_size
if SITE_NAME != 'WPD' and (size > 32 * 1024 * 1024 or not g.v.patron and size > 64 * 1024 * 1024):
os.remove(old)
abort(414)
2022-06-22 17:44:43 +00:00
extension = file.filename.split('.')[-1].lower()
if extension not in ['avi', 'mp4', 'webm', 'm4v', 'mov', 'mkv']:
extension = 'mp4'
new = old + '.' + extension
2022-05-24 20:07:04 +00:00
2022-10-06 04:54:05 +00:00
if extension == 'webm':
new = new.replace('.webm', '.mp4')
file.save(new)
gevent.spawn(webm_to_mp4, old, new)
else:
subprocess.run(["ffmpeg", "-y", "-loglevel", "warning", "-i", old, "-map_metadata", "-1", "-c:v", "copy", "-c:a", "copy", new], check=True)
os.remove(old)
2022-09-18 20:39:40 +00:00
return f'{SITE_FULL}{new}'
2022-05-23 19:00:14 +00:00
2022-05-22 16:13:19 +00:00
2022-05-07 06:04:14 +00:00
def process_image(filename=None, resize=0, trim=False):
2022-05-07 06:04:14 +00:00
size = os.stat(filename).st_size
2022-10-06 04:31:08 +00:00
if size > 16 * 1024 * 1024 or not g.v.patron and size > 8 * 1024 * 1024:
os.remove(filename)
abort(413)
2022-05-04 23:09:46 +00:00
i = Image.open(filename)
if resize and i.width > resize:
if trim and len(list(Iterator(i))) == 1:
subprocess.run(["convert", filename, "-coalesce", "-trim", "-resize", f"{resize}>", filename])
else:
try: subprocess.run(["convert", filename, "-coalesce", "-resize", f"{resize}>", filename])
except: pass
2022-05-04 23:09:46 +00:00
elif i.format.lower() != "webp":
exif = i.getexif()
for k in exif.keys():
if k != 0x0112:
exif[k] = None
del exif[k]
i.info["exif"] = exif.tobytes()
if i.format.lower() == "gif":
2022-08-12 23:40:59 +00:00
gifwebp(input_image=filename, output_image=filename, option="-mixed -metadata none -f 100 -mt -m 6")
2022-05-04 23:09:46 +00:00
else:
i = ImageOps.exif_transpose(i)
2022-08-12 23:40:59 +00:00
i.save(filename, format="WEBP", method=6, quality=88)
2022-05-04 23:09:46 +00:00
if resize in (300,400,1200):
if os.stat(filename).st_size > 1 * 1024 * 1024:
os.remove(filename)
abort(413)
if resize == 1200:
path = f'files/assets/images/{SITE_NAME}/banners'
elif resize == 400:
path = f'files/assets/images/{SITE_NAME}/sidebar'
else:
path = f'files/assets/images/badges'
hashes = {}
for img in os.listdir(path):
if resize == 400 and img in ('256.webp','585.webp'): continue
img_path = f'{path}/{img}'
if img_path == filename: continue
img = Image.open(img_path)
i_hash = str(imagehash.phash(img))
if i_hash in hashes.keys():
print(hashes[i_hash], flush=True)
print(img_path, flush=True)
else: hashes[i_hash] = img_path
i = Image.open(filename)
i_hash = str(imagehash.phash(i))
if i_hash in hashes.keys():
os.remove(filename)
abort(417)
2022-09-26 01:27:31 +00:00
return filename