Commit 26d0a91b authored by Seb's avatar Seb
Browse files

Track models + signals refactor to work with swappable models

parent 392515fa
This diff is collapsed.
...@@ -5,7 +5,7 @@ from wagtail.admin.edit_handlers import StreamFieldPanel ...@@ -5,7 +5,7 @@ from wagtail.admin.edit_handlers import StreamFieldPanel
from wagtailvideos.edit_handlers import VideoChooserPanel from wagtailvideos.edit_handlers import VideoChooserPanel
from wagtailvideos.blocks import VideoChooserBlock from wagtailvideos.blocks import VideoChooserBlock
from wagtailvideos.models import AbstractVideo, AbstractVideoTranscode from wagtailvideos.models import AbstractVideo, AbstractVideoTranscode, AbstractTrack
class CustomVideoModel(AbstractVideo): class CustomVideoModel(AbstractVideo):
...@@ -30,6 +30,9 @@ class CustomVideoTranscode(AbstractVideoTranscode): ...@@ -30,6 +30,9 @@ class CustomVideoTranscode(AbstractVideoTranscode):
) )
class CustomTrack(AbstractTrack):
video = models.ForeignKey(CustomVideoModel, related_name='tracks', on_delete=models.CASCADE)
class TestPage(Page): class TestPage(Page):
video_field = models.ForeignKey( video_field = models.ForeignKey(
'wagtailvideos.Video', related_name='+', null=True, blank=True, on_delete=models.SET_NULL) 'wagtailvideos.Video', related_name='+', null=True, blank=True, on_delete=models.SET_NULL)
......
...@@ -23,4 +23,6 @@ class WagtailVideosApp(AppConfig): ...@@ -23,4 +23,6 @@ class WagtailVideosApp(AppConfig):
verbose_name = 'Wagtail Videos' verbose_name = 'Wagtail Videos'
def ready(self): def ready(self):
from wagtailvideos.signals import register_signal_handlers
register_signal_handlers()
register(ffmpeg_check) register(ffmpeg_check)
This diff is collapsed.
...@@ -6,17 +6,14 @@ import shutil ...@@ -6,17 +6,14 @@ import shutil
import subprocess import subprocess
import tempfile import tempfile
import threading import threading
from contextlib import contextmanager
from distutils.version import LooseVersion from distutils.version import LooseVersion
import bcp47
import wagtail import wagtail
from django.conf import settings from django.conf import settings
from django.core.exceptions import SuspiciousFileOperation from django.core.exceptions import SuspiciousFileOperation
from django.core.files.base import ContentFile from django.core.files.base import ContentFile
from django.core.files.temp import NamedTemporaryFile
from django.db import models from django.db import models
from django.db.models.signals import post_save, pre_delete
from django.dispatch.dispatcher import receiver
from django.forms.utils import flatatt from django.forms.utils import flatatt
from django.urls import reverse from django.urls import reverse
from django.utils.safestring import mark_safe from django.utils.safestring import mark_safe
...@@ -27,8 +24,6 @@ from wagtail.core.models import CollectionMember ...@@ -27,8 +24,6 @@ from wagtail.core.models import CollectionMember
from wagtail.search import index from wagtail.search import index
from wagtail.search.queryset import SearchableQuerySetMixin from wagtail.search.queryset import SearchableQuerySetMixin
from wagtailvideos import ffmpeg
if LooseVersion(wagtail.__version__) >= LooseVersion('2.7'): if LooseVersion(wagtail.__version__) >= LooseVersion('2.7'):
from wagtail.admin.models import get_object_usage from wagtail.admin.models import get_object_usage
else: else:
...@@ -180,12 +175,9 @@ class AbstractVideo(CollectionMember, index.Indexed, models.Model): ...@@ -180,12 +175,9 @@ class AbstractVideo(CollectionMember, index.Indexed, models.Model):
def get_transcode_model(cls): def get_transcode_model(cls):
return cls.transcodes.rel.related_model return cls.transcodes.rel.related_model
def get_transcode(self, media_format): @classmethod
Transcode = self.get_transcode_model() def get_track_model(cls):
try: return cls.tracks.rel.related_model
return self.transcodes.get(media_format=media_format)
except Transcode.DoesNotExist:
return self.do_transcode(media_format)
def video_tag(self, attrs=None): def video_tag(self, attrs=None):
if attrs is None: if attrs is None:
...@@ -212,7 +204,7 @@ class AbstractVideo(CollectionMember, index.Indexed, models.Model): ...@@ -212,7 +204,7 @@ class AbstractVideo(CollectionMember, index.Indexed, models.Model):
transcode, created = self.transcodes.get_or_create( transcode, created = self.transcodes.get_or_create(
media_format=media_format, media_format=media_format,
) )
if transcode.processing is False: if created or transcode.processing is False:
transcode.processing = True transcode.processing = True
transcode.error_messages = '' transcode.error_messages = ''
transcode.quality = quality transcode.quality = quality
...@@ -292,61 +284,6 @@ class TranscodingThread(threading.Thread): ...@@ -292,61 +284,6 @@ class TranscodingThread(threading.Thread):
shutil.rmtree(output_dir, ignore_errors=True) shutil.rmtree(output_dir, ignore_errors=True)
@contextmanager
def get_local_file(file):
"""
Get a local version of the file, downloading it from the remote storage if
required. The returned value should be used as a context manager to
ensure any temporary files are cleaned up afterwards.
"""
try:
with open(file.path):
yield file.path
except NotImplementedError:
_, ext = os.path.splitext(file.name)
with NamedTemporaryFile(prefix='wagtailvideo-', suffix=ext) as tmp:
try:
file.open('rb')
for chunk in file.chunks():
tmp.write(chunk)
finally:
file.close()
tmp.flush()
yield tmp.name
# Delete files when model is deleted
@receiver(pre_delete, sender=Video)
def video_delete(sender, instance, **kwargs):
instance.thumbnail.delete(False)
instance.file.delete(False)
# Fields that need the actual video file to create
@receiver(post_save, sender=Video)
def video_saved(sender, instance, **kwargs):
if not ffmpeg.installed():
return
if hasattr(instance, '_from_signal'):
return
has_changed = instance._initial_file is not instance.file
filled_out = instance.thumbnail is not None and instance.duration is not None
if has_changed or not filled_out:
with get_local_file(instance.file) as file_path:
if has_changed or instance.thumbnail is None:
instance.thumbnail = ffmpeg.get_thumbnail(file_path)
if has_changed or instance.duration is None:
instance.duration = ffmpeg.get_duration(file_path)
instance.file_size = instance.file.size
instance._from_signal = True
instance.save()
del instance._from_signal
class AbstractVideoTranscode(models.Model): class AbstractVideoTranscode(models.Model):
media_format = EnumChoiceField(MediaFormats) media_format = EnumChoiceField(MediaFormats)
quality = EnumChoiceField(VideoQuality, default=VideoQuality.default) quality = EnumChoiceField(VideoQuality, default=VideoQuality.default)
...@@ -377,7 +314,41 @@ class VideoTranscode(AbstractVideoTranscode): ...@@ -377,7 +314,41 @@ class VideoTranscode(AbstractVideoTranscode):
) )
# Delete files when model is deleted class AbstractTrack(models.Model):
@receiver(pre_delete, sender=VideoTranscode) # TODO move to TextChoices once djangp < 2 is dropped
def transcode_delete(sender, instance, **kwargs): track_kinds = [
instance.file.delete(False) ('subtitles', 'Subtitles'),
('captions', 'Captions'),
('descriptions', 'Descriptions'),
('chapters', 'Chapters'),
('metadata', 'Metadata'),
]
file = models.FileField(
verbose_name=_('file'),
upload_to=get_upload_to
)
kind = models.CharField(max_length=50, choices=track_kinds, default=track_kinds[0][0])
label = models.CharField(
max_length=255, blank=True,
help_text='A user-readable title of the text track.')
language = models.CharField(
max_length=50,
choices=[(k, v) for k, v in bcp47.languages.items()],
default='en', blank=True, help_text='Required if type is "Subtitle"', unique=True)
@property
def url(self):
return self.file.url
def get_upload_to(self, filename):
folder_name = 'video_tracks'
filename = self.file.field.storage.get_valid_name(filename)
return os.path.join(folder_name, filename)
class Meta:
abstract = True
class Track(AbstractTrack):
video = models.ForeignKey(Video, related_name='tracks', on_delete=models.CASCADE)
import os
from contextlib import contextmanager
from django.core.files.temp import NamedTemporaryFile
from django.db import transaction
from django.db.models.signals import post_delete, post_save
from wagtailvideos import ffmpeg, get_video_model
@contextmanager
def get_local_file(file):
"""
Get a local version of the file, downloading it from the remote storage if
required. The returned value should be used as a context manager to
ensure any temporary files are cleaned up afterwards.
"""
try:
with open(file.path):
yield file.path
except NotImplementedError:
_, ext = os.path.splitext(file.name)
with NamedTemporaryFile(prefix='wagtailvideo-', suffix=ext) as tmp:
try:
file.open('rb')
for chunk in file.chunks():
tmp.write(chunk)
finally:
file.close()
tmp.flush()
yield tmp.name
def post_delete_file_cleanup(instance, **kwargs):
# Pass false so FileField doesn't save the model.
transaction.on_commit(lambda: instance.file.delete(False))
if hasattr(instance, 'thumbnail'):
# Delete the thumbnail for videos too
transaction.on_commit(lambda: instance.thumbnail.delete(False))
# Fields that need the actual video file to create using ffmpeg
def video_post_save(instance, **kwargs):
if not ffmpeg.installed():
return
if hasattr(instance, '_from_signal'):
# Sender was us, don't run post save
return
has_changed = instance._initial_file is not instance.file
filled_out = instance.thumbnail is not None and instance.duration is not None
if has_changed or not filled_out:
with get_local_file(instance.file) as file_path:
if has_changed or instance.thumbnail is None:
instance.thumbnail = ffmpeg.get_thumbnail(file_path)
if has_changed or instance.duration is None:
instance.duration = ffmpeg.get_duration(file_path)
instance.file_size = instance.file.size
instance._from_signal = True
instance.save()
del instance._from_signal
def register_signal_handlers():
Video = get_video_model()
VideoTranscode = Video.get_transcode_model()
VideoTrack = Video.get_track_model()
post_save.connect(video_post_save, sender=Video)
post_delete.connect(post_delete_file_cleanup, sender=Video)
post_delete.connect(post_delete_file_cleanup, sender=VideoTranscode)
post_delete.connect(post_delete_file_cleanup, sender=VideoTrack)
Markdown is supported
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment