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