Stream Recorder

Aus TV-Browser Wiki
Version vom 19. Juni 2023, 04:48 Uhr von TuxNix⧼word-separator⧽⧼parentheses⧽ ⧼parentheses⧽
⧼revision-nav⧽
Wechseln zu: Navigation⧼comma-separator⧽Suche

Stream Recorder ist ein Shell-Skript zur zeitgesteuerten Aufnahme von TV Livestreams aus dem TV-Browser heraus.

Voraussetzungen

  • Internetanschluss
  • Linux Pc
  • at (Paket)
  • ffmpeg (Paket)
  • TV-Browser - Aufnahmesteuerung Plugin

Installation

Das Skript

Benötigt wird die Installation der Pakete 'at' und 'ffmpeg'.

Der at Dienst wird von systemd gesteuert. Zur Aktivierung gibt man auf der Konsole (sudo systemctl enable --now atd.service) ein.

Das Skript wird kopiert und mit dem Namen streamrecoder in den Ordner /usr/local/bin/ abgespeichert. Danach wird es mit chmod a+rx /usr/local/bin/streamrecorder ausführbar gemacht.

Konfiguration des TV-Browsers

Das Plugin Aufnahmesteuerung muss geladen werden.

Im TV-Browser Menu Extras -> Aufnahmesteuerung -> Geräte -> Gerät-hinzufügen und den Standardtreiber auswählen. Beim Eintrag "Gerät" kann ein beliebiger Name eingetragen werden.

Als "Applikation" wird das Skript (streamrecoder) mit seinem 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 Skript gelistet (Stand Januar 2021). Man kann beliebig viele Sender hinzufügen. Die Namen der Sender sollten im TV-Browser und im Skript miteinander übereinstimmen. Leerstellen im Sendernamen sollten im Skript mit Unterstrich eingetragen werden. Der Speicherort der Aufnahme ist auf ~/Video vor eingestellt und kann im Skript in der Benutzerkonfiguration angepasst werden. Als Quelle für weitere Seder-URLs eignet sich die Projektseite Free-IPTV

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ängigmachen 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> ausprobieren. Für das eigentliche streaming ist ffmpeg verantwortlich.

Viel Spaß damit tuxnix

Das Skript

<nowiki>
  1. !/bin/bash
  1. Script from tuxnix - Version 1.6 - 2023-06-17


  1. FreeAsFreeBeerLicense:
  2. Do not buy a free beer
  3. Do not sell a free beer
  4. Have fun!


          1. Benutzer Konfiguration

recPath=~/Videos/ storePath=~/Videos/Filme

          1. Das Script

declare -a Pakete=("at" "ffmpeg") for i in "${Pakete[@]}"; do

 if ! | -f "/sbin/$i" || -f "/usr/bin/$i" || -f "/usr/sbin/$i"  ; then
   echo "Das Paket $i muss noch installiert werden"
   exit 1
 fi

done


args=("$@") operation=${args[0]} sStunde=${args[1]}; if [ ${#sStunde} -lt 2 ]; then sStunde="0"${args[1]}; fi sMinute=${args[2]}; if [ ${#sMinute} -lt 2 ]; then sMinute="0"$sMinute; fi sJahr=${args[3]} sMonat=${args[4]}; if [ ${#sMonat} -lt 2 ]; then sMonat="0"$sMonat; fi sTag=${args[5]}; if [ ${#sTag} -lt 2 ]; then sTag="0"$sTag; fi eStunde=${args[6]}; if [ ${#eStunde} -lt 2 ]; then eStunde="0"$eStunde; fi eMinute=${args[7]}; if [ ${#eMinute} -lt 2 ]; then eMinute="0"$eMinute; fi eJahr=${args[8]} eMonat=${args[9]}; if [ ${#args[9]} -lt 2 ]; then eMonat="0"${args[9]}; fi eTag=${args[10]}; if [ ${#args[10]} -lt 2 ]; then eTag="0"${args[10]}; fi Kanal=${args[11]} Titel=${args[12]} if [ ${#args[13]} -ge 3 ]; then ProdJahr="-"${args[13]}; fi if [ ${#args[14]} -ge 3 ]; then EpisodenNr="-"${args[14]}; fi if [ ${#args[15]} -ge 3 ]; then Episode="-"${args[15]}; fi

Name="$Titel$ProdJahr$EpisodenNr${Episode}" Name=`echo $Name | sed -e 's![ ]!_!g' -e 's![/}]!-!g' -e 's![.,:()"\x27]!!g' -e 's!&!und!g'` #-e 's!ß!ss!g' -e 's!ö!oe!g'


 if [ $Kanal == 'Das_Erste_(ARD)' ] ; then stream_url="https://mcdn.daserste.de/daserste/de/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="https://mcdn.one.ard.de/ardone/hls/master.m3u8" elif [ $Kanal == 'tagesschau24' ] ; then stream_url="https://tagesschau.akamaized.net/hls/live/2020115/tagesschau/tagesschau_1/master.m3u8" elif [ $Kanal == 'arte' ] ; then stream_url="https://artesimulcast.akamaized.net/hls/live/2030993/artelive_de/index.m3u8" elif [ $Kanal == 'BR_Sued' ] ; then stream_url="https://brcdn.vo.llnwd.net/br/fs/bfs_sued/hls/de/master.m3u8" elif [ $Kanal == 'BR_Nord' ] ; then stream_url="https://mcdn.br.de/br/fs/bfs_nord/hls/de/master.m3u8" elif [ $Kanal == 'KiKA' ] ; then stream_url="https://kikageohls.akamaized.net/hls/live/2022693/livetvkika_de/master.m3u8" elif [ $Kanal == 'HR' ] ; then stream_url="https://hrhls.akamaized.net/hls/live/2024525/hrhls/master.m3u8" elif [ $Kanal == 'MDR_Thueringen' ] ; then stream_url="https://mdrtvthhls.akamaized.net/hls/live/2016880/mdrtvth/master.m3u8" elif [ $Kanal == 'MDR_Sachsen' ] ; then stream_url="https://mdrtvsnhls.akamaized.net/hls/live/2016928/mdrtvsn/master.m3u8" elif [ $Kanal == 'MDR_Sachsen-Anhalt' ] ; then stream_url="https://mdrtvsahls.akamaized.net/hls/live/2016879/mdrtvsa/master.m3u8" elif [ $Kanal == 'NDR_Niedersachsen' ] ; then stream_url="https://mcdn.ndr.de/ndr/hls/ndr_fs/ndr_nds/master.m3u8" elif [ $Kanal == 'NDR_Schleswig-Holstein' ] ; then stream_url="https://mcdn.ndr.de/ndr/hls/ndr_fs/ndr_sh/master.m3u8" elif [ $Kanal == 'NDR_Hamburg' ] ; then stream_url="https://mcdn.ndr.de/ndr/hls/ndr_fs/ndr_hh/master.m3u8" elif [ $Kanal == 'NDR_Mecklenburg-Vorpommern' ] ; then stream_url="https://mcdn.ndr.de/ndr/hls/ndr_fs/ndr_mv/master.m3u8" elif [ $Kanal == 'PHOENIX' ] ; then stream_url="https://zdf-hls-19.akamaized.net/hls/live/2016502/de/veryhigh/master.m3u8" elif [ $Kanal == 'RBB_Berlin' ] ; then stream_url="https://rbb-hls-berlin.akamaized.net/hls/live/2017824/rbb_berlin/master.m3u8" elif [ $Kanal == 'RBB_Brandenburg' ] ; then stream_url="https://rbb-hls-brandenburg.akamaized.net/hls/live/2017825/rbb_brandenburg/master.m3u8" elif [ $Kanal == '3sat' ] ; then stream_url="https://zdf-hls-18.akamaized.net/hls/live/2016501/dach/veryhigh/master.m3u8" elif [ $Kanal == 'SWR_BW' ] ; then stream_url="https://swrbwd-hls.akamaized.net/hls/live/2018672/swrbwd/master.m3u8" elif [ $Kanal == 'SWR_RP' ] ; then stream_url="https://swrrpd-hls.akamaized.net/hls/live/2018676/swrrpd/master.m3u8" elif [ $Kanal == 'WDR' ] ; then stream_url="https://mcdn.wdr.de/wdr/wdrfs/de/master.m3u8" elif [ $Kanal == 'ZDF' ] ; then stream_url="http://zdf-hls-15.akamaized.net/hls/live/2016498/de/veryhigh/master.m3u8" elif [ $Kanal == 'ZDFinfo' ] ; then stream_url="https://zdf-hls-17.akamaized.net/hls/live/2016500/de/veryhigh/master.m3u8" elif [ $Kanal == 'ZDFneo' ] ; then stream_url="https://zdf-hls-16.akamaized.net/hls/live/2016499/de/veryhigh/master.m3u8"

  1. Geoblocking (In der BRD nur mittels VPN zu streamen)

elif [ $Kanal == 'ORF_eins' ] ; then stream_url="https://orf1.mdn.ors.at/out/u/orf1/qxb/manifest.m3u8" elif [ $Kanal == 'ORF_2' ] ; then stream_url="https://orf2e.mdn.ors.at/out/u/orf2e/qxb/manifest.m3u8" elif [ $Kanal == 'ORF_III' ] ; then stream_url="https://orf3.mdn.ors.at/out/u/orf3/qxb/manifest.m3u8" elif [ $Kanal == 'ORF_SPORT_PLUS' ] ; then stream_url="https://orfs.mdn.ors.at/out/u/orfs/qxb/manifest.m3u8"

else

   echo "Abbruch: Der Sender '$Kanal' ist in streamrecorder nicht gelistet."
   echo ""
   echo "Die Namen der Sender im TV-Browser müssen mit denen im Skript übereinstimmen."
   echo "Bitte Groß und Kleinschreibung beachten und Leerzeichen im Sendernamen im TV-Browser vermeiden oder im Skript durch einen Unterstrich ersetzen."
   echo "Umlaute werden nicht unterstützt."
   exit 1

fi

sysTime=`date --date='now' +%s` startTime=`date --date="$sJahr-$sMonat-$sTag $sStunde:$sMinute" +%s` endTime=`date --date="$eJahr-$eMonat-$eTag $eStunde:$eMinute" +%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 -ge $endTime ] ; then

   echo "Keine Aufnahme!"
   echo "Die Sendung liegt in der Vergangenheit."
   exit 1

elif [ $operation == 'record' ] && -f $storePath$Name.mp4 ; then

   echo "Keine Aufnahme!"
   echo "Die Sendung wurde bereits in der Vergangenheit aufgezeichnet:"
   echo $storePath$Name".mp4"
   exit 1

elif [ $operation == 'record' ] && [ $sysTime -ge $startTime ] ; then

   let secsPart=$endTime-$sysTime
   hPart=$((secsPart/3600)); if [ ${#hPart} -lt 2 ]; then hPart="0"$hPart; fi
   mPart=$((secsPart%3600/60)); if [ ${#mPart} -lt 2 ]; then mPart="0"$mPart; fi
   durationPart="$hPart:$mPart:00"
   echo "`date --date='now' +%R` (jetzt) bis $eStunde:$eMinute Uhr, Dauer: $durationPart"
   echo "Aufnahme speichern:"
   echo $recPath$Name"-part.mp4"
   echo "ffmpeg -i $stream_url -c copy -t $durationPart $recPath$Name-part.mp4" | at $startNow > /dev/null 2>&1

elif [ $operation == 'record' ] && [ $sysTime -le $startTime ] ; then

   atJob=(`atq | awk '{print $1}'`)
   let secs=$endTime-$startTime
   H=$((secs/3600)); if [ ${#H} -lt 2 ]; then H="0"$H; fi
   M=$((secs%3600/60)); if [ ${#M} -lt 2 ]; then M="0"$M; fi
   duration="$H:$M:00"
   echo "Tag: $eTag.$eMonat.$eJahr - $sStunde:$sMinute bis $eStunde:$eMinute Uhr, Dauer: $duration"
   echo "Aufnahme speichern:"
   echo $recPath$Name".mp4"
   echo "ffmpeg -i $stream_url -c copy -t $duration $recPath$Name.mp4" | at $startZeit > /dev/null 2>&1

elif [ $operation == 'delete' ] && [ $sysTime -ge $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 '$Name' wird beendet!"

elif [ $operation == 'delete' ] && [ $sysTime -lt $startTime ] ; then

   Monat=([1]=Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec)
   sTag=${args[5]}
   if [ ${#sTag} -lt 2 ] ; then sTag=" "$sTag ; fi
   while  grep -e "${Monat[${args[4]}]} $sTag $sStunde:$sMinute:00 $sJahr"` ; do
       Job=( `atq | grep -e "${Monat[${args[4]}]} $sTag $sStunde:$sMinute:00 $sJahr"`)
       if  grep -e $Name` ; then atrm $Job; echo "Der Job $Job wurde gelöscht!"; fi
   done

else

   echo "Abbruch: Der Operator $operation ist nicht korrekt. Bitte überprüfe die Parameter im TV-Browser Aufnahme Plugin"

fi