Stream Recorder
Stream Recorder ist ein Skript zur zeitgesteuerten Aufnahme von TV Livestreams aus dem TV-Browser heraus.
Inhaltsverzeichnis
Vorraussetzungen
- Internetanschluß
- Linux Pc
- at (Paket)
- mencoder (Paket)
Installation
Das Skript
Benötigt wird die Installation der Pakete at und mencoder.
Der at Dienst wird von systemd gesteuert. Zur Aktivuerung gibt man auf der Konsole (sudo systemctl enable --now atd.service)
ein.
Das Skript wird kopieren und z.B. mit dem Namen (streamrecoder)
in einen Ordner mit Benutzerschreibrechten abgespeichert und mit (chmod a+rx streamrecorder)
ausführbar gemacht.
Konfiguration des TV-Browsers
Das Plugin Aufnahmesteuerung wird geladen. <p>Im Tv-Browser Menu Extras > Aufnahmesteuerung > Geräte > Gerät-hinzufügen den Standardtreiber auswählen. Beim Eintrag Gerät kann ein beliebiger Namen eingetragen werden.
Als "Applikation" wird das Skript (streamrecoder)
mit Pfad angegeben.
Parameter für die Aufnahme:
record {start_hour} {start_minute} {start_year} {start_month} {start_day} {end_hour} {end_minute} {end_year} {end_month} {end_day} {channel_name} "{title}" {production_year} {episode_number} "{episode}"
Parameter für das Löschen:
delete {start_hour} {start_minute} {start_year} {start_month} {start_day} {end_hour} {end_minute} {end_year} {end_month} {end_day} {channel_name} "{title}" {production_year} {episode_number} "{episode}"
Bei den Einstellungen können mehere Aufnahme gleichzeitig erlaubt werden.
Auch wird die Aufnahme schon laufender Sendungen unterstützt.
Die Sender URLs der öffentlich rechtlichen Sender sind im Script gelistet (Stand November 2017). Man kann beliebig viele Sender hinzufügen. Die Namen der Sender sollten im TV-Browser und im Script miteinander übereinstimmen. Leerstellen im Sendernamen sollten im Skript mit Unterstrich eingetragen werden. Der Speicherort der Aufnahme ist auf ~/Video vor eingestellt und kann im Script in der Benutzer Konfiguration auf eigene Bedürfnisse angepasst werden. Es kann sein, dass sich gelegentlich die URL eines Livestreams ändert. Eine gute Quelle für aktuelle Livestream URLs bietet das Programm Medithek-View.
Benutzung
Die Benutzung ist im Grunde selbsterklärend. Mit der rechten Maustaste wählt man eine Sendung aus. Mit der Linken Maustaste wird auf Aufnahme gedrückt. Zum Rückgängig machen der Aufnahmejobs geht man entsprechend vor. Ein Dialogfenster von TV-Browser öffnet sich und gibt jeweils Auskunft über den Programmablauf.
Hintergrund
Zur Zeitsteuerung habe ich at ausgewählt. Cronie und systemd.timers sind besser für wiederkehrende Ereignisse geeignet. Wer wissen möchte wie die at-Jobs gespeichert werden, kann auf der Konsole die Befehle atq, at -c <job> und atrm <job> mal ausprobieren. Für das eigentliche streaming ist mencoder verantwortlich.
Scriptversionen
Folgende Funktionen wurden bei den Versionen hinzugefügt: Version 0.2: Paralelle Aufnahmen, Version 0.3: Titelkorrektur, Version 0.4: Aufnahmen laufender Sendungen, Version 0.5 Serien mit Angabe der Episode. 1.0 Bugfixes, 1.1 Leerstellen beim Sendernamen, Aktualisierung der Sender URLs.
Viel Spaß damit TuxNix
Das Script
#!/bin/bash ##### Programm: streamrecorder Version 1.1 - 19.Nov. 2017 ##### From TuxNix Lizenz GPLv2 with no garanty at all. ##### Parameter für Tv-Browser: Aufnahme #record {start_hour} {start_minute} {start_year} {start_month} {start_day} {end_hour} {end_minute} {end_year} {end_month} {end_day} {channel_name} "{title}" {production_year} {episode_number} "{episode}" ##### Parameter für Tv-Browser: Löschen #delete {start_hour} {start_minute} {start_year} {start_month} {start_day} {end_hour} {end_minute} {end_year} {end_month} {end_day} {channel_name} "{title}" {production_year} {episode_number} "{episode}" ##### benötigt die Programm-Pakete #at #mencoder # Der at service muss mit dem Befehl: 'sudo systemctl enable --now atd.service' aktiviert werden. ##### Benutzer Konfiguration recPath=~/Videos/ ##### Das Script args=("$@") operation=${args[0]} sStunde=${args[1]} if (( ${#sStunde} < 2 )) ; then sStunde="0"${args[1]} fi sMinute=${args[2]} if (( ${#sMinute} < 2 )) ; then sMinute="0"$sMinute fi sJahr=${args[3]} sMonat=${args[4]} if (( ${#sMonat} < 2 )) ; then sMonat="0"$sMonat fi sTag=${args[5]} if (( ${#sTag} < 2 )) ; then sTag="0"$sTag fi eStunde=${args[6]} if (( ${#eStunde} < 2 )) ; then eStunde="0"$eStunde fi eMinute=${args[7]} if (( ${#eMinute} < 2 )) ; then eMinute="0"$eMinute fi eJahr=${args[8]} eMonat=${args[9]} if (( ${#args[9]} < 2 )) ; then eMonat="0"${args[9]} fi eTag=${args[10]} if (( ${#args[10]} < 2 )) ; then eTag="0"${args[10]} fi Kanal="${args[11]}" Kanal=`echo $Kanal | sed 's/ /_/g'` Kanal=`echo $Kanal | sed 's!"!!g'` Titel=${args[12]} if (( ${#args[13]} > 3 )) ; then ProdJahr="-${args[13]}" fi if (( ${#args[14]} > 2 )) ; then EpisodenNr="-${args[14]}" fi if (( ${#args[15]} > 0 )) ; then Episode="-${args[15]}" fi Name="$Titel$ProdJahr$EpisodenNr$Episode" Name=`echo $Name | sed 's/ /_/g'` Name=`echo $Name | sed 's!/!-!g'` Name=`echo $Name | sed 's![()"]!!g'` Name=`echo $Name | sed "s!\x27!!g"` Name=`echo $Name | sed "s!&!u!g"` if [ $Kanal == 'Das_Erste_(ARD)' ] ; then stream_url='http://daserste_live-lh.akamaihd.net/i/daserste_de@91204/master.m3u8' elif [ $Kanal == 'ARD-alpha' ] ; then stream_url="http://livestreams.br.de/i/bralpha_germany@119899/master.m3u8" elif [ $Kanal == 'One' ] ; then stream_url="http://wdr_einsfestival-lh.akamaihd.net/i/wdr_einsfestival@328300/master.m3u8" elif [ $Kanal == 'tagesschau24' ] ; then stream_url="http://tagesschau-lh.akamaihd.net/i/tagesschau_1@119231/master.m3u8" elif [ $Kanal == 'arte' ] ; then stream_url="http://artelive-lh.akamaihd.net/i/artelive_de@393591/master.m3u8" elif [ $Kanal == 'ARTE_(francais)' ] ; then stream_url="http://artelive-lh.akamaihd.net/i/artelive_fr@344805/master.m3u8" elif [ $Kanal == 'BR' ] ; then stream_url="http://livestreams.br.de/i/bfsnord_germany@119898/master.m3u8" elif [ $Kanal == 'KiKA' ] ; then stream_url="http://kika_geo-lh.akamaihd.net/i/livetvkika_de@75114/master.m3u8" elif [ $Kanal == 'NDR_Niedersachsen' ] ; then stream_url="http://ndrfs-lh.akamaihd.net/i/ndrfs_nds@430233/master.m3u8" elif [ $Kanal == 'PHOENIX' ] ; then stream_url="http://zdf0910-lh.akamaihd.net/i/de09_v1@392871/master.m3u8" elif [ $Kanal == 'RBB_Brandenburg' ] ; then stream_url="http://rbblive-lh.akamaihd.net/i/rbb_brandenburg@349369/master.m3u8" elif [ $Kanal == '3sat' ] ; then stream_url="http://zdf0910-lh.akamaihd.net/i/dach10_v1@392872/master.m3u8" elif [ $Kanal == 'SWR_BW' ] ; then stream_url="http://swrbw-lh.akamaihd.net/i/swrbw_live@196738/master.m3u8" elif [ $Kanal == 'WDR' ] ; then stream_url="http://wdrfsgeo-lh.akamaihd.net/i/wdrfs_geogeblockt@530016/master.m3u8" elif [ $Kanal == 'ZDF' ] ; then stream_url="http://zdf1314-lh.akamaihd.net/i/de14_v1@392878/master.m3u8" elif [ $Kanal == 'ZDFinfo' ] ; then stream_url="http://zdf1112-lh.akamaihd.net/i/de12_v1@392882/master.m3u8" elif [ $Kanal == 'ZDFneo' ] ; then stream_url="http://zdf1314-lh.akamaihd.net/i/de13_v1@392877/master.m3u8" else echo "Abbruch: Der Sender '$Kanal' ist im Aufnahmeskript nicht gelistet" echo "Bitte Groß und Kleinschreibung beachten" echo "Die Namen der Sender die im TV-Browser ein Leerzeichen haben, müssen im Skript mit Unterstrich eintragen werden" echo "Umlaute werden nicht unterstützt" exit 1 fi sysTime=`date --date='now' +%s` startTime=`date --date="$sJahr-$sMonat-$sTag $sStunde:$sMinute" +%s` startNow=`date --date='now' +%R' '%F` stopNow=`date --date="1 minute" +%R' '%F` startZeit=$sStunde:$sMinute' '$sJahr-$sMonat-$sTag endZeit=$eStunde:$eMinute' '$eJahr-$eMonat-$eTag if [ $operation == 'record' ] && (( $sysTime > $startTime )) ; then echo "`date --date='now' +%R` - $eStunde:$eMinute Uhr $eTag.$eMonat.$eJahr: $Titel" echo "" echo "Die Aufnahme wurde gestartet und wird nach" echo $recPath$Name"-part.mp4 gespeichert." echo "mencoder $stream_url -vid 9 -oac pcm -ovc copy -o $recPath$Name-part.mp4" | at $startNow > /dev/null 2>&1 echo "fuser -k -TERM $recPath$Name-part.mp4" | at $endZeit > /dev/null 2>&1 elif [ $operation == 'record' ] && (( $sysTime <= $startTime )) ; then echo "$sStunde:$sMinute - $eStunde:$eMinute Uhr $eTag.$eMonat.$eJahr: $Titel" echo "" echo "Die Aufnahme wird veranlasst und nach" echo $recPath$Name".mp4 gespeichert." echo "mencoder $stream_url -vid 9 -oac pcm -ovc copy -o $recPath$Name.mp4" | at $startZeit > /dev/null 2>&1 echo "fuser -k -TERM $recPath$Name.mp4" | at $endZeit > /dev/null 2>&1 elif [ $operation == 'delete' ] ; then Monate=([1]=Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec) if (( $sysTime >= $startTime )) ; then echo "fuser -k -TERM $recPath$Name.mp4" | at $stopNow > /dev/null 2>&1 echo "fuser -k -TERM $recPath$Name-part.mp4" | at $stopNow > /dev/null 2>&1 echo "Die laufende Aufnahme von '$Titel' wird beendet!" else sTag=${args[5]} if (( ${#sTag} < 2 )) ; then sTag=" "$sTag fi atq | grep "${Monate[${args[4]}]} $sTag $sStunde:$sMinute:00 $sJahr" > atq.txt Jobs=`cut -f1 atq.txt` for Job in $Jobs ; do if at -c $Job | grep $Name > /dev/null 2>&1 ; then atrm $Job echo "Der at Start-Job $Job wurde geloescht!" fi done fi eTag=${args[10]} if (( ${#eTag} < 2 )) ; then eTag=" "$eTag fi atq | grep "${Monate[${args[9]}]} $eTag $eStunde:$eMinute:00 $eJahr" > atq.txt Jobs=`cut -f1 atq.txt` for Job in $Jobs ; do if at -c $Job | grep $Name > /dev/null 2>&1 ; then atrm $Job echo "Der at Stop-Job $Job wurde geloescht!" fi done else echo "Abbruch: Der Operator $operation ist nicht korrekt. Bitte überprüfe die Parameter im TV-Browser Aufnahme Plugin" exit 2 fi
Anregungen und Verbesserungsvorschläge zum Script können im Forum ausgetauscht werden.