Persoenliche Mediazentrale
ulrich
2021-04-10 9865bd176bde875b1f1eaf4d4319c5c3b298ff69
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)) {
U 94           response = kommandoSenden(fs, elems[4], PL_CMD_AVD_STOP);
95         } else if(elems[5].equalsIgnoreCase(PL_CMD_PAUSE)) {
96           response = kommandoSenden(fs, elems[4], PL_CMD_AVD_PAUSE);
97         } else if(elems[5].equalsIgnoreCase(PL_CMD_PLAYON)) {
98           response = kommandoSenden(fs, elems[4], PL_CMD_AVD_PLAYON);
0e9cd3 99         } else {
U 100           response = meldung("Ungueltiges Kommando: " + elems[5], AbstractHandler.RTC_NOT_FOUND);
101         }
102         break;
b56bb3 103       case 8:
5f7e0b 104         response = ersterTitel(fs, elems[4], elems[7]);
0e9cd3 105         break;
U 106       default:
107         response = "Ungueltiger URL";
b56bb3 108         break;
U 109     }
110     return response;
589850 111   }
U 112   
113   private String kommandoSenden(Storage s, String aName, String kommando) {
114     Entity entity = s.read(FileStorage.ST_ABSPIELER, aName);
115     if (entity instanceof Abspieler) {
116       Abspieler abspieler = (Abspieler) entity;
117       StringBuilder kmd = new StringBuilder();
118       kmd.append(abspieler.getUrl());
119       kmd.append(kommando);
120       return kommando + " gesendet.";
121     } else {
122       return meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
123     }
b56bb3 124   }
U 125   
5f7e0b 126   private String ersterTitel(Storage s, String aName, String lName) {
U 127     String response;
128     Entity entity = s.read(FileStorage.ST_ABSPIELLISTE, lName);
129     if (entity instanceof Abspielliste) {
130       Abspielliste liste = (Abspielliste) entity;
131       response = listentitelSpielen(s, aName, liste, 0);
132     } else {
133       response = meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
134     }
135     return response;
0e9cd3 136   }
U 137
138   private String naechsterTitel(Storage s, String abspielerName) {
139     String response;
140     Object o = spielt.get(abspielerName);
141     if (o instanceof Abspielvorgang) {
142       Abspielvorgang av = (Abspielvorgang) o;
143       Entity entity = s.read(FileStorage.ST_ABSPIELLISTE, av.getListe());
144       if (entity instanceof Abspielliste) {
b56bb3 145         Abspielliste liste = (Abspielliste) entity;
0e9cd3 146         int titelNr = av.getTitelNr();
U 147         if (liste.getTitel().size() > ++titelNr) {
005d7a 148           response = listentitelSpielen(s, abspielerName, liste, titelNr);
0e9cd3 149         } else {
U 150           response = "Liste " + liste.getName() + " ist zuende gespielt.";
589850 151           logger.info(response);
0e9cd3 152         }
U 153       } else {
154         response = meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
b56bb3 155       }
0e9cd3 156       //response = listenTitelSpielen(e, elems[4]);
U 157     } else {
158       response = meldung("Abspielvorgang fuer Abspieler " + abspielerName, AbstractHandler.RTC_NOT_FOUND);
b56bb3 159     }
U 160     return response;
161   }
0e9cd3 162
005d7a 163   private String listentitelSpielen(Storage s, String aName, Abspielliste liste, int titelNr) {
0e9cd3 164     String response;
U 165     Entity entity = s.read(FileStorage.ST_ABSPIELER, aName);
166     if (entity instanceof Abspieler) {
167       Abspieler abspieler = (Abspieler) entity;
168       String kommando = kommandoFuerTitel(s, liste, abspieler, titelNr);
169       //String kommando = kmd.toString();
170       logger.info(kommando);
5f7e0b 171       abspielerKommandoSenden(kommando);
0e9cd3 172       response = "Abspielen der Liste " + liste.getName() + " auf Abspieler " + aName + " gestartet.";
U 173     } else {
174       response = meldung("Abspieler nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
175     }
176     return response;
177   }
178
179   /**
180    * Das Kommando zum Abspielen fuer den Titel einer Abspielliste und einen bestimmten Abspieler
181    * ermitteln.
182    *
183    * @param s die Ablage, in der Abspieler und Abspiellisten zu finden sind
184    * @param liste Name der Liste, die den gewuenschten Titel enthaelt
185    * @param abspieler Name des Abspielers, der zum Abspielen dienen soll
186    * @param titelNr Nummer des Titels in der Liste
187    * @return das Kommando zum Abspielen (ein URL)
188    */
189   private String kommandoFuerTitel(Storage s, Abspielliste liste, Abspieler abspieler, int titelNr) {
190     // ersten Titel aus Liste holen
191     Titel titel = liste.getTitel().get(titelNr);
192
193     // URL des Titels ermitteln
194     String titelUrl = titel.getKatalogUrl() + titel.getPfad() + titel.getName();
195     logger.log(Level.INFO, "abspielen von {0} auf {1}", new Object[]{titelUrl, abspieler.getUrl()});
196
197     // Titel als 'spielt' vermerken
198     Abspielvorgang vorgang = new Abspielvorgang();
199     vorgang.setAbspieler(abspieler.getName());
200     vorgang.setListe(liste.getName());
201     vorgang.setTitelNr(titelNr);
202     spielt.put(abspieler.getName(), vorgang);
005d7a 203     
U 204     String server = getEinstellung(s, App.getRs(App.RB_HOST), DEFAULT_HOST);
0e9cd3 205
U 206     // Kommando an den Abspieler zusammenbauen
207     StringBuilder kmd = new StringBuilder();
208     kmd.append(abspieler.getUrl());
209     kmd.append(PL_CMD_PLAY);
210     // Parameter fuer den Abspieler holen
005d7a 211     kmd.append(getEinstellung(s, App.getRs(App.RB_PLAYERPARAMS), PL_DEFAULT_PARAMS));
U 212     kmd.append(server);
0e9cd3 213     kmd.append(titelUrl);
005d7a 214     kmd.append(PL_PARAM_RUECK);
U 215     kmd.append(server);
216     kmd.append(PL_API_STRG);
217     kmd.append(abspieler.getName());
218     kmd.append("/ende");
0e9cd3 219
U 220     return kmd.toString();
221   }
005d7a 222   
U 223   private String getEinstellung(Storage s, String key, String standardWert) {
224     Entity entity = s.read(Einstellung.class.getSimpleName(), key);
225     if (entity instanceof Einstellung) {
226       Einstellung einstellung = (Einstellung) entity;
227       Object o = einstellung.getValue();
228       if(o instanceof String) {
229         return o.toString();
230       } else {
231         return standardWert;
232       }
233     } else {
234       return standardWert;
235     }
236   }
237   
238   private void abspielerKommandoSenden(String kommando) {
239     /*
240       TODO hier evtl. mit mehreren Versuchen ausgleichen, 
241       dass ein einzelner Versuch nicht 'durchkommt'...
242     */
243     logger.info(kommando);
244     try {
245       HttpURLConnection conn = (HttpURLConnection) new URL(kommando).openConnection();
246       conn.setRequestMethod("GET");
247       conn.connect();
248       int status = conn.getResponseCode();
249       String msg = conn.getResponseMessage();
250       logger.log(Level.INFO, "Kommando {0} mit Status {1} {2} gesendet.", new Object[]{kommando, status, msg});
251     } catch(IOException ex) {
252       logger.log(Level.INFO, ex.getMessage(), ex);
253     }
254   }
255   
5f7e0b 256   private String meldung(String text, int code) {
U 257     setReturnCode(code);
258     return text;
259   }
260   
b56bb3 261   // rpi4-az:9090/avd/play?titel=/Filme/S/sound_city.m4v&th=60&ti=60&o=local
U 262   // aUrl http://rpi4-wz:9090/
263   // titelUrl /media/test/A/The-Alan-Parsons-Project/I-Robot/02-I-Wouldnt-Want-to-Be-Like-You.mp3
264 }