Compare commits

..

No commits in common. "master" and "docs" have entirely different histories.
master ... docs

29 changed files with 369 additions and 1077 deletions

View file

@ -1,13 +0,0 @@
## Is it a bug or a feature request?
## Describe what happens
### what should happen, instead?
### describe all the steps we can do to reproduce the bug
### provide debug logs relevant to the bug
## Which version of larigira are you using? From git?
### Which OS do you use? which version?

View file

@ -40,10 +40,6 @@ action
For example, ``{ 'kind': 'randomdir', 'paths': ['/my/dir', '/other/path'] }`` For example, ``{ 'kind': 'randomdir', 'paths': ['/my/dir', '/other/path'] }``
will pick a random file from one of the two paths. will pick a random file from one of the two paths.
Its main atribute is ``kind``. The kind essentialy specifies the function that will be run among a
predefined set of :doc:`audiogenerators` . Every other attribute is an argument to the specified
audiogenerator.
event event
An event is an alarm plus a list of actions. At given times, do those things An event is an alarm plus a list of actions. At given times, do those things

View file

@ -1,22 +0,0 @@
larigira\.filters package
=========================
Submodules
----------
larigira\.filters\.basic module
-------------------------------
.. automodule:: larigira.filters.basic
:members:
:undoc-members:
:show-inheritance:
Module contents
---------------
.. automodule:: larigira.filters
:members:
:undoc-members:
:show-inheritance:

View file

@ -1,15 +1,12 @@
Audiogenerators Audiogenerators
=============== ===============
mpd mpdrandom
--------- ---------
picks ``howmany`` song randomly from your mpd library. It follows this picks ``howmany`` song randomly from your mpd library. It follows this
strategy: it picks ``howmany`` artists from your MPD library, then picks a random song for each one strategy: it picks ``howmany`` artists from your MPD library, then picks a random song for each one
if you specify a ``prefix``, then only files inside the ``prefix`` directory
will be picked.
randomdir randomdir
---------- ----------
@ -36,48 +33,6 @@ mostrecent
It is similar to randomdir, but instead of picking randomly, picks the most It is similar to randomdir, but instead of picking randomly, picks the most
recent file (according to the ctime). recent file (according to the ctime).
podcast
------------
This is probably the most powerful generator that comes included with
``larigira``. To use this generator, you would need to have a valid podcast
URL. Beware, here the world *podcast* refer to its very specific meaning of
an xml-based format which resembles a RSS feed but has more media-specific
entities. See `this specification
<https://github.com/simplepie/simplepie-ng/wiki/Spec:-iTunes-Podcast-RSS>`_ for
more technical details.
So, if you have a valid podcast URL, larigira can look at it, extract audios,
download and play the most recent one. Here are some typical usecases for this:
* You want to play replica based on what you host on your radio's website.
* You want to play some audio from some other radio (or other kind of podcast
source)
The podcast form has many many options, but I promise you that 90% of the cases
are easily solved using ONLY the first option: enter the URL of the podcast
and... it works!
So, what are all the other options for? Well, to cover some other use cases.
For example, let's say that at night you want to play a *random* show (not the
last one, which is the default) that happened on your radio. Then you can
change the "sort by" to be "random". Easy, right?
Another typical usecase is selecting an audio that has a duration which "fits"
with the schedule of your radio: not too long and not too short. You can do
that with the "min len" and "max len" fields. For example, setting a `min_len`
of `30min` and `max_len` of `1h15m` you can avoid picking flash news (too
short) and very long shows.
You can do many other things with its options, but I left those to your
immagination. Let's just clarify the workflow:
* the podcast URL is fetched and audio information is retrieved
* filter: audios are filtered by min/max length
* sort: audios are sorted according to `sort_by` and `reverse`
* select: the n-th episode is fetched, according to `start` field
script script
-------- --------

View file

@ -14,10 +14,9 @@
# serve to show the default. # serve to show the default.
from __future__ import print_function from __future__ import print_function
import sys
import os import os
import subprocess import subprocess
import sys
from sphinx.util.console import red from sphinx.util.console import red
@ -35,35 +34,35 @@ from sphinx.util.console import red
# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom # extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
# ones. # ones.
extensions = [ extensions = [
"sphinx.ext.autodoc", 'sphinx.ext.autodoc',
"sphinx.ext.coverage", 'sphinx.ext.coverage',
"sphinx.ext.viewcode", 'sphinx.ext.viewcode',
] ]
# Add any paths that contain templates here, relative to this directory. # Add any paths that contain templates here, relative to this directory.
templates_path = ["_templates"] templates_path = ['_templates']
# The suffix of source filenames. # The suffix of source filenames.
source_suffix = ".rst" source_suffix = '.rst'
# The encoding of source files. # The encoding of source files.
#source_encoding = 'utf-8-sig' #source_encoding = 'utf-8-sig'
# The master toctree document. # The master toctree document.
master_doc = "index" master_doc = 'index'
# General information about the project. # General information about the project.
project = "larigira" project = 'larigira'
copyright = "2015-2017, boyska" copyright = '2015-2017, boyska'
# The version info for the project you're documenting, acts as replacement for # The version info for the project you're documenting, acts as replacement for
# |version| and |release|, also used in various other places throughout the # |version| and |release|, also used in various other places throughout the
# built documents. # built documents.
# #
# The short X.Y version. # The short X.Y version.
version = "1.3" version = '1.3'
# The full version, including alpha/beta/rc tags. # The full version, including alpha/beta/rc tags.
release = "1.3.3" release = '1.3.1'
# The language for content autogenerated by Sphinx. Refer to documentation # The language for content autogenerated by Sphinx. Refer to documentation
# for a list of supported languages. # for a list of supported languages.
@ -95,7 +94,7 @@ exclude_patterns = []
#show_authors = False #show_authors = False
# The name of the Pygments (syntax highlighting) style to use. # The name of the Pygments (syntax highlighting) style to use.
pygments_style = "sphinx" pygments_style = 'sphinx'
# A list of ignored prefixes for module index sorting. # A list of ignored prefixes for module index sorting.
#modindex_common_prefix = [] #modindex_common_prefix = []
@ -108,7 +107,7 @@ pygments_style = "sphinx"
# The theme to use for HTML and HTML Help pages. See the documentation for # The theme to use for HTML and HTML Help pages. See the documentation for
# a list of builtin themes. # a list of builtin themes.
html_theme = "default" html_theme = 'default'
# Theme options are theme-specific and customize the look and feel of a theme # Theme options are theme-specific and customize the look and feel of a theme
# further. For a list of options available for each theme, see the # further. For a list of options available for each theme, see the
@ -137,7 +136,7 @@ html_theme = "default"
# Add any paths that contain custom static files (such as style sheets) here, # Add any paths that contain custom static files (such as style sheets) here,
# relative to this directory. They are copied after the builtin static files, # relative to this directory. They are copied after the builtin static files,
# so a file named "default.css" will overwrite the builtin "default.css". # so a file named "default.css" will overwrite the builtin "default.css".
html_static_path = ["_static"] html_static_path = ['_static']
# Add any extra paths that contain custom files (such as robots.txt or # Add any extra paths that contain custom files (such as robots.txt or
# .htaccess) here, relative to this directory. These files are copied # .htaccess) here, relative to this directory. These files are copied
@ -186,7 +185,7 @@ html_static_path = ["_static"]
#html_file_suffix = None #html_file_suffix = None
# Output file base name for HTML help builder. # Output file base name for HTML help builder.
htmlhelp_basename = "larigiradoc" htmlhelp_basename = 'larigiradoc'
# -- Options for LaTeX output --------------------------------------------- # -- Options for LaTeX output ---------------------------------------------
@ -194,8 +193,10 @@ htmlhelp_basename = "larigiradoc"
latex_elements = { latex_elements = {
# The paper size ('letterpaper' or 'a4paper'). # The paper size ('letterpaper' or 'a4paper').
#'papersize': 'letterpaper', #'papersize': 'letterpaper',
# The font size ('10pt', '11pt' or '12pt'). # The font size ('10pt', '11pt' or '12pt').
#'pointsize': '10pt', #'pointsize': '10pt',
# Additional stuff for the LaTeX preamble. # Additional stuff for the LaTeX preamble.
#'preamble': '', #'preamble': '',
} }
@ -204,7 +205,8 @@ latex_elements = {
# (source start file, target name, title, # (source start file, target name, title,
# author, documentclass [howto, manual, or own class]). # author, documentclass [howto, manual, or own class]).
latex_documents = [ latex_documents = [
("index", "larigira.tex", "larigira Documentation", "boyska", "manual") ('index', 'larigira.tex', 'larigira Documentation',
'boyska', 'manual'),
] ]
# The name of an image file (relative to this directory) to place at the top of # The name of an image file (relative to this directory) to place at the top of
@ -232,7 +234,10 @@ latex_documents = [
# One entry per manual page. List of tuples # One entry per manual page. List of tuples
# (source start file, name, description, authors, manual section). # (source start file, name, description, authors, manual section).
man_pages = [("index", "larigira", "larigira Documentation", ["boyska"], 1)] man_pages = [
('index', 'larigira', 'larigira Documentation',
['boyska'], 1)
]
# If true, show URL addresses after external links. # If true, show URL addresses after external links.
#man_show_urls = False #man_show_urls = False
@ -244,15 +249,9 @@ man_pages = [("index", "larigira", "larigira Documentation", ["boyska"], 1)]
# (source start file, target name, title, author, # (source start file, target name, title, author,
# dir menu entry, description, category) # dir menu entry, description, category)
texinfo_documents = [ texinfo_documents = [
( ('index', 'larigira', 'larigira Documentation',
"index", 'boyska', 'larigira', 'One line description of project.',
"larigira", 'Miscellaneous'),
"larigira Documentation",
"boyska",
"larigira",
"One line description of project.",
"Miscellaneous",
)
] ]
# Documents to append as an appendix to all manuals. # Documents to append as an appendix to all manuals.
@ -270,32 +269,32 @@ texinfo_documents = [
def run_apidoc(_): def run_apidoc(_):
cur_dir = os.path.abspath(os.path.dirname(__file__)) cur_dir = os.path.abspath(os.path.dirname(__file__))
proj_dir = os.path.abspath(os.path.join(cur_dir, "..", "..")) proj_dir = os.path.abspath(os.path.join(cur_dir, '..', '..'))
modules = ["larigira"] modules = ['larigira']
exclude_files = [ exclude_files = [os.path.abspath(os.path.join(proj_dir, excl))
os.path.abspath(os.path.join(proj_dir, excl)) for excl in ('larigira/rpc.py', 'larigira/dbadmin/')]
for excl in ("larigira/rpc.py", "larigira/dbadmin/") output_path = os.path.join(cur_dir, 'api')
] cmd_path = 'sphinx-apidoc'
output_path = os.path.join(cur_dir, "api") if hasattr(sys, 'real_prefix'): # Are we in a virtualenv?
cmd_path = "sphinx-apidoc"
if hasattr(sys, "real_prefix"): # Are we in a virtualenv?
# assemble the path manually # assemble the path manually
cmd_path = os.path.abspath( cmd_path = os.path.abspath(os.path.join(sys.prefix,
os.path.join(sys.prefix, "bin", "sphinx-apidoc") 'bin',
) 'sphinx-apidoc'))
if not os.path.exists(cmd_path): if not os.path.exists(cmd_path):
print(red("No apidoc available!"), file=sys.stderr) print(red("No apidoc available!"), file=sys.stderr)
return return
for module in modules: for module in modules:
try: try:
subprocess.check_call( subprocess.check_call([cmd_path,
[cmd_path, "--force", "-o", output_path, module] '--force',
+ exclude_files, '-o', output_path,
cwd=proj_dir, module
] + exclude_files,
cwd=proj_dir
) )
except subprocess.CalledProcessError: except subprocess.CalledProcessError:
print(red("APIdoc failed for module {}".format(module))) print(red("APIdoc failed for module {}".format(module)))
def setup(app): def setup(app):
app.connect("builder-inited", run_apidoc) app.connect('builder-inited', run_apidoc)

View file

@ -12,13 +12,11 @@ Contents:
:maxdepth: 2 :maxdepth: 2
about about
quickstart
install install
timegenerators timegenerators
audiogenerators audiogenerators
eventfilters eventfilters
audiogenerators-write audiogenerators-write
troubleshooting
debug debug
api/modules api/modules

View file

@ -6,7 +6,7 @@ using ``pip install larigira``. Or you can ``git clone
https://git.lattuga.net/boyska/larigira.git`` and run ``python setup.py install``. https://git.lattuga.net/boyska/larigira.git`` and run ``python setup.py install``.
As always, the usage of a virtualenv is recommended. As always, the usage of a virtualenv is recommended.
Python greater or equal than 3.4 is supported. The only supported python version is 3.4.
Configuration Configuration
--------------- ---------------
@ -22,7 +22,7 @@ inside ``~/.mpdconf``, add the following line::
bind_to_address "~/.mpd/socket" bind_to_address "~/.mpd/socket"
For larigira, you need to set the ``MPD_HOST`` environment variable to For larigira, you need to set the ``MPD_HOST`` environment variable to
``$HOME/.mpd/socket``. If you don't do this, you'll find many lines like these in the logs:: ``$HOME/.mpd/socket``. If you don't do this, you'll find many::
15:37:10|ERROR[Player:93] Cannot insert song file:///tmp/larigira.1002/audiogen-randomdir-8eoklcee.mp3 15:37:10|ERROR[Player:93] Cannot insert song file:///tmp/larigira.1002/audiogen-randomdir-8eoklcee.mp3
Traceback (most recent call last): Traceback (most recent call last):
@ -68,16 +68,12 @@ MPD_PORT
If you are not using a socket, but a TCP address (which is *not* suggested), this is how you can specify the If you are not using a socket, but a TCP address (which is *not* suggested), this is how you can specify the
port. port.
DEBUG DEBUG
you can set it to ``true`` or ``false``. Defaults to ``false``. Will enable extremely verbose output. you can set it to ``true`` or ``false``. Defaults to ``false``.
TMPDIR TMPDIR
The base for larigira tmpdir. Please note that larigira will create its own directory inside this The base for larigira tmpdir. Please note that larigira will create its own directory inside this
temporary directory. This defaults to the system-wide ``$TMPDIR``, or to ``/tmp/`` if not ``TMPDIR`` is temporary directory. This defaults to the system-wide ``$TMPDIR``, or to ``/tmp/`` if not ``TMPDIR`` is
not set. Choose it wisely, keeping in mind that in this directory a lot of cache files will be stored, and not set. Choose it wisely, keeping in mind that in this directory a lot of cache files will be stored, and
could therefore require hundreds of MB. could therefore require hundreds of MB.
UMASK
Umask affects created files permissions. This is important if you want to pass files to a MPD instance
that is running with a different users. There is no default umask, so that you can apply umask via your
standard system tools.
Events Events
^^^^^^^^^ ^^^^^^^^^
@ -85,25 +81,13 @@ Events
CONTINOUS_AUDIOSPEC CONTINOUS_AUDIOSPEC
when the playlist is too short, larigira picks something new. How? this is controlled by this variable. when the playlist is too short, larigira picks something new. How? this is controlled by this variable.
This variable should be set to the JSON representation of an audiospec describing how to generate new This variable should be set to the JSON representation of an audiospec describing how to generate new
audios. The default is ``{"kind": "mpd", "howmany": 1}``, which picks a random song from MPD library. You could, for example, change it to: audios. The default is ``{"kind": "mpd", "howmany": 1}``. You could, for example, change it to
``{ "kind": "randomdir", "paths": ["/var/music"], "howmany": 10}``
- ``{ "kind": "randomdir", "paths": ["/var/music"], "howmany": 5}``
to pick files from a specified directory, ignoring MPD library completely. Here, ``howmany`` is set to 5 for performance sake
- ``{"kind": "mpd", "howmany": 10, "prefix": "background"}``
if you want to use the MPD library, but only use one of its subdirectories.
Since using ``mpd`` with a ``prefix`` can be slow, so this is setting ``howmany`` to 10 to
gain some performance
Yes, there's a typo in the name, but I'll keep it like this for compatibility
EVENT_FILTERS EVENT_FILTERS
See :doc:`eventfilters` See :doc:`eventfilters`
LOG_CONFIG LOG_CONFIG
Path to an INI-formatted file to configure logging. See `python logging documentation Path to an INI-formatted file to configure logging. See `python logging documentation
<https://docs.python.org/2/library/logging.config.html#logging-config-dictschema>`_ <https://docs.python.org/2/library/logging.config.html#logging-config-dictschema>`_
MPD_ENFORCE_ALWAYS_PLAYING
If this is set to 1, larigira will make sure that MPD is always playing. This means that you can't stop
mpd, not even manually running ``mpd stop``. That's probably useful for radios in which mpd is meant to be
run unattended. Default: 0
Internals Internals
^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^
@ -139,10 +123,6 @@ MPD_WAIT_START_RETRYSECS
Web interface Web interface
^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^
HTTP_ADDRESS
The address that the HTTP interface will listen to; defaults to `0.0.0.0`
HTTP_PORT
The port that the HTTP interface will listen to; defaults to `5000`
FILE_PATH_SUGGESTION FILE_PATH_SUGGESTION
A list of paths. Those paths will be scanned for suggestions in audiogenerator forms. A list of paths. Those paths will be scanned for suggestions in audiogenerator forms.
UI_CALENDAR_FREQUENCY_THRESHOLD UI_CALENDAR_FREQUENCY_THRESHOLD
@ -165,9 +145,3 @@ LARIGIRA_UI_CALENDAR_DATE_FMT
The format to show in ``/db/calendar`` page. The format is specified `here <http://babel.pocoo.org/en/latest/dates.html>`_. Default is ``medium``. The format to show in ``/db/calendar`` page. The format is specified `here <http://babel.pocoo.org/en/latest/dates.html>`_. Default is ``medium``.
As an example, ``eee dd LLL`` will show ``Sun 10 Mar`` for english, and ``dom 10 mar`` for italian. As an example, ``eee dd LLL`` will show ``Sun 10 Mar`` for english, and ``dom 10 mar`` for italian.
Debug and development
^^^^^^^^^^^^^^^^^^^^^^^
REMOVE_UNUSED_FILES
By default, larigira removes the file it generates, as soon as they are no longer in MPD playlist. There is no good reason to change this behavior, unless you need to debug what's going on. For example, if you want to inspect files. Set this to `false` keep everything.

View file

@ -1,152 +0,0 @@
Quick start: install larigira on Debian buster
==============================================
This guides have this assumptions or conventions:
* you have a Debian buster installation
- actually 99% of this should work in any distro with recent-enough python3 and systemd
- if you don't like systemd, you are free to use any other service manager; larigira integrates nicely with systemd, but has no requirement at all on it.
* you have a non-root main user, which we'll call ``radio``
* all commands are meant to be run as root. Use ``sudo -i`` if you don't have root password
Install
-----------
Let's start!::
apt-get install python3 python3-dev build-essential virtualenv mpd mpc libxml2-dev libxslt1-dev zlib1g-dev
virtualenv -p /usr/bin/python3 /opt/larigira/
/opt/larigira/bin/pip3 install --no-binary :all: larigira
touch /etc/default/larigira
mkdir -p /home/radio/.mpd/ /etc/larigira/ /var/log/larigira/
chown radio. /home/radio/.mpd/
chown radio:adm /var/log/larigira/
touch /etc/systemd/system/larigira.service
Edit ``/etc/systemd/system/larigira.service`` and put this content::
[Unit]
Description=Radio Automation
After=mpd.service
[Service]
Type=notify
NotifyAccess=all
EnvironmentFile=/etc/default/larigira
User=radio
ExecStart=/opt/larigira/bin/larigira
Restart=always
[Install]
WantedBy=multi-user.target
Now let's edit ``/etc/mpd.conf``::
music_directory "/home/radio/Music/"
playlist_directory "/home/radio/.mpd/playlists"
db_file "/home/radio/.mpd/tag_cache"
log_file "syslog"
pid_file "/home/radio/.mpd/pid"
state_file "/home/radio/.mpd/state"
sticker_file "/home/radio/.mpd/sticker.sql"
user "radio"
bind_to_address "/home/radio/.mpd/socket"
bind_to_address "127.0.0.1"
port "6600"
log_level "default"
replaygain "track"
replaygain_limit "yes"
volume_normalization "yes"
max_connections "30"
Now let's edit larigira settings, editing the file ``/etc/default/larigira``::
MPD_HOST=/home/radio/.mpd/socket
LARIGIRA_DEBUG=false
LARIGIRA_LOG_CONFIG=/etc/larigira/logging.ini
LARIGIRA_EVENT_FILTERS='["percentwait"]'
LARIGIRA_EF_MAXWAIT_PERC=400
LARIGIRA_MPD_ENFORCE_ALWAYS_PLAYING=1
LARIGIRA_SECRET_KEY="changeme with a random, secret string of any length"
Let's include logging configuration, editing ``/etc/larigira/logging.ini``::
[loggers]
keys=root
[formatters]
keys=brief,ext,debug
[handlers]
keys=syslog,own,owndebug,ownerr
[logger_root]
handlers=syslog,own,owndebug,ownerr
level=DEBUG
[handler_syslog]
class=handlers.SysLogHandler
level=INFO
args=('/dev/log', handlers.SysLogHandler.LOG_USER)
formatter=brief
[handler_own]
class=handlers.WatchedFileHandler
level=INFO
args=('/var/log/larigira/larigira.log',)
formatter=ext
[handler_owndebug]
class=handlers.WatchedFileHandler
level=DEBUG
args=('/var/log/larigira/larigira.debug',)
formatter=debug
[handler_ownerr]
class=handlers.WatchedFileHandler
level=ERROR
args=('/var/log/larigira/larigira.err',)
formatter=ext
[formatter_ext]
format=%(asctime)s|%(levelname)s[%(name)s] %(message)s
[formatter_debug]
format=%(asctime)s|%(levelname)s[%(name)s:%(lineno)d] %(message)s
[formatter_brief]
format=%(levelname)s:%(message)s
For hygiene's sake, let's configure rotation for this log, editing ``/etc/logrotate.d/larigira``::
/var/log/larigira/*.err
/var/log/larigira/*.log {
daily
missingok
rotate 14
compress
notifempty
copytruncate
create 600
}
/var/log/larigira/*.debug {
daily
rotate 2
missingok
compress
notifempty
copytruncate
create 600
}
Restart everything::
systemctl daemon-reload
systemctl restart mpd
systemctl restart larigira
systemctl enable larigira
systemctl enable mpd
Everything should work now!

View file

@ -1,16 +0,0 @@
Common problems
====================
I got AccessDenied in the logs
------------------------------
You are not using the UNIX socket to access MPD. See the configuration variable MPD_HOST. This is required by
MPD.
I got ``mpd.base.CommandError: [50@0] {} No such song``
-------------------------------------------------------
There is permissions issues going on. Probably you are running larigira and mpd with two different users, and
you haven't set up a common group for them.

View file

@ -1,63 +0,0 @@
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

View file

@ -3,7 +3,7 @@ import sys
import argparse import argparse
from .entrypoints_utils import get_one_entrypoint from .entrypoints_utils import get_one_entrypoint
import json import json
from logging import getLogger, basicConfig from logging import getLogger
log = getLogger("audiogen") log = getLogger("audiogen")
@ -15,7 +15,6 @@ def get_audiogenerator(kind):
def get_parser(): def get_parser():
parser = argparse.ArgumentParser(description="Generate audio and output paths") parser = argparse.ArgumentParser(description="Generate audio and output paths")
parser.add_argument("--log-level", choices=['DEBUG', 'INFO', 'WARNING', 'DEBUG'], default='WARNING')
parser.add_argument( parser.add_argument(
"audiospec", "audiospec",
metavar="AUDIOSPEC", metavar="AUDIOSPEC",
@ -50,7 +49,6 @@ def audiogenerate(spec):
def main(): def main():
"""Main function for the "larigira-audiogen" executable""" """Main function for the "larigira-audiogen" executable"""
args = get_parser().parse_args() args = get_parser().parse_args()
basicConfig(level=args.log_level)
spec = read_spec(args.audiospec[0]) spec = read_spec(args.audiospec[0])
errors = tuple(check_spec(spec)) errors = tuple(check_spec(spec))
if errors: if errors:

View file

@ -1,4 +1,31 @@
from larigira.fsutils import download_http import os
import logging
import posixpath
from tempfile import mkstemp
import urllib.request
from urllib.parse import urlparse
log = logging.getLogger(__name__)
def put(url, destdir=None, copy=False):
if url.split(":")[0] not in ("http", "https"):
log.warning("Not a valid URL: %s", url)
return None
ext = url.split(".")[-1]
if ext.lower() not in ("mp3", "ogg", "oga", "wma", "m4a"):
log.warning('Invalid format (%s) for "%s"', ext, url)
return None
if not copy:
return url
fname = posixpath.basename(urlparse(url).path)
# sanitize
fname = "".join(c for c in fname if c.isalnum() or c in list("._-")).rstrip()
tmp = mkstemp(suffix="." + ext, prefix="http-%s-" % fname, dir=destdir)
os.close(tmp[0])
log.info("downloading %s -> %s", url, tmp[1])
fname, headers = urllib.request.urlretrieve(url, tmp[1])
return "file://%s" % os.path.realpath(tmp[1])
def generate(spec): def generate(spec):
@ -8,10 +35,10 @@ def generate(spec):
Recognized argument is "paths" (list of static paths) Recognized argument is "paths" (list of static paths)
""" """
if "urls" not in spec: if "urls" not in spec:
raise ValueError("Malformed audiospec: missing 'urls'") raise ValueError("Malformed audiospec: missing 'paths'")
for url in spec["urls"]: for url in spec["urls"]:
ret = download_http(url, copy=True, prefix="http") ret = put(url, copy=True)
if ret is None: if ret is None:
continue continue
yield ret yield ret

View file

@ -1,43 +1,25 @@
import logging import logging
log = logging.getLogger("mpdrandom")
import random import random
from mpd import MPDClient from mpd import MPDClient
from .config import get_conf from .config import get_conf
log = logging.getLogger(__name__)
def generate_by_artist(spec): def generate_by_artist(spec):
"""Choose HOWMANY random artists, and for each one choose a random song.""" """choose HOWMANY random artists, and for each one choose a random song"""
spec.setdefault("howmany", 1) spec.setdefault("howmany", 1)
prefix = spec.get("prefix", "").rstrip("/")
log.info("generating") log.info("generating")
conf = get_conf() conf = get_conf()
c = MPDClient(use_unicode=True) c = MPDClient(use_unicode=True)
c.connect(conf["MPD_HOST"], conf["MPD_PORT"]) c.connect(conf["MPD_HOST"], conf["MPD_PORT"])
if prefix:
# TODO: listallinfo is discouraged.
# how else could we achieve the same result?
artists = list(
{r["artist"] for r in c.listallinfo(prefix) if "artist" in r}
)
else:
artists = c.list("artist") artists = c.list("artist")
log.debug("got %d artists", len(artists))
if not artists: if not artists:
raise ValueError("no artists in your mpd database") raise ValueError("no artists in your mpd database")
for _ in range(spec["howmany"]): for _ in range(spec["howmany"]):
artist = random.choice(artists) # pick one artist artist = random.choice(artists)
if type(artist) is not str: yield random.choice(c.find("artist", artist))["file"]
# different mpd library versions have different behavior
artist = artist['artist']
# pick one song from that artist
artist_songs = (res["file"] for res in c.find("artist", artist))
if prefix:
artist_songs = [
fname
for fname in artist_songs
if fname.startswith(prefix + "/")
]
yield random.choice(list(artist_songs))

View file

@ -1,216 +0,0 @@
import datetime
import logging
import os
import random
import sys
from subprocess import CalledProcessError, check_output
import dateutil.parser
import requests
from lxml import html
from pytimeparse.timeparse import timeparse
from larigira.fsutils import download_http
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
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 "<Audio {} ({} {})>".format(
self.url, self._duration, delta_humanreadable(self.age)
)
@property
def duration(self):
"""lazy-calculation"""
if self._duration is None:
try:
self._duration = get_duration(self.url.encode("utf-8"))
except:
logging.exception(
"Error while computing duration of %s; set it to 0",
self.url,
)
self._duration = 0
return self._duration
@property
def urls(self):
return [self.url]
@property
def age(self):
if self.date is None:
return None
now = datetime.datetime.utcnow().replace(tzinfo=datetime.timezone.utc)
return now - self.date
@property
def valid(self):
return self.end_date >= datetime.datetime.utcnow().replace(
tzinfo=datetime.timezone.utc
)
def get_tree(feed_url):
if feed_url.startswith("http:") or feed_url.startswith("https:"):
tree = html.fromstring(requests.get(feed_url).content)
else:
if not os.path.exists(feed_url):
raise ValueError("file not found: {}".format(feed_url))
tree = html.parse(open(feed_url))
return tree
def get_item_date(el):
el_date = el.find("pubdate")
if el_date is None:
return None
for time_format in ("%Y-%m-%dT%H:%M:%S%z", "%a, %d %b %Y %H:%M:%S %z"):
try:
return datetime.datetime.strptime(el_date.text, time_format)
except:
continue
return dateutil.parser.parse(el_date.text)
def get_audio_from_item(item):
encl = item.find("enclosure")
if encl is not None:
url = encl.get("url")
else:
return None
audio_args = {}
if item.find("duration") is not None:
duration_parts = item.findtext("duration").split(":")
total_seconds = 0
for i, num in enumerate(reversed(duration_parts)):
total_seconds += int(float(num)) * (60 ** i)
if total_seconds:
audio_args["duration"] = total_seconds
else:
contents = item.xpath("group/content")
if not contents:
contents = item.xpath("content")
for child in contents:
if child.get("url") == url and child.get("duration") is not None:
audio_args["duration"] = int(float(child.get("duration")))
break
return Audio(url, **audio_args)
def get_urls(tree):
items = tree.xpath("//item")
for i, it in enumerate(items):
try:
audio = get_audio_from_item(it)
except Exception:
logging.error("Could not parse item #%d, skipping", i)
continue
if audio is None:
continue
if audio.date is None:
try:
audio.date = get_item_date(it)
except Exception:
logging.warn("Could not find date for item #%d", i)
yield audio
def parse_duration(arg):
if arg.isdecimal():
secs = int(arg)
else:
secs = timeparse(arg)
if secs is None:
raise ValueError("%r is not a valid duration" % arg)
return secs
def generate(spec):
if "url" not in spec:
raise ValueError("Malformed audiospec: missing 'url'")
audios = list(get_urls(get_tree(spec["url"])))
if spec.get("min_len", False):
audios = [
a for a in audios if a.duration >= parse_duration(spec["min_len"])
]
if spec.get("max_len", False):
audios = [
a for a in audios if a.duration <= parse_duration(spec["max_len"])
]
# sort
sort_by = spec.get("sort_by", "none")
if sort_by == "random":
random.shuffle(audios)
elif sort_by == "date":
audios.sort(key=lambda x: x.age)
elif sort_by == "duration":
audios.sort(key=lambda x: x.duration)
if spec.get("reverse", False):
audios.reverse()
# slice
audios = audios[int(spec.get("start", 0)) :]
audios = audios[: int(spec.get("howmany", 1))]
# copy local
local_audios = [
download_http(a.url, copy=spec.get("copy", True), prefix="podcast")
for a in audios
]
return local_audios
# TODO: testing
# TODO: lxml should maybe be optional?
# TODO: ui
if __name__ == "__main__":
# less than proper testing
logging.basicConfig(level=logging.DEBUG)
for u in get_urls(get_tree(sys.argv[1])):
print(" -", repr(u))

View file

@ -16,7 +16,6 @@ def get_conf(prefix="LARIGIRA_"):
conf["CONTINOUS_AUDIOSPEC"] = dict(kind="mpd", howmany=1) conf["CONTINOUS_AUDIOSPEC"] = dict(kind="mpd", howmany=1)
conf["MPD_HOST"] = os.getenv("MPD_HOST", "localhost") conf["MPD_HOST"] = os.getenv("MPD_HOST", "localhost")
conf["MPD_PORT"] = int(os.getenv("MPD_PORT", "6600")) conf["MPD_PORT"] = int(os.getenv("MPD_PORT", "6600"))
conf["UMASK"] = None
conf["CACHING_TIME"] = 10 conf["CACHING_TIME"] = 10
conf["DB_URI"] = os.path.join(conf_dir, "db.json") conf["DB_URI"] = os.path.join(conf_dir, "db.json")
conf["SCRIPTS_PATH"] = os.path.join(conf_dir, "scripts") conf["SCRIPTS_PATH"] = os.path.join(conf_dir, "scripts")
@ -27,23 +26,15 @@ def get_conf(prefix="LARIGIRA_"):
conf["SECRET_KEY"] = "Please replace me!" conf["SECRET_KEY"] = "Please replace me!"
conf["MPD_WAIT_START"] = True conf["MPD_WAIT_START"] = True
conf["MPD_WAIT_START_RETRYSECS"] = 5 conf["MPD_WAIT_START_RETRYSECS"] = 5
conf["MPD_ENFORCE_ALWAYS_PLAYING"] = False
conf["CHECK_SECS"] = 20 # period for checking playlist length conf["CHECK_SECS"] = 20 # period for checking playlist length
conf["EVENT_TICK_SECS"] = 30 # period for scheduling events conf["EVENT_TICK_SECS"] = 30 # period for scheduling events
conf["DEBUG"] = False conf["DEBUG"] = False
conf[
"REMOVE_UNUSED_FILES"
] = True # please keep it to True unless you are into deep debugging!
conf["LOG_CONFIG"] = False conf["LOG_CONFIG"] = False
conf["TMPDIR"] = os.getenv("TMPDIR", "/tmp/") conf["TMPDIR"] = os.getenv("TMPDIR", "/tmp/")
conf["FILE_PATH_SUGGESTION"] = () # tuple of paths conf["FILE_PATH_SUGGESTION"] = () # tuple of paths
# UI_CALENDAR_FREQUENCY_THRESHOLD"] has been removed
# use UI_CALENDAR_OCCURRENCIES_THRESHOLD instead
conf["UI_CALENDAR_FREQUENCY_THRESHOLD"] = 4 * 60 * 60 # 4 hours conf["UI_CALENDAR_FREQUENCY_THRESHOLD"] = 4 * 60 * 60 # 4 hours
conf["UI_CALENDAR_OCCURRENCIES_THRESHOLD"] = 40
conf["UI_CALENDAR_DATE_FMT"] = "medium" conf["UI_CALENDAR_DATE_FMT"] = "medium"
conf["EVENT_FILTERS"] = [] conf["EVENT_FILTERS"] = []
conf["HTTP_ADDRESS"] = "0.0.0.0"
conf["HTTP_PORT"] = 5000 conf["HTTP_PORT"] = 5000
conf["HOME_URL"] = "/db/calendar" conf["HOME_URL"] = "/db/calendar"
conf.update(from_envvars(prefix=prefix)) conf.update(from_envvars(prefix=prefix))
@ -75,9 +66,7 @@ def from_envvars(prefix=None, envvars=None, as_json=True):
if not envvars: if not envvars:
envvars = { envvars = {
k: k[len(prefix) :] k: k[len(prefix) :] for k in os.environ.keys() if k.startswith(prefix)
for k in os.environ.keys()
if k.startswith(prefix)
} }
for env_name, name in envvars.items(): for env_name, name in envvars.items():

View file

@ -4,22 +4,30 @@ This module contains a flask blueprint for db administration stuff
Templates are self-contained in this directory Templates are self-contained in this directory
""" """
from __future__ import print_function from __future__ import print_function
import mimetypes
import os import os
from datetime import datetime, timedelta, time
from collections import defaultdict from collections import defaultdict
from datetime import datetime, time, timedelta import mimetypes
from flask import (Blueprint, Response, abort, current_app, flash, jsonify, from flask import (
redirect, render_template, request, url_for) current_app,
Blueprint,
Response,
render_template,
jsonify,
abort,
request,
redirect,
url_for,
flash,
)
from larigira import forms
from larigira.audiogen import get_audiogenerator
from larigira.config import get_conf
from larigira.entrypoints_utils import get_avail_entrypoints from larigira.entrypoints_utils import get_avail_entrypoints
from larigira.timegen import get_timegenerator, timegenerate from larigira.audiogen import get_audiogenerator
from larigira.timegen_every import FrequencyAlarm from larigira.timegen_every import FrequencyAlarm
from larigira.timegen import get_timegenerator, timegenerate
from larigira import forms
from larigira.config import get_conf
from .suggestions import get_suggestions from .suggestions import get_suggestions
db = Blueprint( db = Blueprint(
@ -31,13 +39,10 @@ db = Blueprint(
def request_wants_json(): def request_wants_json():
best = request.accept_mimetypes.best_match( best = request.accept_mimetypes.best_match(["application/json", "text/html"])
["application/json", "text/html"]
)
return ( return (
best == "application/json" best == "application/json"
and request.accept_mimetypes[best] and request.accept_mimetypes[best] > request.accept_mimetypes["text/html"]
> request.accept_mimetypes["text/html"]
) )
@ -62,31 +67,26 @@ def events_list():
def events_calendar(): def events_calendar():
model = current_app.larigira.controller.monitor.model model = current_app.larigira.controller.monitor.model
today = datetime.now().date() today = datetime.now().date()
max_days = 30 maxdays = 30
max_occurrences = get_conf()["UI_CALENDAR_OCCURRENCIES_THRESHOLD"]
# {date: {datetime: [(alarm1,actions1), (alarm2,actions2)]}} # {date: {datetime: [(alarm1,actions1), (alarm2,actions2)]}}
days = defaultdict(lambda: defaultdict(list)) days = defaultdict(lambda: defaultdict(list))
show_all = request.args.get("all", "0") == "1" freq_threshold = get_conf()["UI_CALENDAR_FREQUENCY_THRESHOLD"]
for alarm in model.get_all_alarms(): for alarm in model.get_all_alarms():
if (
freq_threshold
and alarm["kind"] == "frequency"
and FrequencyAlarm(alarm).interval < freq_threshold
):
continue
actions = tuple(model.get_actions_by_alarm(alarm)) actions = tuple(model.get_actions_by_alarm(alarm))
if not actions: if not actions:
continue continue
today_dt = datetime.fromtimestamp(int(today.strftime("%s"))) t = datetime.fromtimestamp(int(today.strftime("%s")))
max_dt = datetime.combine(today_dt + timedelta(days=max_days), time()) for t in timegenerate(alarm, now=t, howmany=maxdays):
occurrences = [] if t is None or t > datetime.combine(
for t in timegenerate( today + timedelta(days=maxdays), time()
alarm, now=today_dt, howmany=max_occurrences + 1
): ):
if t is None:
break break
if t > max_dt:
break
occurrences.append(t)
if not occurrences:
continue
if not show_all and len(occurrences) > max_occurrences:
continue
for t in occurrences:
days[t.date()][t].append((alarm, actions)) days[t.date()][t].append((alarm, actions))
# { weeknum: [day1, day2, day3] } # { weeknum: [day1, day2, day3] }
@ -94,9 +94,7 @@ def events_calendar():
for d in sorted(days.keys()): for d in sorted(days.keys()):
weeks[d.isocalendar()[:2]].append(d) weeks[d.isocalendar()[:2]].append(d)
return render_template( return render_template("calendar.html", days=days, weeks=weeks)
"calendar.html", days=days, weeks=weeks, show_all=show_all
)
@db.route("/add/time") @db.route("/add/time")
@ -123,15 +121,9 @@ def edit_time(alarmid):
data = receiver(form) data = receiver(form)
model.update_alarm(alarmid, data) model.update_alarm(alarmid, data)
model.reload() model.reload()
return redirect( return redirect(url_for("db.events_calendar", highlight="%d" % alarmid))
url_for("db.events_calendar", highlight="%d" % alarmid)
)
return render_template( return render_template(
"add_time_kind.html", "add_time_kind.html", form=form, kind=kind, mode="edit", alarmid=alarmid
form=form,
kind=kind,
mode="edit",
alarmid=alarmid,
) )
@ -158,9 +150,7 @@ def addtime_kind(kind):
resp.status_code = 400 resp.status_code = 400
return resp return resp
return render_template( return render_template("add_time_kind.html", form=form, kind=kind, mode="add")
"add_time_kind.html", form=form, kind=kind, mode="add"
)
@db.route("/add/audio") @db.route("/add/audio")
@ -190,10 +180,7 @@ def addaudio_kind(kind):
return resp return resp
return render_template( return render_template(
"add_audio_kind.html", "add_audio_kind.html", form=form, kind=kind, suggestions=get_suggestions()
form=form,
kind=kind,
suggestions=get_suggestions(),
) )
@ -227,7 +214,6 @@ def edit_event(alarmid):
if alarm is None: if alarm is None:
abort(404) abort(404)
allactions = model.get_all_actions() allactions = model.get_all_actions()
allactions.sort(key=lambda a: a.eid, reverse=True)
actions = tuple(model.get_actions_by_alarm(alarm)) actions = tuple(model.get_actions_by_alarm(alarm))
return render_template( return render_template(
"edit_event.html", "edit_event.html",

View file

@ -19,19 +19,6 @@ li.alarm .alarm-actions { display: none; }
{% block content %} {% block content %}
<div class="container-fluid"> <div class="container-fluid">
<div class="row">
{% if not show_all %}
<a href="{{url_for('db.events_calendar')}}?all=1"
class="btn btn-sm btn-default state-collapsed pull-right">
<span class="glyphicon glyphicon-resize-full"></span>
Mostra tutti gli eventi</a>
{% else %}
<a href="{{url_for('db.events_calendar')}}?all=0"
class="btn btn-sm btn-default state-collapsed pull-right">
<span class="glyphicon glyphicon-resize-small"></span>
Nascondi gli eventi troppo frequenti</a>
{% endif %}
</div>
{% for week, weekdays in weeks|dictsort %} {% for week, weekdays in weeks|dictsort %}
<div class="week row" id="week-{{week[0]}}-{{week[1]}}"> <div class="week row" id="week-{{week[0]}}-{{week[1]}}">
{% for day in weeks[week] %} {% for day in weeks[week] %}
@ -58,6 +45,5 @@ li.alarm .alarm-actions { display: none; }
<hr/> <hr/>
{%endfor %} {%endfor %}
</div><!-- container --> </div><!-- container -->
{% endblock content %} {% endblock content %}
{# vim: set ts=2 sw=2 noet: #} {# vim: set ts=2 sw=2 noet: #}

View file

@ -7,10 +7,6 @@
padding: 5px; padding: 5px;
border: 2px dashed #999; border: 2px dashed #999;
} }
#available-actions {
max-height: 50vw;
overflow-y: scroll;
}
</style> </style>
{%endblock styles%} {%endblock styles%}
{%block scripts %} {%block scripts %}
@ -53,8 +49,7 @@ $(function() {
You are currently editing: <code>{{alarm|tojson}}</code> You are currently editing: <code>{{alarm|tojson}}</code>
{% endif %} {% endif %}
<h2>Change actions</h2> <h2>Change actions</h2>
<div class="row"> <div>
<div id="available-actions" class="col-md-8">
Available actions: Available actions:
<ul> <ul>
{% for a in all_actions %} {% for a in all_actions %}
@ -65,9 +60,7 @@ $(function() {
{% endfor %} {% endfor %}
</ul> </ul>
</div> </div>
<div class="col-md-4"> <div>
<div class="row">
<div class="col-md-12">
Actions currently added: Actions currently added:
<ul id="selected"> <ul id="selected">
{% for a in actions %} {% for a in actions %}
@ -77,13 +70,11 @@ $(function() {
class="ui-state-default">{% if a.nick %} {{a.nick}} {% else %} {{a.eid}} {%endif%}</li> class="ui-state-default">{% if a.nick %} {{a.nick}} {% else %} {{a.eid}} {%endif%}</li>
{% endfor %} {% endfor %}
</ul> </ul>
</div><!-- .col-md-12 --> </div>
<div class="col-md-12"> <div>
<button>Save</button> <button>Save</button>
</div><!-- .col-md-12 --> </div>
</div><!-- .row --> </div>
</div><!-- .col-md-4 -->
</div><!-- .row -->
{% endblock content %} {% endblock content %}
{# vim: set ts=2 sw=2 noet: #} {# vim: set ts=2 sw=2 noet: #}

View file

@ -1,19 +1,17 @@
from __future__ import print_function from __future__ import print_function
from gevent import monkey
monkey.patch_all(subprocess=True)
import logging import logging
from datetime import datetime, timedelta from datetime import datetime, timedelta
import gevent import gevent
from gevent import monkey
from gevent.queue import Queue from gevent.queue import Queue
from .audiogen import audiogenerate
from .db import EventModel
from .eventutils import ParentedLet, Timer from .eventutils import ParentedLet, Timer
from .timegen import timegenerate from .timegen import timegenerate
from .audiogen import audiogenerate
monkey.patch_all(subprocess=True) from .db import EventModel
logging.getLogger("mpd").setLevel(logging.WARNING) logging.getLogger("mpd").setLevel(logging.WARNING)
@ -53,7 +51,6 @@ class Monitor(ParentedLet):
logging.exception( logging.exception(
"Could not generate " "an alarm from timespec %s", timespec "Could not generate " "an alarm from timespec %s", timespec
) )
return None
if when is None: if when is None:
# expired # expired
return None return None
@ -78,11 +75,9 @@ class Monitor(ParentedLet):
# but it is "tricky"; any small delay would cause the event to be # but it is "tricky"; any small delay would cause the event to be
# missed # missed
if delta is None: if delta is None:
# this is way too much logging! we need more levels! self.log.debug(
# self.log.debug( "Skipping event %s: will never ring", alarm.get("nick", alarm.eid)
# "Skipping event %s: will never ring", alarm.get("nick", alarm.eid) )
# )
pass
elif delta <= 2 * self.conf["EVENT_TICK_SECS"]: elif delta <= 2 * self.conf["EVENT_TICK_SECS"]:
self.log.debug( self.log.debug(
"Scheduling event %s (%ds) => %s", "Scheduling event %s (%ds) => %s",
@ -92,7 +87,7 @@ class Monitor(ParentedLet):
) )
self.schedule(alarm, actions, delta) self.schedule(alarm, actions, delta)
else: else:
self.log.debugv( self.log.debug(
"Skipping event %s too far (%ds)", "Skipping event %s too far (%ds)",
alarm.get("nick", alarm.eid), alarm.get("nick", alarm.eid),
delta, delta,
@ -109,9 +104,7 @@ class Monitor(ParentedLet):
if delta is None: if delta is None:
delta = self._alarm_missing_time(timespec) delta = self._alarm_missing_time(timespec)
audiogen = gevent.spawn_later( audiogen = gevent.spawn_later(delta, self.process_action, timespec, audiospecs)
delta, self.process_action, timespec, audiospecs
)
audiogen.parent_greenlet = self audiogen.parent_greenlet = self
audiogen.doc = 'Will wait {} seconds, then generate audio "{}"'.format( audiogen.doc = 'Will wait {} seconds, then generate audio "{}"'.format(
delta, ",".join(aspec.get("nick", "") for aspec in audiospecs) delta, ",".join(aspec.get("nick", "") for aspec in audiospecs)

View file

@ -64,12 +64,6 @@ def percentwait(songs, context, conf, getdur=get_duration):
continue continue
eventduration += songduration eventduration += songduration
if eventduration == 0:
# must be an error! mutagen support is not always perfect
return (
True,
("mutagen could not calculate length of %s" % ",".join(songs["uris"])),
)
wait = eventduration * (percentwait / 100.0) wait = eventduration * (percentwait / 100.0)
if remaining > wait: if remaining > wait:
return False, "remaining %d max allowed %d" % (remaining, wait) return False, "remaining %d max allowed %d" % (remaining, wait)

View file

@ -1,13 +1,6 @@
import fnmatch
import logging
import mimetypes
import os import os
import posixpath import fnmatch
import urllib.request import mimetypes
from tempfile import mkstemp
from urllib.parse import urlparse
log = logging.getLogger(__name__)
def scan_dir(dirname, extension=None): def scan_dir(dirname, extension=None):
@ -44,27 +37,3 @@ def shortname(path):
name = name.rsplit(".", 1)[0] # no extension name = name.rsplit(".", 1)[0] # no extension
name = "".join(c for c in name if c.isalnum()) # no strange chars name = "".join(c for c in name if c.isalnum()) # no strange chars
return name return name
def download_http(url, destdir=None, copy=False, prefix="httpdl"):
if url.split(":")[0] not in ("http", "https"):
log.warning("Not a valid URL: %s", url)
return None
ext = url.split(".")[-1]
if ext.lower() not in ("mp3", "ogg", "oga", "wma", "m4a"):
log.warning('Invalid format (%s) for "%s"', ext, url)
return None
if not copy:
return url
fname = posixpath.basename(urlparse(url).path)
# sanitize
fname = "".join(
c for c in fname if c.isalnum() or c in list("._-")
).rstrip()
tmp = mkstemp(
suffix="." + ext, prefix="%s-%s-" % (prefix, fname), dir=destdir
)
os.close(tmp[0])
log.info("downloading %s -> %s", url, tmp[1])
fname, headers = urllib.request.urlretrieve(url, tmp[1])
return "file://%s" % os.path.realpath(tmp[1])

View file

@ -2,27 +2,26 @@
This module is for the main application logic This module is for the main application logic
""" """
from __future__ import print_function from __future__ import print_function
import json
import logging
import logging.config
import os
import signal
import subprocess
import sys
import tempfile
from time import sleep
import gevent
from gevent import monkey from gevent import monkey
from gevent.pywsgi import WSGIServer
from larigira.config import get_conf
from larigira.mpc import Controller, get_mpd_client
from larigira.rpc import create_app
monkey.patch_all(subprocess=True) monkey.patch_all(subprocess=True)
import sys
import os
import tempfile
import signal
from time import sleep
import logging
import logging.config
import subprocess
import gevent
from gevent.pywsgi import WSGIServer
from .mpc import Controller, get_mpd_client
from .config import get_conf
from .rpc import create_app
def on_main_crash(*args, **kwargs): def on_main_crash(*args, **kwargs):
print('A crash occurred in "main" greenlet. Aborting...') print('A crash occurred in "main" greenlet. Aborting...')
@ -31,14 +30,12 @@ def on_main_crash(*args, **kwargs):
class Larigira(object): class Larigira(object):
def __init__(self): def __init__(self):
self.log = logging.getLogger("larigira") self.log = logging.getLogger("larigira")
self.conf = get_conf() self.conf = get_conf()
self.controller = Controller(self.conf) self.controller = Controller(self.conf)
self.controller.link_exception(on_main_crash) self.controller.link_exception(on_main_crash)
self.http_server = WSGIServer( self.http_server = WSGIServer(
(self.conf["HTTP_ADDRESS"], int(self.conf["HTTP_PORT"])), ("", int(self.conf["HTTP_PORT"])), create_app(self.controller.q, self)
create_app(self.controller.q, self),
) )
def start(self): def start(self):
@ -60,57 +57,28 @@ def sd_notify(ready=False, status=None):
def main(): def main():
logging.addLevelName(9, "DEBUGV")
if get_conf()["LOG_CONFIG"]:
logging.config.fileConfig(
get_conf()["LOG_CONFIG"], disable_existing_loggers=True
)
else:
log_format = (
"%(asctime)s|%(levelname)s[%(name)s:%(lineno)d] %(message)s"
)
logging.basicConfig(
level="DEBUGV" if get_conf()["DEBUG"] else logging.INFO,
format=log_format,
datefmt="%H:%M:%S",
)
def debugv(self, message, *args, **kws):
if self.isEnabledFor(9):
self._log(9, message, args, **kws)
logging.Logger.debugv = debugv
logging.debug(
"Starting larigira with this conf:\n%s",
json.dumps(get_conf(), indent=2),
)
if get_conf()["UMASK"]:
umask = int(get_conf()["UMASK"], base=8)
logging.debug(
"Setting umask %s (decimal: %d)", get_conf()["UMASK"], umask
)
os.umask(umask)
tempfile.tempdir = os.environ["TMPDIR"] = os.path.join( tempfile.tempdir = os.environ["TMPDIR"] = os.path.join(
os.getenv("TMPDIR", "/tmp"), "larigira.%d" % os.getuid() os.getenv("TMPDIR", "/tmp"), "larigira.%d" % os.getuid()
) )
if not os.path.isdir(os.environ["TMPDIR"]): if not os.path.isdir(os.environ["TMPDIR"]):
os.makedirs(os.environ["TMPDIR"]) os.makedirs(os.environ["TMPDIR"])
if get_conf()["LOG_CONFIG"]:
logging.config.fileConfig(
get_conf()["LOG_CONFIG"], disable_existing_loggers=True
)
else:
log_format = "%(asctime)s|%(levelname)s[%(name)s:%(lineno)d] %(message)s"
logging.basicConfig(
level=logging.DEBUG if get_conf()["DEBUG"] else logging.INFO,
format=log_format,
datefmt="%H:%M:%S",
)
if get_conf()["MPD_WAIT_START"]: if get_conf()["MPD_WAIT_START"]:
while True: while True:
try: try:
get_mpd_client(get_conf()) get_mpd_client(get_conf())
except Exception as exc: except Exception:
print("exc", exc, file=sys.stderr) logging.debug("Could not connect to MPD, waiting")
logging.debug(
"Could not connect to MPD at (%s,%s), waiting",
get_conf()["MPD_HOST"],
get_conf()["MPD_PORT"],
)
sd_notify(status="Waiting MPD connection") sd_notify(status="Waiting MPD connection")
sleep(int(get_conf()["MPD_WAIT_START_RETRYSECS"])) sleep(int(get_conf()["MPD_WAIT_START_RETRYSECS"]))
else: else:

View file

@ -1,19 +1,17 @@
from __future__ import print_function from __future__ import print_function
import logging import logging
import signal import signal
import mpd
from pkg_resources import iter_entry_points from pkg_resources import iter_entry_points
import gevent import gevent
from gevent.queue import Queue from gevent.queue import Queue
import mpd
from .audiogen import audiogenerate
from .entrypoints_utils import get_avail_entrypoints
from .event import Monitor from .event import Monitor
from .eventutils import ParentedLet, Timer from .eventutils import ParentedLet, Timer
from .audiogen import audiogenerate
from .unused import UnusedCleaner from .unused import UnusedCleaner
from .entrypoints_utils import get_avail_entrypoints
def get_mpd_client(conf): def get_mpd_client(conf):
@ -53,9 +51,7 @@ class MPDWatcher(ParentedLet):
FileNotFoundError, FileNotFoundError,
) as exc: ) as exc:
self.log.warning( self.log.warning(
"Connection to MPD failed (%s: %s)", "Connection to MPD failed (%s: %s)", exc.__class__.__name__, exc
exc.__class__.__name__,
exc,
) )
self.client = None self.client = None
first_after_connection = True first_after_connection = True
@ -87,15 +83,9 @@ class Player:
mpd_client = mpd.MPDClient(use_unicode=True) mpd_client = mpd.MPDClient(use_unicode=True)
try: try:
mpd_client.connect(self.conf["MPD_HOST"], self.conf["MPD_PORT"]) mpd_client.connect(self.conf["MPD_HOST"], self.conf["MPD_PORT"])
except ( except (mpd.ConnectionError, ConnectionRefusedError, FileNotFoundError) as exc:
mpd.ConnectionError,
ConnectionRefusedError,
FileNotFoundError,
) as exc:
self.log.warning( self.log.warning(
"Connection to MPD failed (%s: %s)", "Connection to MPD failed (%s: %s)", exc.__class__.__name__, exc
exc.__class__.__name__,
exc,
) )
raise gevent.GreenletExit() raise gevent.GreenletExit()
return mpd_client return mpd_client
@ -134,8 +124,8 @@ class Player:
uris = greenlet.value uris = greenlet.value
for uri in uris: for uri in uris:
assert type(uri) is str, type(uri) assert type(uri) is str, type(uri)
mpd_client.add(uri.strip())
self.tmpcleaner.watch(uri.strip()) self.tmpcleaner.watch(uri.strip())
mpd_client.add(uri.strip())
picker.link_value(add) picker.link_value(add)
picker.start() picker.start()
@ -156,9 +146,7 @@ class Player:
try: try:
ret = ef(songs=songs, context=ctx, conf=self.conf) ret = ef(songs=songs, context=ctx, conf=self.conf)
except ImportError as exc: except ImportError as exc:
self.log.warn( self.log.warn("Filter %s skipped: %s" % (entrypoint.name, exc))
"Filter %s skipped: %s" % (entrypoint.name, exc)
)
continue continue
if ret is None: # bad behavior! if ret is None: # bad behavior!
continue continue
@ -169,11 +157,6 @@ class Player:
reason = "Filtered by %s (%s)" % (entrypoint.name, reason) reason = "Filtered by %s (%s)" % (entrypoint.name, reason)
if ret is False: if ret is False:
return ret, reason return ret, reason
else:
if reason:
self.log.debug(
"filter %s says ok: %s", entrypoint.name, reason
)
return True, "Passed through %s" % ",".join(availfilters) return True, "Passed through %s" % ",".join(availfilters)
def enqueue(self, songs): def enqueue(self, songs):
@ -214,11 +197,6 @@ class Player:
self.log.exception("Cannot insert song %s", uri) self.log.exception("Cannot insert song %s", uri)
self.tmpcleaner.watch(uri.strip()) self.tmpcleaner.watch(uri.strip())
def play(self):
"""make sure that MPD is playing"""
mpd_client = self._get_mpd()
mpd_client.play()
class Controller(gevent.Greenlet): class Controller(gevent.Greenlet):
def __init__(self, conf): def __init__(self, conf):
@ -246,6 +224,7 @@ class Controller(gevent.Greenlet):
gevent.Greenlet.spawn(self.player.check_playlist) gevent.Greenlet.spawn(self.player.check_playlist)
while True: while True:
value = self.q.get() value = self.q.get()
self.log.debug("<- %s", str(value))
# emitter = value['emitter'] # emitter = value['emitter']
kind = value["kind"] kind = value["kind"]
args = value["args"] args = value["args"]
@ -253,8 +232,6 @@ class Controller(gevent.Greenlet):
kind == "mpc" and args[0] in ("player", "playlist", "connect") kind == "mpc" and args[0] in ("player", "playlist", "connect")
): ):
gevent.Greenlet.spawn(self.player.check_playlist) gevent.Greenlet.spawn(self.player.check_playlist)
if self.conf["MPD_ENFORCE_ALWAYS_PLAYING"]:
gevent.Greenlet.spawn(self.player.play)
try: try:
self.player.tmpcleaner.check_playlist() self.player.tmpcleaner.check_playlist()
except: except:
@ -272,9 +249,7 @@ class Controller(gevent.Greenlet):
self.log.exception( self.log.exception(
"Error while adding to queue; " "bad audiogen output?" "Error while adding to queue; " "bad audiogen output?"
) )
elif ( elif (kind == "signal" and args[0] == signal.SIGALRM) or kind == "refresh":
kind == "signal" and args[0] == signal.SIGALRM
) or kind == "refresh":
# it's a tick! # it's a tick!
self.log.debug("Reload") self.log.debug("Reload")
self.monitor.q.put(dict(kind="forcetick")) self.monitor.q.put(dict(kind="forcetick"))

View file

@ -1,17 +1,24 @@
import gc
import logging import logging
import gc
from copy import deepcopy from copy import deepcopy
from flask import (Blueprint, Flask, abort, current_app, jsonify, redirect,
render_template, request)
from flask.ext.babel import Babel
from flask_bootstrap import Bootstrap
from cachelib import SimpleCache
from greenlet import greenlet from greenlet import greenlet
from flask import (
current_app,
Blueprint,
Flask,
jsonify,
render_template,
request,
abort,
redirect,
)
from flask_bootstrap import Bootstrap
from flask.ext.babel import Babel
from werkzeug.contrib.cache import SimpleCache
from .config import get_conf
from .dbadmin import db from .dbadmin import db
from .config import get_conf
rpc = Blueprint("rpc", __name__, url_prefix=get_conf()["ROUTE_PREFIX"] + "/api") rpc = Blueprint("rpc", __name__, url_prefix=get_conf()["ROUTE_PREFIX"] + "/api")
viewui = Blueprint( viewui = Blueprint(

View file

@ -1,5 +1,4 @@
/* global jQuery */ /* global jQuery */
jQuery(function ($) { jQuery(function ($) {
$('.button').button({ $('.button').button({
icons: { icons: {

View file

@ -6,7 +6,7 @@ monkey.patch_all(subprocess=True)
import pytest import pytest
from larigira.rpc import create_app from larigira.rpc import create_app
from larigira.main import Larigira from larigira.larigira import Larigira
@pytest.fixture @pytest.fixture

View file

@ -1,12 +1,18 @@
import logging import logging
from datetime import datetime
from datetime import datetime
from pytimeparse.timeparse import timeparse from pytimeparse.timeparse import timeparse
from flask_wtf import Form from flask_wtf import Form
from wtforms import (
StringField,
validators,
SubmitField,
SelectMultipleField,
ValidationError,
)
from larigira.formutils import EasyDateTimeField from larigira.formutils import EasyDateTimeField
from wtforms import (SelectMultipleField, StringField, SubmitField,
ValidationError, validators)
log = logging.getLogger(__name__) log = logging.getLogger(__name__)
@ -84,9 +90,9 @@ class FrequencyAlarmForm(Form):
def populate_from_timespec(self, timespec): def populate_from_timespec(self, timespec):
if "nick" in timespec: if "nick" in timespec:
self.nick.data = timespec["nick"] self.nick.data = timespec["nick"]
if timespec.get("start"): if "start" in timespec:
self.start.data = datetime.fromtimestamp(timespec["start"]) self.start.data = datetime.fromtimestamp(timespec["start"])
if timespec.get("end"): if "end" in timespec:
self.end.data = datetime.fromtimestamp(timespec["end"]) self.end.data = datetime.fromtimestamp(timespec["end"])
if "weekdays" in timespec: if "weekdays" in timespec:
self.weekdays.data = timespec["weekdays"] self.weekdays.data = timespec["weekdays"]
@ -117,6 +123,6 @@ def frequencyalarm_receive(form):
obj["start"] = int(form.start.data.strftime("%s")) obj["start"] = int(form.start.data.strftime("%s"))
else: else:
obj["start"] = 0 obj["start"] = 0
if form.end.data:
obj["end"] = int(form.end.data.strftime("%s")) if form.end.data else None obj["end"] = int(form.end.data.strftime("%s"))
return obj return obj

View file

@ -4,10 +4,9 @@ This component will look for files to be removed. There are some assumptions:
own specific TMPDIR own specific TMPDIR
* MPD URIs are parsed, and only file:/// is supported * MPD URIs are parsed, and only file:/// is supported
""" """
import logging
import os import os
from os.path import normpath from os.path import normpath
import logging
import mpd import mpd
@ -65,13 +64,11 @@ class UnusedCleaner:
"""check playlist + internal watchlist to see what can be removed""" """check playlist + internal watchlist to see what can be removed"""
mpdc = self._get_mpd() mpdc = self._get_mpd()
files_in_playlist = { files_in_playlist = {
song["file"] song["file"] for song in mpdc.playlistid() if song["file"].startswith("/")
for song in mpdc.playlistid()
if song["file"].startswith("/")
} }
for fpath in self.waiting_removal_files - files_in_playlist: for fpath in self.waiting_removal_files - files_in_playlist:
# we can remove it! # we can remove it!
self.log.debug("removing unused: %s", fpath) self.log.debug("removing unused: %s", fpath)
self.waiting_removal_files.remove(fpath) self.waiting_removal_files.remove(fpath)
if os.path.exists(fpath) and self.conf["REMOVE_UNUSED_FILES"]: if os.path.exists(fpath):
os.unlink(fpath) os.unlink(fpath)

153
setup.py
View file

@ -1,5 +1,5 @@
import os
import sys import sys
import os
from setuptools import setup from setuptools import setup
from setuptools.command.test import test as TestCommand from setuptools.command.test import test as TestCommand
@ -11,7 +11,7 @@ def read(fname):
class PyTest(TestCommand): class PyTest(TestCommand):
user_options = [("pytest-args=", "a", "Arguments to pass to py.test")] user_options = [('pytest-args=', 'a', "Arguments to pass to py.test")]
def initialize_options(self): def initialize_options(self):
TestCommand.initialize_options(self) TestCommand.initialize_options(self)
@ -25,104 +25,89 @@ class PyTest(TestCommand):
def run_tests(self): def run_tests(self):
# import here, cause outside the eggs aren't loaded # import here, cause outside the eggs aren't loaded
import pytest import pytest
errno = pytest.main(self.pytest_args) errno = pytest.main(self.pytest_args)
sys.exit(errno) sys.exit(errno)
setup(name='larigira',
setup( version='1.3.1',
name="larigira", description='A radio automation based on MPD',
version="1.3.3", long_description=read('README.rst'),
description="A radio automation based on MPD", long_description_content_type='text/x-rst',
long_description=read("README.rst"), author='boyska',
long_description_content_type="text/x-rst", author_email='piuttosto@logorroici.org',
author="boyska", license='AGPL',
author_email="piuttosto@logorroici.org", packages=['larigira', 'larigira.dbadmin', 'larigira.filters'],
license="AGPL",
packages=["larigira", "larigira.dbadmin", "larigira.filters"],
install_requires=[ install_requires=[
"Babel==2.6.0", 'Babel==2.6.0',
"Flask-Babel==1.0.0", 'Flask-Babel==0.12.2',
"pyxdg==0.26", 'pyxdg',
"gevent==1.4.0", 'gevent',
"flask-bootstrap", 'flask-bootstrap',
"python-mpd2", 'python-mpd2',
"wtforms==2.2.1", 'wtforms',
"Flask-WTF==0.14.2", 'Flask-WTF',
"flask==0.11", 'flask==0.11',
"pytimeparse==1.1.8", 'pytimeparse',
"croniter==0.3.29", 'croniter==0.3.29',
"werkzeug==0.14.1", 'tinydb'
"cachelib==0.1",
"tinydb==3.12.2",
"lxml==4.5.1",
"requests==2.23.0",
], ],
tests_require=["pytest-timeout==1.0", "py>=1.4.29", "pytest==3.0"], tests_require=['pytest-timeout==1.0', 'py>=1.4.29', 'pytest==3.0', ],
python_requires=">=3.5", python_requires='>=3.5',
extras_require={"percentwait": ["mutagen"]}, extras_require={
cmdclass={"test": PyTest}, 'percentwait': ['mutagen'],
},
cmdclass={'test': PyTest},
zip_safe=False, zip_safe=False,
include_package_data=True, include_package_data=True,
entry_points={ entry_points={
"console_scripts": [ 'console_scripts': ['larigira=larigira.larigira:main',
"larigira=larigira.main:main", 'larigira-timegen=larigira.timegen:main',
"larigira-timegen=larigira.timegen:main", 'larigira-audiogen=larigira.audiogen:main',
"larigira-audiogen=larigira.audiogen:main", 'larigira-dbmanage=larigira.event_manage:main'],
"larigira-dbmanage=larigira.event_manage:main", 'larigira.audiogenerators': [
'mpd = larigira.audiogen_mpdrandom:generate_by_artist',
'static = larigira.audiogen_static:generate',
'http = larigira.audiogen_http:generate',
'randomdir = larigira.audiogen_randomdir:generate',
'mostrecent = larigira.audiogen_mostrecent:generate',
'script = larigira.audiogen_script:generate',
], ],
"larigira.audiogenerators": [ 'larigira.timegenerators': [
"mpd = larigira.audiogen_mpdrandom:generate_by_artist", 'frequency = larigira.timegen_every:FrequencyAlarm',
"static = larigira.audiogen_static:generate", 'single = larigira.timegen_every:SingleAlarm',
"http = larigira.audiogen_http:generate", 'cron = larigira.timegen_cron:CronAlarm',
"podcast = larigira.audiogen_podcast:generate",
"randomdir = larigira.audiogen_randomdir:generate",
"mostrecent = larigira.audiogen_mostrecent:generate",
"script = larigira.audiogen_script:generate",
], ],
"larigira.timegenerators": [ 'larigira.timeform_create': [
"frequency = larigira.timegen_every:FrequencyAlarm", 'single = larigira.timeform_base:SingleAlarmForm',
"single = larigira.timegen_every:SingleAlarm", 'frequency = larigira.timeform_base:FrequencyAlarmForm',
"cron = larigira.timegen_cron:CronAlarm", 'cron = larigira.timeform_cron:CronAlarmForm',
], ],
"larigira.timeform_create": [ 'larigira.timeform_receive': [
"single = larigira.timeform_base:SingleAlarmForm", 'single = larigira.timeform_base:singlealarm_receive',
"frequency = larigira.timeform_base:FrequencyAlarmForm", 'frequency = larigira.timeform_base:frequencyalarm_receive',
"cron = larigira.timeform_cron:CronAlarmForm", 'cron = larigira.timeform_cron:cronalarm_receive',
], ],
"larigira.timeform_receive": [ 'larigira.audioform_create': [
"single = larigira.timeform_base:singlealarm_receive", 'static = larigira.audioform_static:StaticAudioForm',
"frequency = larigira.timeform_base:frequencyalarm_receive", 'http = larigira.audioform_http:AudioForm',
"cron = larigira.timeform_cron:cronalarm_receive", 'script = larigira.audioform_script:ScriptAudioForm',
'randomdir = larigira.audioform_randomdir:Form',
'mostrecent = larigira.audioform_mostrecent:AudioForm',
], ],
"larigira.audioform_create": [ 'larigira.audioform_receive': [
"static = larigira.audioform_static:StaticAudioForm", 'static = larigira.audioform_static:staticaudio_receive',
"http = larigira.audioform_http:AudioForm", 'http = larigira.audioform_http:audio_receive',
"podcast = larigira.audioform_podcast:AudioForm", 'script = larigira.audioform_script:scriptaudio_receive',
"script = larigira.audioform_script:ScriptAudioForm", 'randomdir = larigira.audioform_randomdir:receive',
"randomdir = larigira.audioform_randomdir:Form", 'mostrecent = larigira.audioform_mostrecent:audio_receive',
"mostrecent = larigira.audioform_mostrecent:AudioForm",
], ],
"larigira.audioform_receive": [ 'larigira.eventfilter': [
"static = larigira.audioform_static:staticaudio_receive", 'maxwait = larigira.filters:maxwait',
"http = larigira.audioform_http:audio_receive", 'percentwait = larigira.filters:percentwait',
"podcast = larigira.audioform_podcast:audio_receive",
"script = larigira.audioform_script:scriptaudio_receive",
"randomdir = larigira.audioform_randomdir:receive",
"mostrecent = larigira.audioform_mostrecent:audio_receive",
],
"larigira.eventfilter": [
"maxwait = larigira.filters:maxwait",
"percentwait = larigira.filters:percentwait",
], ],
}, },
classifiers=[ classifiers=[
"License :: OSI Approved :: GNU Affero General Public License v3", "License :: OSI Approved :: GNU Affero General Public License v3",
"Development Status :: 5 - Production/Stable", "Programming Language :: Python :: 3",
"Programming Language :: Python :: 3.4", ]
"Programming Language :: Python :: 3.5",
"Programming Language :: Python :: 3.6",
"Programming Language :: Python :: 3.7",
"Topic :: Multimedia :: Sound/Audio",
],
) )