Persoenliche Mediazentrale
ulrich
2021-04-11 630b44fd74991e2757b2870eb76abbf3ad0f616f
commit | author | age
b56bb3 1 /*
U 2   Mediazentrale - Personal Media Center
3   Copyright (C) 2021  Ulrich Hilger
4
5   This program is free software: you can redistribute it and/or modify
6   it under the terms of the GNU Affero General Public License as
7   published by the Free Software Foundation, either version 3 of the
8   License, or (at your option) any later version.
9
10   This program is distributed in the hope that it will be useful,
11   but WITHOUT ANY WARRANTY; without even the implied warranty of
12   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13   GNU Affero General Public License for more details.
14
15   You should have received a copy of the GNU Affero General Public License
16   along with this program.  If not, see <https://www.gnu.org/licenses/>.
17  */
18 package de.uhilger.mediaz.api;
19
20 import com.sun.net.httpserver.HttpExchange;
21 import de.uhilger.mediaz.App;
22 import de.uhilger.mediaz.Server;
0e9cd3 23 import de.uhilger.mediaz.entity.Abspielvorgang;
b56bb3 24 import de.uhilger.mediaz.entity.Abspieler;
U 25 import de.uhilger.mediaz.entity.Abspielliste;
0e9cd3 26 import de.uhilger.mediaz.entity.Einstellung;
b56bb3 27 import de.uhilger.mediaz.entity.Entity;
U 28 import de.uhilger.mediaz.entity.Titel;
29 import de.uhilger.mediaz.store.FileStorage;
0e9cd3 30 import de.uhilger.mediaz.store.Storage;
005d7a 31 import java.io.IOException;
U 32 import java.net.HttpURLConnection;
33 import java.net.URL;
b56bb3 34 import java.util.HashMap;
U 35 import java.util.Map;
0e9cd3 36 import java.util.logging.Level;
b56bb3 37 import java.util.logging.Logger;
U 38
39 /**
0e9cd3 40  * Die MediaSteuerung verarbeitet HTTP-Signale zur Steuerung von Media-Operationen wie z.B. dem
U 41  * Spielen einer Abspielliste oder dem Starten oder Stoppen eines Videos auf einem entfernten
42  * Abspielgeraet.
43  *
44  * HTTP GET /mz/api/strg/abspieler/play/liste/[name] 
45  * HTTP GET /mz/api/strg/abspieler/ende
46  *
47  * HTTP GET /mz/api/strg/abspieler/play/[url]
48  *
589850 49  * HTTP GET /mz/api/strg/abspieler/pause 
U 50  * HTTP GET /mz/api/strg/abspieler/stop 
51  * HTTP GET /mz/api/strg/abspieler/weiter
b56bb3 52  * 
0e9cd3 53  * Faustregel: Anzahl Elemente eines URL plus 1 ist die Anzahl der Elemente des 
U 54  * Ergebnisses von String.split.
55  *
56  *
b56bb3 57  * @author Ulrich Hilger
U 58  * @version 1, 9.4.2021
59  */
60 public class MediaSteuerung extends AbstractHandler {
61
62   private static final Logger logger = Logger.getLogger(MediaSteuerung.class.getName());
0e9cd3 63
U 64   public static final String PL_CMD_PLAY = "avd/play";
65   public static final String PL_DEFAULT_PARAMS = "?titel=";
005d7a 66   public static final String PL_PARAM_RUECK = "&r=";
U 67   public static final String PL_API_STRG = "/api/strg/"; 
0e9cd3 68   public static final String PL_CMD_ENDE = "ende";
589850 69   public static final String PL_CMD_STOP = "stop";
U 70   public static final String PL_CMD_PAUSE = "pause";
71   public static final String PL_CMD_PLAYON = "playon";
72   public static final String PL_CMD_AVD_STOP = "avd/stop";
73   public static final String PL_CMD_AVD_PAUSE = "avd/pause";
74   public static final String PL_CMD_AVD_PLAYON = "avd/playon";
005d7a 75   public static final String DEFAULT_HOST = "http://localhost:9090";
0e9cd3 76
U 77   private final Map spielt = new HashMap();
b56bb3 78
U 79   @Override
80   protected String get(HttpExchange e) {
0e9cd3 81     String response;
b56bb3 82     String path = e.getRequestURI().toString();
0e9cd3 83     String[] elems = path.split(Server.SLASH);
U 84     FileStorage fs = new FileStorage(App.getInitParameter(App.getRs(App.RB_AP_CONF)));
589850 85     logger.fine(path);
0e9cd3 86     
U 87     // Faustregel: Anzahl Elemente eines URL plus 1 ist die Anzahl der Elemente des 
88     // Ergebnisses von String.split.
89     switch (elems.length) {
90       case 6:
91         if (elems[5].equalsIgnoreCase(PL_CMD_ENDE)) {
92           response = naechsterTitel(fs, elems[4]);
589850 93         } else if(elems[5].equalsIgnoreCase(PL_CMD_STOP)) {
d12f6e 94           spielt.remove(elems[4]);
589850 95           response = kommandoSenden(fs, elems[4], PL_CMD_AVD_STOP);
U 96         } else if(elems[5].equalsIgnoreCase(PL_CMD_PAUSE)) {
97           response = kommandoSenden(fs, elems[4], PL_CMD_AVD_PAUSE);
98         } else if(elems[5].equalsIgnoreCase(PL_CMD_PLAYON)) {
99           response = kommandoSenden(fs, elems[4], PL_CMD_AVD_PLAYON);
0e9cd3 100         } else {
U 101           response = meldung("Ungueltiges Kommando: " + elems[5], AbstractHandler.RTC_NOT_FOUND);
102         }
103         break;
b56bb3 104       case 8:
5f7e0b 105         response = ersterTitel(fs, elems[4], elems[7]);
0e9cd3 106         break;
U 107       default:
108         response = "Ungueltiger URL";
b56bb3 109         break;
U 110     }
111     return response;
589850 112   }
U 113   
114   private String kommandoSenden(Storage s, String aName, String kommando) {
115     Entity entity = s.read(FileStorage.ST_ABSPIELER, aName);
116     if (entity instanceof Abspieler) {
117       Abspieler abspieler = (Abspieler) entity;
118       StringBuilder kmd = new StringBuilder();
119       kmd.append(abspieler.getUrl());
120       kmd.append(kommando);
9e14ef 121       String signal = kmd.toString();
U 122       abspielerKommandoSenden(signal);
123       return signal + " gesendet.";
589850 124     } else {
U 125       return meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
126     }
b56bb3 127   }
U 128   
5f7e0b 129   private String ersterTitel(Storage s, String aName, String lName) {
U 130     String response;
131     Entity entity = s.read(FileStorage.ST_ABSPIELLISTE, lName);
132     if (entity instanceof Abspielliste) {
133       Abspielliste liste = (Abspielliste) entity;
134       response = listentitelSpielen(s, aName, liste, 0);
135     } else {
136       response = meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
137     }
138     return response;
0e9cd3 139   }
U 140
141   private String naechsterTitel(Storage s, String abspielerName) {
142     String response;
143     Object o = spielt.get(abspielerName);
144     if (o instanceof Abspielvorgang) {
145       Abspielvorgang av = (Abspielvorgang) o;
146       Entity entity = s.read(FileStorage.ST_ABSPIELLISTE, av.getListe());
147       if (entity instanceof Abspielliste) {
b56bb3 148         Abspielliste liste = (Abspielliste) entity;
0e9cd3 149         int titelNr = av.getTitelNr();
U 150         if (liste.getTitel().size() > ++titelNr) {
005d7a 151           response = listentitelSpielen(s, abspielerName, liste, titelNr);
0e9cd3 152         } else {
U 153           response = "Liste " + liste.getName() + " ist zuende gespielt.";
589850 154           logger.info(response);
0e9cd3 155         }
U 156       } else {
157         response = meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
b56bb3 158       }
0e9cd3 159       //response = listenTitelSpielen(e, elems[4]);
U 160     } else {
161       response = meldung("Abspielvorgang fuer Abspieler " + abspielerName, AbstractHandler.RTC_NOT_FOUND);
b56bb3 162     }
U 163     return response;
164   }
0e9cd3 165
005d7a 166   private String listentitelSpielen(Storage s, String aName, Abspielliste liste, int titelNr) {
0e9cd3 167     String response;
U 168     Entity entity = s.read(FileStorage.ST_ABSPIELER, aName);
169     if (entity instanceof Abspieler) {
170       Abspieler abspieler = (Abspieler) entity;
171       String kommando = kommandoFuerTitel(s, liste, abspieler, titelNr);
172       //String kommando = kmd.toString();
173       logger.info(kommando);
5f7e0b 174       abspielerKommandoSenden(kommando);
0e9cd3 175       response = "Abspielen der Liste " + liste.getName() + " auf Abspieler " + aName + " gestartet.";
U 176     } else {
177       response = meldung("Abspieler nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
178     }
179     return response;
180   }
181
182   /**
183    * Das Kommando zum Abspielen fuer den Titel einer Abspielliste und einen bestimmten Abspieler
184    * ermitteln.
185    *
186    * @param s die Ablage, in der Abspieler und Abspiellisten zu finden sind
187    * @param liste Name der Liste, die den gewuenschten Titel enthaelt
188    * @param abspieler Name des Abspielers, der zum Abspielen dienen soll
189    * @param titelNr Nummer des Titels in der Liste
190    * @return das Kommando zum Abspielen (ein URL)
191    */
192   private String kommandoFuerTitel(Storage s, Abspielliste liste, Abspieler abspieler, int titelNr) {
193     // ersten Titel aus Liste holen
194     Titel titel = liste.getTitel().get(titelNr);
195
196     // URL des Titels ermitteln
197     String titelUrl = titel.getKatalogUrl() + titel.getPfad() + titel.getName();
198     logger.log(Level.INFO, "abspielen von {0} auf {1}", new Object[]{titelUrl, abspieler.getUrl()});
199
200     // Titel als 'spielt' vermerken
201     Abspielvorgang vorgang = new Abspielvorgang();
202     vorgang.setAbspieler(abspieler.getName());
203     vorgang.setListe(liste.getName());
204     vorgang.setTitelNr(titelNr);
205     spielt.put(abspieler.getName(), vorgang);
005d7a 206     
U 207     String server = getEinstellung(s, App.getRs(App.RB_HOST), DEFAULT_HOST);
0e9cd3 208
U 209     // Kommando an den Abspieler zusammenbauen
210     StringBuilder kmd = new StringBuilder();
211     kmd.append(abspieler.getUrl());
212     kmd.append(PL_CMD_PLAY);
213     // Parameter fuer den Abspieler holen
005d7a 214     kmd.append(getEinstellung(s, App.getRs(App.RB_PLAYERPARAMS), PL_DEFAULT_PARAMS));
U 215     kmd.append(server);
0e9cd3 216     kmd.append(titelUrl);
005d7a 217     kmd.append(PL_PARAM_RUECK);
U 218     kmd.append(server);
219     kmd.append(PL_API_STRG);
220     kmd.append(abspieler.getName());
221     kmd.append("/ende");
0e9cd3 222
U 223     return kmd.toString();
224   }
005d7a 225   
U 226   private String getEinstellung(Storage s, String key, String standardWert) {
227     Entity entity = s.read(Einstellung.class.getSimpleName(), key);
228     if (entity instanceof Einstellung) {
229       Einstellung einstellung = (Einstellung) entity;
230       Object o = einstellung.getValue();
231       if(o instanceof String) {
232         return o.toString();
233       } else {
234         return standardWert;
235       }
236     } else {
237       return standardWert;
238     }
239   }
240   
241   private void abspielerKommandoSenden(String kommando) {
242     /*
243       TODO hier evtl. mit mehreren Versuchen ausgleichen, 
244       dass ein einzelner Versuch nicht 'durchkommt'...
245     */
246     logger.info(kommando);
247     try {
248       HttpURLConnection conn = (HttpURLConnection) new URL(kommando).openConnection();
249       conn.setRequestMethod("GET");
250       conn.connect();
251       int status = conn.getResponseCode();
252       String msg = conn.getResponseMessage();
253       logger.log(Level.INFO, "Kommando {0} mit Status {1} {2} gesendet.", new Object[]{kommando, status, msg});
254     } catch(IOException ex) {
255       logger.log(Level.INFO, ex.getMessage(), ex);
256     }
257   }
258   
5f7e0b 259   private String meldung(String text, int code) {
U 260     setReturnCode(code);
261     return text;
262   }
263   
b56bb3 264   // rpi4-az:9090/avd/play?titel=/Filme/S/sound_city.m4v&th=60&ti=60&o=local
U 265   // aUrl http://rpi4-wz:9090/
266   // titelUrl /media/test/A/The-Alan-Parsons-Project/I-Robot/02-I-Wouldnt-Want-to-Be-Like-You.mp3
267 }