Persoenliche Mediazentrale
ulrich
2021-04-10 5f7e0b47ee8d31a87bef7d6fa7268fc8ce839126
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  *
49  * HTTP GET /mz/api/strg/abspieler/pause HTTP GET /mz/api/strg/abspieler/stop HTTP GET
50  * /mz/api/strg/abspieler/weiter
b56bb3 51  * 
0e9cd3 52  * Faustregel: Anzahl Elemente eines URL plus 1 ist die Anzahl der Elemente des 
U 53  * Ergebnisses von String.split.
54  *
55  *
b56bb3 56  * @author Ulrich Hilger
U 57  * @version 1, 9.4.2021
58  */
59 public class MediaSteuerung extends AbstractHandler {
60
61   private static final Logger logger = Logger.getLogger(MediaSteuerung.class.getName());
0e9cd3 62
U 63   public static final String PL_CMD_PLAY = "avd/play";
64   public static final String PL_DEFAULT_PARAMS = "?titel=";
005d7a 65   public static final String PL_PARAM_RUECK = "&r=";
U 66   public static final String PL_API_STRG = "/api/strg/"; 
0e9cd3 67   public static final String PL_CMD_ENDE = "ende";
005d7a 68   public static final String DEFAULT_HOST = "http://localhost:9090";
0e9cd3 69
U 70   private final Map spielt = new HashMap();
b56bb3 71
U 72   @Override
73   protected String get(HttpExchange e) {
0e9cd3 74     String response;
b56bb3 75     String path = e.getRequestURI().toString();
0e9cd3 76     String[] elems = path.split(Server.SLASH);
U 77     FileStorage fs = new FileStorage(App.getInitParameter(App.getRs(App.RB_AP_CONF)));
78     
79     // Faustregel: Anzahl Elemente eines URL plus 1 ist die Anzahl der Elemente des 
80     // Ergebnisses von String.split.
81     switch (elems.length) {
82       case 6:
83         if (elems[5].equalsIgnoreCase(PL_CMD_ENDE)) {
84           response = naechsterTitel(fs, elems[4]);
85         } else {
86           response = meldung("Ungueltiges Kommando: " + elems[5], AbstractHandler.RTC_NOT_FOUND);
87         }
88         break;
b56bb3 89       case 8:
5f7e0b 90         response = ersterTitel(fs, elems[4], elems[7]);
0e9cd3 91         break;
U 92       default:
93         response = "Ungueltiger URL";
b56bb3 94         break;
U 95     }
96     return response;
97   }
98   
5f7e0b 99   private String ersterTitel(Storage s, String aName, String lName) {
U 100     String response;
101     Entity entity = s.read(FileStorage.ST_ABSPIELLISTE, lName);
102     if (entity instanceof Abspielliste) {
103       Abspielliste liste = (Abspielliste) entity;
104       response = listentitelSpielen(s, aName, liste, 0);
105     } else {
106       response = meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
107     }
108     return response;
0e9cd3 109   }
U 110
111   private String naechsterTitel(Storage s, String abspielerName) {
112     String response;
113     Object o = spielt.get(abspielerName);
114     if (o instanceof Abspielvorgang) {
115       Abspielvorgang av = (Abspielvorgang) o;
116       Entity entity = s.read(FileStorage.ST_ABSPIELLISTE, av.getListe());
117       if (entity instanceof Abspielliste) {
b56bb3 118         Abspielliste liste = (Abspielliste) entity;
0e9cd3 119         int titelNr = av.getTitelNr();
U 120         if (liste.getTitel().size() > ++titelNr) {
005d7a 121           response = listentitelSpielen(s, abspielerName, liste, titelNr);
0e9cd3 122         } else {
U 123           response = "Liste " + liste.getName() + " ist zuende gespielt.";
124         }
125       } else {
126         response = meldung("Abspielliste nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
b56bb3 127       }
0e9cd3 128       //response = listenTitelSpielen(e, elems[4]);
U 129     } else {
130       response = meldung("Abspielvorgang fuer Abspieler " + abspielerName, AbstractHandler.RTC_NOT_FOUND);
b56bb3 131     }
U 132     return response;
133   }
0e9cd3 134
005d7a 135   private String listentitelSpielen(Storage s, String aName, Abspielliste liste, int titelNr) {
0e9cd3 136     String response;
U 137     Entity entity = s.read(FileStorage.ST_ABSPIELER, aName);
138     if (entity instanceof Abspieler) {
139       Abspieler abspieler = (Abspieler) entity;
140       String kommando = kommandoFuerTitel(s, liste, abspieler, titelNr);
141       //String kommando = kmd.toString();
142       logger.info(kommando);
5f7e0b 143       abspielerKommandoSenden(kommando);
0e9cd3 144       response = "Abspielen der Liste " + liste.getName() + " auf Abspieler " + aName + " gestartet.";
U 145     } else {
146       response = meldung("Abspieler nicht gefunden.", AbstractHandler.RTC_NOT_FOUND);
147     }
148     return response;
149   }
150
151   /**
152    * Das Kommando zum Abspielen fuer den Titel einer Abspielliste und einen bestimmten Abspieler
153    * ermitteln.
154    *
155    * @param s die Ablage, in der Abspieler und Abspiellisten zu finden sind
156    * @param liste Name der Liste, die den gewuenschten Titel enthaelt
157    * @param abspieler Name des Abspielers, der zum Abspielen dienen soll
158    * @param titelNr Nummer des Titels in der Liste
159    * @return das Kommando zum Abspielen (ein URL)
160    */
161   private String kommandoFuerTitel(Storage s, Abspielliste liste, Abspieler abspieler, int titelNr) {
162     // ersten Titel aus Liste holen
163     Titel titel = liste.getTitel().get(titelNr);
164
165     // URL des Titels ermitteln
166     String titelUrl = titel.getKatalogUrl() + titel.getPfad() + titel.getName();
167     logger.log(Level.INFO, "abspielen von {0} auf {1}", new Object[]{titelUrl, abspieler.getUrl()});
168
169     // Titel als 'spielt' vermerken
170     Abspielvorgang vorgang = new Abspielvorgang();
171     vorgang.setAbspieler(abspieler.getName());
172     vorgang.setListe(liste.getName());
173     vorgang.setTitelNr(titelNr);
174     spielt.put(abspieler.getName(), vorgang);
005d7a 175     
U 176     String server = getEinstellung(s, App.getRs(App.RB_HOST), DEFAULT_HOST);
0e9cd3 177
U 178     // Kommando an den Abspieler zusammenbauen
179     StringBuilder kmd = new StringBuilder();
180     kmd.append(abspieler.getUrl());
181     kmd.append(PL_CMD_PLAY);
182     // Parameter fuer den Abspieler holen
005d7a 183     kmd.append(getEinstellung(s, App.getRs(App.RB_PLAYERPARAMS), PL_DEFAULT_PARAMS));
U 184     kmd.append(server);
0e9cd3 185     kmd.append(titelUrl);
005d7a 186     kmd.append(PL_PARAM_RUECK);
U 187     kmd.append(server);
188     kmd.append(PL_API_STRG);
189     kmd.append(abspieler.getName());
190     kmd.append("/ende");
0e9cd3 191
U 192     return kmd.toString();
193   }
005d7a 194   
U 195   private String getEinstellung(Storage s, String key, String standardWert) {
196     Entity entity = s.read(Einstellung.class.getSimpleName(), key);
197     if (entity instanceof Einstellung) {
198       Einstellung einstellung = (Einstellung) entity;
199       Object o = einstellung.getValue();
200       if(o instanceof String) {
201         return o.toString();
202       } else {
203         return standardWert;
204       }
205     } else {
206       return standardWert;
207     }
208   }
209   
210   private void abspielerKommandoSenden(String kommando) {
211     /*
212       TODO hier evtl. mit mehreren Versuchen ausgleichen, 
213       dass ein einzelner Versuch nicht 'durchkommt'...
214     */
215     logger.info(kommando);
216     try {
217       HttpURLConnection conn = (HttpURLConnection) new URL(kommando).openConnection();
218       conn.setRequestMethod("GET");
219       conn.connect();
220       int status = conn.getResponseCode();
221       String msg = conn.getResponseMessage();
222       logger.log(Level.INFO, "Kommando {0} mit Status {1} {2} gesendet.", new Object[]{kommando, status, msg});
223     } catch(IOException ex) {
224       logger.log(Level.INFO, ex.getMessage(), ex);
225     }
226   }
227   
5f7e0b 228   private String meldung(String text, int code) {
U 229     setReturnCode(code);
230     return text;
231   }
232   
b56bb3 233   // rpi4-az:9090/avd/play?titel=/Filme/S/sound_city.m4v&th=60&ti=60&o=local
U 234   // aUrl http://rpi4-wz:9090/
235   // titelUrl /media/test/A/The-Alan-Parsons-Project/I-Robot/02-I-Wouldnt-Want-to-Be-Like-You.mp3
236 }