diff --git a/larigira/audioform_podcast.py b/larigira/audioform_podcast.py new file mode 100644 index 0000000..1b364db --- /dev/null +++ b/larigira/audioform_podcast.py @@ -0,0 +1,63 @@ +from flask_wtf import Form +from wtforms import (BooleanField, IntegerField, SelectField, StringField, + SubmitField, validators) +from wtforms.fields.html5 import URLField + + +class AudioForm(Form): + nick = StringField( + "Audio nick", + validators=[validators.required()], + description="A simple name to recognize this audio", + ) + url = URLField( + "URL", + validators=[validators.required()], + description="URL of the podcast; it must be valid xml", + ) + + # TODO: group by filters/sort/select + min_len = StringField( + "Accetta solo audio lunghi almeno:", + description="Leaving this empty will disable this filter", + ) + max_len = StringField( + "Accetta solo audio lunghi al massimo:", + description="Leaving this empty will disable this filter", + ) + sort_by = SelectField( + "Sort episodes", + choices=[ + ("none", "Don't sort"), + ("random", "Random"), + ("duration", "Duration"), + ("date", "date"), + ], + ) + start = IntegerField( + "Play from episode number", + description="Episodes count from 0; 0 is a sane default", + ) + reverse = BooleanField("Reverse sort (descending)") + submit = SubmitField("Submit") + + def populate_from_audiospec(self, audiospec): + for key in ("nick", "url", "sort_by", "reverse", "min_len", "max_len"): + if key in audiospec: + getattr(self, key).data = audiospec[key] + self.start.data = int(audiospec.get("start", 0)) + + +def audio_receive(form): + d = {"kind": "podcast"} + for key in ( + "nick", + "url", + "sort_by", + "reverse", + "min_len", + "max_len", + "start", + ): + d[key] = getattr(form, key).data + return d diff --git a/larigira/audiogen_podcast.py b/larigira/audiogen_podcast.py new file mode 100644 index 0000000..b18d738 --- /dev/null +++ b/larigira/audiogen_podcast.py @@ -0,0 +1,184 @@ +import datetime +import logging +import os +import random +import sys +from subprocess import CalledProcessError, check_output + +import requests + +from larigira.fsutils import download_http +from lxml import html +from pytimeparse.timeparse import timeparse + + +def delta_humanreadable(tdelta): + if tdelta is None: + return "" + days = tdelta.days + hours = (tdelta - datetime.timedelta(days=days)).seconds // 3600 + if days: + return "{}d{}h".format(days, hours) + return "{}h".format(hours) + + +def get_duration(url): + try: + lineout = check_output( + [ + "ffprobe", + "-v", + "error", + "-show_entries", + "format=duration", + "-i", + url, + ] + ).split(b"\n") + except CalledProcessError as exc: + raise ValueError("error probing `%s`" % url) from exc + duration = next(l for l in lineout if l.startswith(b"duration=")) + value = duration.split(b"=")[1] + return int(float(value)) + + +class Audio(object): + def __init__(self, url, duration=None, date=None): + self.url = url + if duration is None: + duration = get_duration(url.encode("utf-8")) + self.duration = duration + self.date = date + self.end_date = datetime.datetime( + 9999, 12, 31, tzinfo=datetime.timezone.utc + ) + + def __str__(self): + return self.url + + def __repr__(self): + return "