mirror of
https://github.com/MarcZierle/photo-log-backend.git
synced 2025-01-04 05:37:58 +00:00
443 lines
12 KiB
Python
443 lines
12 KiB
Python
from celery import shared_task, chain, group, chord
|
|
import boto3
|
|
from boto3.s3.transfer import TransferConfig
|
|
import imghdr
|
|
from PIL import Image, ExifTags
|
|
|
|
import os, shutil
|
|
from uuid import uuid4
|
|
|
|
from django.conf import settings
|
|
from django.core.files import File
|
|
from django.apps import apps
|
|
|
|
from channels.layers import get_channel_layer
|
|
from asgiref.sync import async_to_sync
|
|
|
|
from django_tex.shortcuts import compile_template_to_pdf
|
|
|
|
from .photolog_layout import generate_tex
|
|
from .autocrop.autocrop import autocrop
|
|
|
|
|
|
s3_resource = boto3.resource(
|
|
's3',
|
|
endpoint_url=settings.AWS_S3_ENDPOINT_URL,
|
|
aws_access_key_id=settings.AWS_ACCESS_KEY_ID,
|
|
aws_secret_access_key=settings.AWS_SECRET_ACCESS_KEY,
|
|
)
|
|
client = s3_resource.meta.client
|
|
|
|
|
|
@shared_task
|
|
def chordfinisher(*args, **kwargs):
|
|
"""
|
|
Used at the end of chord( group( ... ), chordfinisher.s())
|
|
to chain multiple groups together.
|
|
"""
|
|
return 'OK'
|
|
|
|
|
|
@shared_task
|
|
def notify_client(description, content):
|
|
channel_layer = get_channel_layer()
|
|
async_to_sync(channel_layer.group_send)("notifications", {
|
|
"type": "task.finished",
|
|
"description": description,
|
|
"content": content
|
|
})
|
|
|
|
|
|
@shared_task
|
|
def download_s3_file(folder_path, s3_file_path, bucket):
|
|
"""
|
|
Downloads a file stored in a S3 object storages.
|
|
:param folder_path Path on local disk where the file should be saved.
|
|
:param s3_file_path Full file path in the S3 storage bucket (without the buckets name).
|
|
:param bucket The name of the bucket where the file is stored in.
|
|
"""
|
|
|
|
global client
|
|
|
|
# create local folder
|
|
if not os.path.exists(folder_path):
|
|
os.makedirs(folder_path, exist_ok=True) # mkdir -p
|
|
|
|
# retrieve the file name if the file is stored in a sub dir on the S3
|
|
file_name = s3_file_path.split('/')[-1]
|
|
|
|
client.download_file(
|
|
bucket,
|
|
s3_file_path,
|
|
os.path.join(folder_path, file_name),
|
|
Config=TransferConfig(use_threads=False)
|
|
)
|
|
|
|
|
|
@shared_task
|
|
def upload_s3_file(file_path, s3_file_path, bucket, content_type='application/octet-stream'):
|
|
"""
|
|
Uploads a file to a S3 object storages.
|
|
:param file_path Path on local disk where the is saved.
|
|
:param s3_file_path Full file path in the S3 storage bucket (without the buckets name).
|
|
:param bucket The name of the bucket where the file will be stored in.
|
|
"""
|
|
|
|
global client
|
|
|
|
client.upload_file(
|
|
file_path,
|
|
bucket,
|
|
s3_file_path,
|
|
ExtraArgs={
|
|
'ContentDisposition': 'inline',
|
|
'ContentType': content_type,
|
|
},
|
|
Config=TransferConfig(use_threads=False)
|
|
)
|
|
|
|
|
|
@shared_task
|
|
def update_model_field(app_name, model_name, pk, field_name, new_value):
|
|
model = apps.get_model(app_name, model_name)
|
|
instance = model.objects.get(pk=pk)
|
|
setattr(instance, field_name, new_value)
|
|
instance.save()
|
|
|
|
|
|
def rotateByExif(img):
|
|
try:
|
|
for orientation in ExifTags.TAGS.keys():
|
|
if ExifTags.TAGS[orientation]=='Orientation':
|
|
break
|
|
|
|
exif = img._getexif()
|
|
|
|
if not exif:
|
|
return img
|
|
|
|
if exif[orientation] == 3:
|
|
img=img.rotate(180, expand=True)
|
|
elif exif[orientation] == 6:
|
|
img=img.rotate(270, expand=True)
|
|
elif exif[orientation] == 8:
|
|
img=img.rotate(90, expand=True)
|
|
|
|
except (AttributeError, KeyError, IndexError):
|
|
# cases: image don't have getexif
|
|
pass
|
|
return img
|
|
|
|
|
|
@shared_task
|
|
def max_resize_image(image_path, max_width):
|
|
if not imghdr.what(image_path):
|
|
return
|
|
|
|
img = Image.open(image_path)
|
|
img = rotateByExif(img)
|
|
|
|
if img.size[0] <= max_width:
|
|
return
|
|
|
|
wpercent = (max_width/float(img.size[0]))
|
|
hsize = int((float(img.size[1])*float(wpercent)))
|
|
|
|
img = img.resize((max_width,hsize), Image.ANTIALIAS)
|
|
img.save(image_path)
|
|
|
|
|
|
@shared_task
|
|
def crop_image_bbox(image_path, bbox, rotate_angle):
|
|
if not imghdr.what(image_path):
|
|
return
|
|
|
|
img = Image.open(image_path)
|
|
img = rotateByExif(img)
|
|
|
|
if rotate_angle:
|
|
img = img.rotate(rotate_angle, expand=1)
|
|
|
|
if bbox:
|
|
img = img.crop((
|
|
bbox[0][0],
|
|
bbox[0][1],
|
|
bbox[2][0],
|
|
bbox[2][1]
|
|
))
|
|
|
|
img = img.convert('RGB')
|
|
img.save(image_path)
|
|
|
|
|
|
@shared_task
|
|
def crop_image_auto(image_path):
|
|
if not imghdr.what(image_path):
|
|
return
|
|
|
|
img = Image.open(image_path)
|
|
img = rotateByExif(img)
|
|
|
|
try:
|
|
cropped_img, _, bbox, intersections = autocrop(img)
|
|
except Exception:
|
|
cropped_img = img
|
|
bbox = None
|
|
intersections = None
|
|
|
|
# TODO: save bbox an intersections in photo.bbox_cords and photo.intersections
|
|
|
|
cropped_img.save(image_path)
|
|
|
|
|
|
@shared_task
|
|
def delete_folder(folder_path):
|
|
shutil.rmtree(folder_path, ignore_errors=True)
|
|
|
|
|
|
@shared_task
|
|
def generate_photolog_from_latex(title, date, render_date, start_slide_image, slides, id_to_name, work_dir, out_file):
|
|
try:
|
|
template_name = 'photolog.tex'
|
|
context = {
|
|
'content': generate_tex(title, date, render_date, start_slide_image, slides, id_to_name, work_dir)
|
|
}
|
|
|
|
pdf_bytes = compile_template_to_pdf(template_name, context)
|
|
|
|
with open(out_file, 'wb+') as file:
|
|
file.write(pdf_bytes)
|
|
except Exception as e:
|
|
notify_client(
|
|
description='error',
|
|
content={
|
|
'exception': str(e),
|
|
}
|
|
)
|
|
raise e
|
|
|
|
|
|
def max_resize_image_chain(full_file_name, max_width, work_folder_name=uuid4()):
|
|
BASE_DIR = settings.CELERY_WORK_DIR
|
|
|
|
folder = os.path.join(BASE_DIR, str(work_folder_name))
|
|
local_file = os.path.join(folder, full_file_name.split('/')[-1])
|
|
|
|
bucket = settings.AWS_STORAGE_BUCKET_NAME
|
|
|
|
chain(
|
|
download_s3_file.si(
|
|
folder,
|
|
full_file_name,
|
|
bucket
|
|
),
|
|
max_resize_image.si(
|
|
local_file,
|
|
max_width=max_width
|
|
),
|
|
upload_s3_file.si(
|
|
local_file,
|
|
full_file_name,
|
|
bucket,
|
|
content_type='image/png'
|
|
),
|
|
delete_folder.si(
|
|
folder
|
|
),
|
|
)()
|
|
|
|
|
|
def crop_image_bbox_chain(photo_pk, full_file_name, bbox, rotate_angle, cropped_img_name=None, work_folder_name=uuid4()):
|
|
BASE_DIR = settings.CELERY_WORK_DIR
|
|
|
|
if not cropped_img_name:
|
|
cropped_img_name = os.path.join('cropped_images', str(uuid4()) + '.png')
|
|
|
|
folder = os.path.join(BASE_DIR, str(work_folder_name))
|
|
local_file = os.path.join(folder, os.path.basename(full_file_name))
|
|
|
|
bucket = settings.AWS_STORAGE_BUCKET_NAME
|
|
s3_upload_name = 'cropped_images/' + os.path.basename(cropped_img_name)
|
|
|
|
chain(
|
|
download_s3_file.si(
|
|
folder,
|
|
full_file_name,
|
|
bucket
|
|
),
|
|
crop_image_bbox.si(
|
|
local_file,
|
|
bbox,
|
|
rotate_angle
|
|
),
|
|
upload_s3_file.si(
|
|
local_file,
|
|
s3_upload_name,
|
|
bucket,
|
|
content_type='image/png'
|
|
),
|
|
update_model_field.si(# photo_log.Photo(pk=photo_pk).cropped_image = s3_upload_name
|
|
'photo_log', 'Photo', photo_pk,
|
|
'cropped_image', s3_upload_name
|
|
),
|
|
notify_client.si(
|
|
description='update_photo',
|
|
content={
|
|
'id': photo_pk
|
|
}
|
|
),
|
|
delete_folder.si(
|
|
folder
|
|
),
|
|
)()
|
|
|
|
|
|
def crop_image_auto_chain(photo_pk, full_file_name, cropped_img_name=None, work_folder_name=uuid4()):
|
|
BASE_DIR = settings.CELERY_WORK_DIR
|
|
|
|
if not cropped_img_name:
|
|
cropped_img_name = os.path.join('cropped_images', str(uuid4()) + '.png')
|
|
|
|
folder = os.path.join(BASE_DIR, str(work_folder_name))
|
|
local_file = os.path.join(folder, os.path.basename(full_file_name))
|
|
|
|
bucket = settings.AWS_STORAGE_BUCKET_NAME
|
|
s3_upload_name = 'cropped_images/' + os.path.basename(cropped_img_name)
|
|
|
|
chain(
|
|
download_s3_file.si(
|
|
folder,
|
|
full_file_name,
|
|
bucket
|
|
),
|
|
crop_image_auto.si(
|
|
local_file
|
|
),
|
|
upload_s3_file.si(
|
|
local_file,
|
|
s3_upload_name,
|
|
bucket,
|
|
content_type='image/png'
|
|
),
|
|
update_model_field.si(# photo_log.Photo(pk=photo_pk).cropped_image = s3_upload_name
|
|
'photo_log', 'Photo', photo_pk,
|
|
'cropped_image', s3_upload_name
|
|
),
|
|
delete_folder.si(
|
|
folder
|
|
),
|
|
)()
|
|
|
|
|
|
def get_photo_log_assets_tasks():
|
|
BASE_DIR = settings.CELERY_WORK_DIR
|
|
folder = os.path.join(BASE_DIR, 'photolog_assets')
|
|
|
|
bucket = settings.AWS_STORAGE_BUCKET_NAME
|
|
|
|
download_tasks = []
|
|
|
|
asset_files = [
|
|
'Gill Sans MT Bold.ttf',
|
|
'Gill Sans MT Medium.ttf',
|
|
'smile.png',
|
|
'wood_floor.jpg'
|
|
]
|
|
|
|
for file in asset_files:
|
|
path = os.path.join(folder, file)
|
|
if not os.path.exists(path):
|
|
download_tasks.append(
|
|
download_s3_file.si(
|
|
folder,
|
|
'photolog_assets/' + file,
|
|
bucket
|
|
)
|
|
)
|
|
|
|
return download_tasks
|
|
|
|
|
|
def generate_photo_log_chain(photo_log_id, work_folder_name=uuid4()):
|
|
from photo_log.models import Photo, PhotoLog
|
|
|
|
BASE_DIR = settings.CELERY_WORK_DIR
|
|
folder = os.path.join(BASE_DIR, str(work_folder_name))
|
|
|
|
bucket = settings.AWS_STORAGE_BUCKET_NAME
|
|
|
|
photo_log = PhotoLog.objects.get(pk=photo_log_id)
|
|
|
|
slides = photo_log.slides
|
|
slides = [photo for slide in slides for photo in slide] # flatten slides lists
|
|
|
|
if photo_log.start_slide_image:
|
|
slides.append(photo_log.start_slide_image)
|
|
|
|
photos = Photo.objects.filter(pk__in=slides).values('id', 'original_image', 'cropped_image')
|
|
|
|
photo_id_to_file_name = {}
|
|
download_files_tasks = []
|
|
for photo in photos:
|
|
image = photo['cropped_image']
|
|
if not image:
|
|
image = photo['original_image']
|
|
|
|
download_files_tasks.append(
|
|
download_s3_file.si(
|
|
folder,
|
|
image,
|
|
bucket
|
|
)
|
|
)
|
|
|
|
photo_id_to_file_name[photo['id']], _ = os.path.splitext(os.path.basename(image))
|
|
|
|
download_files_tasks.extend(get_photo_log_assets_tasks())
|
|
|
|
#download_files_tasks = chord( group(download_files_tasks), chordfinisher.si() )
|
|
download_files_tasks = chain(download_files_tasks)
|
|
|
|
pdf_file = photo_log.pdf.name
|
|
if pdf_file:
|
|
pdf_file = os.path.basename(pdf_file)
|
|
else:
|
|
pdf_file = os.path.join(str(uuid4()) + '.pdf')
|
|
|
|
pdf_file = os.path.join(folder, pdf_file)
|
|
pdf_s3_path = 'photolog_pdf/' + os.path.basename(pdf_file)
|
|
|
|
chain(
|
|
download_files_tasks,
|
|
generate_photolog_from_latex.si(
|
|
photo_log.title,
|
|
photo_log.date,
|
|
photo_log.render_date,
|
|
photo_log.start_slide_image,
|
|
photo_log.slides,
|
|
photo_id_to_file_name,
|
|
folder,
|
|
pdf_file
|
|
),
|
|
upload_s3_file.si(
|
|
pdf_file,
|
|
pdf_s3_path,
|
|
bucket,
|
|
content_type='application/pdf'
|
|
),
|
|
update_model_field.si(
|
|
'photo_log', 'PhotoLog', photo_log.id,
|
|
'pdf', pdf_s3_path
|
|
),
|
|
notify_client.si(
|
|
description='update_photolog_pdf',
|
|
content={
|
|
'pdf': pdf_s3_path,
|
|
'id': photo_log.id
|
|
}
|
|
),
|
|
delete_folder.si(
|
|
folder
|
|
)
|
|
)()
|