2004-12-30 23:43:55 +01:00
|
|
|
/*
|
2010-12-02 09:57:17 +01:00
|
|
|
* $Id: connectionHTTP.c,v 1.21 2010/08/03 10:46:41 schmirl Exp $
|
2004-12-30 23:43:55 +01:00
|
|
|
*/
|
2005-04-24 18:26:14 +02:00
|
|
|
|
|
|
|
#include <ctype.h>
|
2012-11-16 02:00:09 +01:00
|
|
|
#include <time.h>
|
|
|
|
#include <stdarg.h>
|
2013-02-02 22:34:47 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2012-12-16 13:29:15 +01:00
|
|
|
#include <vdr/thread.h>
|
|
|
|
#include <vdr/recording.h>
|
2004-12-30 23:43:55 +01:00
|
|
|
|
|
|
|
#include "server/connectionHTTP.h"
|
2008-03-28 16:11:40 +01:00
|
|
|
#include "server/menuHTTP.h"
|
2008-10-14 13:05:46 +02:00
|
|
|
#include "server/server.h"
|
2004-12-30 23:43:55 +01:00
|
|
|
#include "server/setup.h"
|
|
|
|
|
2005-02-11 17:44:14 +01:00
|
|
|
cConnectionHTTP::cConnectionHTTP(void):
|
|
|
|
cServerConnection("HTTP"),
|
|
|
|
m_Status(hsRequest),
|
|
|
|
m_StreamType((eStreamType)StreamdevServerSetup.HTTPStreamType),
|
2012-12-16 13:29:15 +01:00
|
|
|
m_Channel(NULL),
|
2013-10-01 23:47:25 +02:00
|
|
|
m_RecPlayer(NULL),
|
|
|
|
m_ReplayPos(0),
|
|
|
|
m_ReplayFakeRange(false),
|
2013-02-03 11:02:25 +01:00
|
|
|
m_MenuList(NULL)
|
2005-02-11 17:44:14 +01:00
|
|
|
{
|
|
|
|
Dprintf("constructor hsRequest\n");
|
2010-07-20 14:26:29 +02:00
|
|
|
m_Apid[0] = m_Apid[1] = 0;
|
|
|
|
m_Dpid[0] = m_Dpid[1] = 0;
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
|
|
|
|
2005-02-11 17:44:14 +01:00
|
|
|
cConnectionHTTP::~cConnectionHTTP()
|
|
|
|
{
|
2014-08-30 23:58:11 +02:00
|
|
|
SetStreamer(NULL);
|
2013-10-01 23:47:25 +02:00
|
|
|
delete m_RecPlayer;
|
2013-10-20 17:40:22 +02:00
|
|
|
delete m_MenuList;
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
|
|
|
|
2008-10-14 13:05:46 +02:00
|
|
|
bool cConnectionHTTP::CanAuthenticate(void)
|
|
|
|
{
|
|
|
|
return opt_auth != NULL;
|
|
|
|
}
|
|
|
|
|
2005-02-11 17:44:14 +01:00
|
|
|
bool cConnectionHTTP::Command(char *Cmd)
|
|
|
|
{
|
|
|
|
Dprintf("command %s\n", Cmd);
|
2004-12-30 23:43:55 +01:00
|
|
|
switch (m_Status) {
|
|
|
|
case hsRequest:
|
2010-07-20 14:26:29 +02:00
|
|
|
// parse METHOD PATH[?QUERY] VERSION
|
|
|
|
{
|
|
|
|
char *p, *q, *v;
|
|
|
|
p = strchr(Cmd, ' ');
|
|
|
|
if (p) {
|
|
|
|
*p = 0;
|
|
|
|
v = strchr(++p, ' ');
|
|
|
|
if (v) {
|
|
|
|
*v = 0;
|
|
|
|
SetHeader("REQUEST_METHOD", Cmd);
|
|
|
|
q = strchr(p, '?');
|
2012-11-16 02:00:09 +01:00
|
|
|
if (q) {
|
2010-07-20 14:26:29 +02:00
|
|
|
*q = 0;
|
2012-11-16 02:00:09 +01:00
|
|
|
SetHeader("QUERY_STRING", q + 1);
|
|
|
|
while (q++) {
|
|
|
|
char *n = strchr(q, '&');
|
|
|
|
if (n)
|
|
|
|
*n = 0;
|
|
|
|
char *e = strchr(q, '=');
|
|
|
|
if (e)
|
|
|
|
*e++ = 0;
|
|
|
|
else
|
|
|
|
e = n ? n : v;
|
|
|
|
m_Params.insert(tStrStr(q, e));
|
|
|
|
q = n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
SetHeader("QUERY_STRING", "");
|
2010-07-20 14:26:29 +02:00
|
|
|
SetHeader("PATH_INFO", p);
|
|
|
|
m_Status = hsHeaders;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2004-12-30 23:43:55 +01:00
|
|
|
|
|
|
|
case hsHeaders:
|
|
|
|
if (*Cmd == '\0') {
|
2005-02-11 17:44:14 +01:00
|
|
|
m_Status = hsBody;
|
|
|
|
return ProcessRequest();
|
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
else if (isspace(*Cmd)) {
|
|
|
|
; //TODO: multi-line header
|
2008-10-14 13:05:46 +02:00
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
else {
|
|
|
|
// convert header name to CGI conventions:
|
|
|
|
// uppercase, '-' replaced with '_', prefix "HTTP_"
|
|
|
|
char *p;
|
|
|
|
for (p = Cmd; *p != 0 && *p != ':'; p++) {
|
|
|
|
if (*p == '-')
|
|
|
|
*p = '_';
|
|
|
|
else
|
|
|
|
*p = toupper(*p);
|
|
|
|
}
|
|
|
|
if (*p == ':') {
|
|
|
|
*p = 0;
|
|
|
|
p = skipspace(++p);
|
|
|
|
// don't disclose Authorization header
|
|
|
|
if (strcmp(Cmd, "AUTHORIZATION") == 0) {
|
|
|
|
char *q;
|
|
|
|
for (q = p; *q != 0 && *q != ' '; q++)
|
|
|
|
*q = toupper(*q);
|
|
|
|
if (p != q) {
|
|
|
|
*q = 0;
|
|
|
|
SetHeader("AUTH_TYPE", p);
|
|
|
|
m_Authorization = (std::string) skipspace(++q);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
SetHeader(Cmd, p, "HTTP_");
|
2008-10-14 13:05:46 +02:00
|
|
|
}
|
2008-03-28 16:11:40 +01:00
|
|
|
}
|
2005-02-11 17:44:14 +01:00
|
|
|
return true;
|
2007-04-16 13:01:02 +02:00
|
|
|
default:
|
2009-02-13 08:02:18 +01:00
|
|
|
// skip additional blank lines
|
|
|
|
if (*Cmd == '\0')
|
|
|
|
return true;
|
2007-04-16 13:01:02 +02:00
|
|
|
break;
|
2005-02-11 17:44:14 +01:00
|
|
|
}
|
|
|
|
return false; // ??? shouldn't happen
|
|
|
|
}
|
|
|
|
|
2005-05-09 22:22:29 +02:00
|
|
|
bool cConnectionHTTP::ProcessRequest(void)
|
|
|
|
{
|
2010-07-20 14:26:29 +02:00
|
|
|
// keys for Headers() hash
|
|
|
|
const static std::string AUTH_TYPE("AUTH_TYPE");
|
|
|
|
const static std::string REQUEST_METHOD("REQUEST_METHOD");
|
|
|
|
const static std::string PATH_INFO("PATH_INFO");
|
|
|
|
|
2005-02-11 17:44:14 +01:00
|
|
|
Dprintf("process\n");
|
2010-07-20 14:26:29 +02:00
|
|
|
if (!StreamdevHosts.Acceptable(RemoteIpAddr())) {
|
|
|
|
bool authOk = opt_auth && !m_Authorization.empty();
|
|
|
|
if (authOk) {
|
|
|
|
tStrStrMap::const_iterator it = Headers().find(AUTH_TYPE);
|
|
|
|
|
|
|
|
if (it == Headers().end()) {
|
|
|
|
// no authorization header present
|
|
|
|
authOk = false;
|
|
|
|
}
|
|
|
|
else if (it->second.compare("BASIC") == 0) {
|
|
|
|
// basic auth
|
|
|
|
authOk &= m_Authorization.compare(opt_auth) == 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// unsupported auth type
|
|
|
|
authOk = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!authOk) {
|
2008-10-14 13:05:46 +02:00
|
|
|
isyslog("streamdev-server: HTTP authorization required");
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(401, true, NULL, "WWW-authenticate: basic Realm=\"Streamdev-Server\"");
|
2008-10-14 13:05:46 +02:00
|
|
|
}
|
|
|
|
}
|
2005-02-11 17:44:14 +01:00
|
|
|
|
2014-08-30 23:58:11 +02:00
|
|
|
tStrStrMap::const_iterator it;
|
|
|
|
it = m_Params.find("apid");
|
|
|
|
if (it != m_Params.end())
|
|
|
|
m_Apid[0] = atoi(it->second.c_str());
|
|
|
|
it = m_Params.find("dpid");
|
|
|
|
if (it != m_Params.end())
|
|
|
|
m_Dpid[0] = atoi(it->second.c_str());
|
|
|
|
|
2010-07-22 16:18:17 +02:00
|
|
|
tStrStrMap::const_iterator it_method = Headers().find(REQUEST_METHOD);
|
|
|
|
tStrStrMap::const_iterator it_pathinfo = Headers().find(PATH_INFO);
|
|
|
|
if (it_method == Headers().end() || it_pathinfo == Headers().end()) {
|
|
|
|
// should never happen
|
|
|
|
esyslog("streamdev-server connectionHTTP: Missing method or pathinfo");
|
|
|
|
} else if (it_method->second.compare("GET") == 0 && ProcessURI(it_pathinfo->second)) {
|
2013-02-03 11:02:25 +01:00
|
|
|
if (m_MenuList)
|
|
|
|
return Respond("%s", true, m_MenuList->HttpHeader().c_str());
|
2010-07-20 14:26:29 +02:00
|
|
|
else if (m_Channel != NULL) {
|
2014-06-07 00:24:27 +02:00
|
|
|
if (cStreamdevLiveStreamer::ProvidesChannel(m_Channel, StreamdevServerSetup.HTTPPriority)) {
|
|
|
|
cStreamdevLiveStreamer* liveStreamer = new cStreamdevLiveStreamer(this, m_Channel, StreamdevServerSetup.HTTPPriority, m_StreamType, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
|
|
|
if (liveStreamer->GetDevice()) {
|
|
|
|
SetStreamer(liveStreamer);
|
2007-05-09 11:12:42 +02:00
|
|
|
if (!SetDSCP())
|
|
|
|
LOG_ERROR_STR("unable to set DSCP sockopt");
|
2010-07-20 14:26:29 +02:00
|
|
|
if (m_StreamType == stEXT) {
|
|
|
|
return Respond("HTTP/1.0 200 OK");
|
2011-10-19 23:21:38 +02:00
|
|
|
} else if (m_StreamType == stES && (m_Apid[0] || m_Dpid[0] || ISRADIO(m_Channel))) {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(200, false, "audio/mpeg", "icy-name: %s", m_Channel->Name());
|
2011-10-19 23:21:38 +02:00
|
|
|
} else if (ISRADIO(m_Channel)) {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(200, false, "audio/mpeg");
|
2004-12-30 23:43:55 +01:00
|
|
|
} else {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(200, false, "video/mpeg");
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
|
|
|
}
|
2014-05-18 18:16:51 +02:00
|
|
|
SetStreamer(NULL);
|
2014-06-07 00:24:27 +02:00
|
|
|
delete liveStreamer;
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(503, true);
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
2013-10-01 23:47:25 +02:00
|
|
|
else if (m_RecPlayer != NULL) {
|
2012-12-16 13:29:15 +01:00
|
|
|
Dprintf("GET recording\n");
|
2014-08-30 23:58:11 +02:00
|
|
|
bool isPes = m_RecPlayer->getCurrentRecording()->IsPesRecording();
|
|
|
|
// no remuxing for old PES recordings
|
|
|
|
if (isPes && m_StreamType != stPES)
|
|
|
|
return HttpResponse(503, true);
|
|
|
|
|
2012-12-16 13:29:15 +01:00
|
|
|
int64_t from, to;
|
2013-10-01 23:47:25 +02:00
|
|
|
bool hasRange = ParseRange(from, to);
|
|
|
|
|
|
|
|
cStreamdevRecStreamer* recStreamer;
|
|
|
|
if (from == 0 && hasRange && m_ReplayFakeRange) {
|
2014-08-30 23:58:11 +02:00
|
|
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, (int64_t) 0L, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
2013-10-01 23:47:25 +02:00
|
|
|
from += m_ReplayPos;
|
|
|
|
if (to >= 0)
|
|
|
|
to += m_ReplayPos;
|
|
|
|
}
|
|
|
|
else
|
2014-08-30 23:58:11 +02:00
|
|
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, m_ReplayPos, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
2014-05-18 18:16:51 +02:00
|
|
|
SetStreamer(recStreamer);
|
2014-08-30 23:58:11 +02:00
|
|
|
|
|
|
|
if (m_StreamType == stEXT)
|
|
|
|
return Respond("HTTP/1.0 200 OK");
|
|
|
|
else if (m_StreamType == stES && (m_Apid[0] || m_Dpid[0]))
|
|
|
|
return HttpResponse(200, false, "audio/mpeg");
|
|
|
|
|
|
|
|
const char* contentType = (isPes || m_RecPlayer->getPatPmtData()->Vpid()) ? "video/mpeg" : "audio/mpeg";
|
|
|
|
// range not supported when remuxing
|
|
|
|
if (m_StreamType != stTS && !isPes)
|
|
|
|
return HttpResponse(200, false, contentType);
|
|
|
|
|
2012-12-16 13:29:15 +01:00
|
|
|
uint64_t total = recStreamer->GetLength();
|
2013-10-01 23:47:25 +02:00
|
|
|
if (hasRange) {
|
2012-12-16 13:29:15 +01:00
|
|
|
int64_t length = recStreamer->SetRange(from, to);
|
2013-09-26 09:31:35 +02:00
|
|
|
Dprintf("range response: %lld-%lld/%lld, len %lld\n", (long long)from, (long long)to, (long long)total, (long long)length);
|
2012-12-16 13:29:15 +01:00
|
|
|
if (length < 0L)
|
2014-08-30 23:58:11 +02:00
|
|
|
return HttpResponse(416, true, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes */%llu", (unsigned long long) total);
|
2012-12-16 13:29:15 +01:00
|
|
|
else
|
2014-08-30 23:58:11 +02:00
|
|
|
return HttpResponse(206, false, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes %lld-%lld/%llu\r\nContent-Length: %lld", (long long) from, (long long) to, (unsigned long long) total, (long long) length);
|
2012-12-16 13:29:15 +01:00
|
|
|
}
|
2013-10-01 23:47:25 +02:00
|
|
|
else
|
2014-08-30 23:58:11 +02:00
|
|
|
return HttpResponse(200, false, contentType, "Accept-Ranges: bytes");
|
2012-12-16 13:29:15 +01:00
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
else {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(404, true);
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
2010-07-22 16:18:17 +02:00
|
|
|
} else if (it_method->second.compare("HEAD") == 0 && ProcessURI(it_pathinfo->second)) {
|
2013-02-03 11:02:25 +01:00
|
|
|
if (m_MenuList) {
|
2012-11-02 09:09:15 +01:00
|
|
|
DeferClose();
|
2013-02-03 11:02:25 +01:00
|
|
|
return Respond("%s", true, m_MenuList->HttpHeader().c_str());
|
2012-11-02 09:09:15 +01:00
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
else if (m_Channel != NULL) {
|
2014-06-07 00:24:27 +02:00
|
|
|
if (cStreamdevLiveStreamer::ProvidesChannel(m_Channel, StreamdevServerSetup.HTTPPriority)) {
|
2010-07-20 14:26:29 +02:00
|
|
|
if (m_StreamType == stEXT) {
|
2014-06-07 00:24:27 +02:00
|
|
|
cStreamdevLiveStreamer *liveStreamer = new cStreamdevLiveStreamer(this, m_Channel, IDLEPRIORITY, m_StreamType, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
2014-05-18 18:16:51 +02:00
|
|
|
SetStreamer(liveStreamer);
|
2012-11-02 09:09:15 +01:00
|
|
|
return Respond("HTTP/1.0 200 OK");
|
2011-10-19 23:21:38 +02:00
|
|
|
} else if (m_StreamType == stES && (m_Apid[0] || m_Dpid[0] || ISRADIO(m_Channel))) {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(200, true, "audio/mpeg", "icy-name: %s", m_Channel->Name());
|
2011-10-19 23:21:38 +02:00
|
|
|
} else if (ISRADIO(m_Channel)) {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(200, true, "audio/mpeg");
|
2010-07-20 14:26:29 +02:00
|
|
|
} else {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(200, true, "video/mpeg");
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
|
|
|
}
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(503, true);
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
2013-10-01 23:47:25 +02:00
|
|
|
else if (m_RecPlayer != NULL) {
|
2012-12-16 13:29:15 +01:00
|
|
|
Dprintf("HEAD recording\n");
|
2014-08-30 23:58:11 +02:00
|
|
|
bool isPes = m_RecPlayer->getCurrentRecording()->IsPesRecording();
|
|
|
|
// no remuxing for old PES recordings
|
|
|
|
if (isPes && m_StreamType != stPES)
|
|
|
|
return HttpResponse(503, true);
|
|
|
|
|
2012-12-16 13:29:15 +01:00
|
|
|
int64_t from, to;
|
2013-10-01 23:47:25 +02:00
|
|
|
bool hasRange = ParseRange(from, to);
|
|
|
|
|
|
|
|
cStreamdevRecStreamer* recStreamer;
|
|
|
|
if (from == 0 && hasRange && m_ReplayFakeRange) {
|
2014-08-30 23:58:11 +02:00
|
|
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, m_ReplayPos, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
2013-10-01 23:47:25 +02:00
|
|
|
from += m_ReplayPos;
|
|
|
|
if (to >= 0)
|
|
|
|
to += m_ReplayPos;
|
|
|
|
}
|
|
|
|
else
|
2014-08-30 23:58:11 +02:00
|
|
|
recStreamer = new cStreamdevRecStreamer(this, m_RecPlayer, m_StreamType, m_ReplayPos, m_Apid[0] ? m_Apid : NULL, m_Dpid[0] ? m_Dpid : NULL);
|
2014-05-18 18:16:51 +02:00
|
|
|
SetStreamer(recStreamer);
|
2014-08-30 23:58:11 +02:00
|
|
|
|
|
|
|
if (m_StreamType == stEXT)
|
|
|
|
return Respond("HTTP/1.0 200 OK");
|
|
|
|
else if (m_StreamType == stES && (m_Apid[0] || m_Dpid[0]))
|
|
|
|
return HttpResponse(200, true, "audio/mpeg");
|
|
|
|
|
|
|
|
const char* contentType = (isPes || m_RecPlayer->getPatPmtData()->Vpid()) ? "video/mpeg" : "audio/mpeg";
|
|
|
|
// range not supported when remuxing
|
|
|
|
if (m_StreamType != stTS && !isPes)
|
|
|
|
return HttpResponse(200, false, contentType);
|
|
|
|
|
2012-12-16 13:29:15 +01:00
|
|
|
uint64_t total = recStreamer->GetLength();
|
2013-10-01 23:47:25 +02:00
|
|
|
if (hasRange) {
|
2012-12-16 13:29:15 +01:00
|
|
|
int64_t length = recStreamer->SetRange(from, to);
|
|
|
|
if (length < 0L)
|
2014-08-30 23:58:11 +02:00
|
|
|
return HttpResponse(416, true, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes */%llu", (unsigned long long) total);
|
2012-12-16 13:29:15 +01:00
|
|
|
else
|
2014-08-30 23:58:11 +02:00
|
|
|
return HttpResponse(206, true, contentType, "Accept-Ranges: bytes\r\nContent-Range: bytes %lld-%lld/%llu\r\nContent-Length: %lld", (long long) from, (long long) to, (unsigned long long) total, (long long) length);
|
2012-12-16 13:29:15 +01:00
|
|
|
}
|
|
|
|
else
|
2014-08-30 23:58:11 +02:00
|
|
|
return HttpResponse(200, true, contentType, "Accept-Ranges: bytes");
|
2012-12-16 13:29:15 +01:00
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
else {
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(404, true);
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
2005-02-11 17:44:14 +01:00
|
|
|
|
2012-11-16 02:00:09 +01:00
|
|
|
return HttpResponse(400, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *AAA[] = {
|
|
|
|
"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
|
|
|
|
};
|
|
|
|
static const char *MMM[] = {
|
|
|
|
"Jan", "Feb", "Mar", "Apr", "May", "Jun",
|
|
|
|
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
|
|
|
|
};
|
|
|
|
|
|
|
|
bool cConnectionHTTP::HttpResponse(int Code, bool Last, const char* ContentType, const char* Headers, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, Headers);
|
2012-12-04 17:19:35 +01:00
|
|
|
#if APIVERSNUM >= 10728
|
|
|
|
cString headers = cString::vsprintf(Headers, ap);
|
|
|
|
#else
|
2012-11-16 02:00:09 +01:00
|
|
|
cString headers = cString::sprintf(Headers, ap);
|
2012-12-04 17:19:35 +01:00
|
|
|
#endif
|
2012-11-16 02:00:09 +01:00
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
bool rc;
|
|
|
|
if (Last)
|
|
|
|
DeferClose();
|
|
|
|
switch (Code)
|
|
|
|
{
|
|
|
|
case 200: rc = Respond("HTTP/1.1 200 OK"); break;
|
2012-12-16 13:29:15 +01:00
|
|
|
case 206: rc = Respond("HTTP/1.1 206 Partial Content"); break;
|
2012-11-16 02:00:09 +01:00
|
|
|
case 400: rc = Respond("HTTP/1.1 400 Bad Request"); break;
|
|
|
|
case 401: rc = Respond("HTTP/1.1 401 Authorization Required"); break;
|
|
|
|
case 404: rc = Respond("HTTP/1.1 404 Not Found"); break;
|
2012-12-16 13:29:15 +01:00
|
|
|
case 416: rc = Respond("HTTP/1.1 416 Requested range not satisfiable"); break;
|
2012-11-16 02:00:09 +01:00
|
|
|
case 503: rc = Respond("HTTP/1.1 503 Service Unavailable"); break;
|
|
|
|
default: rc = Respond("HTTP/1.1 500 Internal Server Error");
|
|
|
|
}
|
|
|
|
if (rc && ContentType)
|
|
|
|
rc = Respond("Content-Type: %s", true, ContentType);
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
rc = Respond("Connection: close")
|
|
|
|
&& Respond("Pragma: no-cache")
|
2013-10-02 00:03:39 +02:00
|
|
|
&& Respond("Cache-Control: no-cache")
|
|
|
|
&& Respond("Server: VDR-%s / streamdev-server-%s", true, VDRVERSION, VERSION);
|
2012-11-16 02:00:09 +01:00
|
|
|
|
|
|
|
time_t t = time(NULL);
|
|
|
|
struct tm *gmt = gmtime(&t);
|
|
|
|
if (rc && gmt) {
|
|
|
|
char buf[] = "Date: AAA, DD MMM YYYY HH:MM:SS GMT";
|
|
|
|
if (snprintf(buf, sizeof(buf), "Date: %s, %.2d %s %.4d %.2d:%.2d:%.2d GMT", AAA[gmt->tm_wday], gmt->tm_mday, MMM[gmt->tm_mon], gmt->tm_year + 1900, gmt->tm_hour, gmt->tm_min, gmt->tm_sec) == sizeof(buf) - 1)
|
|
|
|
rc = Respond(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc && strlen(Headers) > 0)
|
|
|
|
rc = Respond(headers);
|
|
|
|
|
|
|
|
tStrStrMap::iterator it = m_Params.begin();
|
|
|
|
while (rc && it != m_Params.end()) {
|
|
|
|
static const char DLNA_POSTFIX[] = ".dlna.org";
|
|
|
|
if (it->first.rfind(DLNA_POSTFIX) + sizeof(DLNA_POSTFIX) - 1 == it->first.length())
|
|
|
|
rc = Respond("%s: %s", true, it->first.c_str(), it->second.c_str());
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
return rc && Respond("");
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
|
|
|
|
2012-12-16 13:29:15 +01:00
|
|
|
bool cConnectionHTTP::ParseRange(int64_t &From, int64_t &To) const
|
|
|
|
{
|
|
|
|
const static std::string RANGE("HTTP_RANGE");
|
|
|
|
From = To = 0L;
|
|
|
|
tStrStrMap::const_iterator it = Headers().find(RANGE);
|
|
|
|
if (it != Headers().end()) {
|
|
|
|
size_t b = it->second.find("bytes=");
|
|
|
|
if (b != std::string::npos) {
|
|
|
|
char* e = NULL;
|
|
|
|
const char* r = it->second.c_str() + b + sizeof("bytes=") - 1;
|
|
|
|
if (strchr(r, ',') != NULL)
|
|
|
|
esyslog("streamdev-server cConnectionHTTP::GetRange: Multi-ranges not supported");
|
|
|
|
From = strtol(r, &e, 10);
|
|
|
|
if (r != e) {
|
|
|
|
if (From < 0L) {
|
|
|
|
To = -1L;
|
|
|
|
return *e == 0 || *e == ',';
|
|
|
|
}
|
|
|
|
else if (*e == '-') {
|
|
|
|
r = e + 1;
|
|
|
|
if (*r == 0 || *e == ',') {
|
|
|
|
To = -1L;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
To = strtol(r, &e, 10);
|
|
|
|
return r != e && To >= From &&
|
|
|
|
(*e == 0 || *e == ',');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2005-05-09 22:22:29 +02:00
|
|
|
void cConnectionHTTP::Flushed(void)
|
|
|
|
{
|
2005-02-11 17:44:14 +01:00
|
|
|
if (m_Status != hsBody)
|
|
|
|
return;
|
|
|
|
|
2013-02-03 11:02:25 +01:00
|
|
|
if (m_MenuList) {
|
|
|
|
if (m_MenuList->HasNext()) {
|
|
|
|
if (!Respond("%s", true, m_MenuList->Next().c_str()))
|
2008-03-28 16:11:40 +01:00
|
|
|
DeferClose();
|
2005-02-11 17:44:14 +01:00
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
else {
|
2013-02-03 11:02:25 +01:00
|
|
|
DELETENULL(m_MenuList);
|
2010-07-20 14:26:29 +02:00
|
|
|
m_Status = hsFinished;
|
|
|
|
DeferClose();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2014-05-18 18:16:51 +02:00
|
|
|
else if (Streamer()) {
|
2004-12-30 23:43:55 +01:00
|
|
|
Dprintf("streamer start\n");
|
2014-05-18 18:16:51 +02:00
|
|
|
Streamer()->Start(this);
|
2005-02-11 17:44:14 +01:00
|
|
|
m_Status = hsFinished;
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
// should never be reached
|
|
|
|
esyslog("streamdev-server cConnectionHTTP::Flushed(): no job to do");
|
|
|
|
m_Status = hsFinished;
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-03 11:02:25 +01:00
|
|
|
cMenuList* cConnectionHTTP::MenuListFromString(const std::string& Path, const std::string& Filebase, const std::string& Fileext) const
|
2005-05-09 22:22:29 +02:00
|
|
|
{
|
2008-03-28 16:11:40 +01:00
|
|
|
std::string groupTarget;
|
2013-02-03 11:02:25 +01:00
|
|
|
cItemIterator *iterator = NULL;
|
2005-02-11 17:44:14 +01:00
|
|
|
|
2012-11-16 02:00:09 +01:00
|
|
|
const static std::string GROUP("group");
|
2010-07-20 14:26:29 +02:00
|
|
|
if (Filebase.compare("tree") == 0) {
|
2012-11-16 02:00:09 +01:00
|
|
|
tStrStrMap::const_iterator it = m_Params.find(GROUP);
|
2013-02-03 11:02:25 +01:00
|
|
|
iterator = new cListTree(it == m_Params.end() ? NULL : it->second.c_str());
|
2010-07-20 14:26:29 +02:00
|
|
|
groupTarget = Filebase + Fileext;
|
|
|
|
} else if (Filebase.compare("groups") == 0) {
|
2008-03-28 16:11:40 +01:00
|
|
|
iterator = new cListGroups();
|
2010-07-20 14:26:29 +02:00
|
|
|
groupTarget = (std::string) "group" + Fileext;
|
|
|
|
} else if (Filebase.compare("group") == 0) {
|
2012-11-16 02:00:09 +01:00
|
|
|
tStrStrMap::const_iterator it = m_Params.find(GROUP);
|
2013-02-03 11:02:25 +01:00
|
|
|
iterator = new cListGroup(it == m_Params.end() ? NULL : it->second.c_str());
|
2010-07-20 14:26:29 +02:00
|
|
|
} else if (Filebase.compare("channels") == 0) {
|
2008-03-28 16:11:40 +01:00
|
|
|
iterator = new cListChannels();
|
2010-07-20 14:26:29 +02:00
|
|
|
} else if (Filebase.compare("all") == 0 ||
|
|
|
|
(Filebase.empty() && Fileext.empty())) {
|
2008-03-28 16:11:40 +01:00
|
|
|
iterator = new cListAll();
|
2013-02-03 12:40:46 +01:00
|
|
|
} else if (Filebase.compare("recordings") == 0) {
|
2014-09-08 22:35:18 +02:00
|
|
|
iterator = new cRecordingsIterator(m_StreamType);
|
2008-03-28 16:11:40 +01:00
|
|
|
}
|
2004-12-30 23:43:55 +01:00
|
|
|
|
2008-03-28 16:11:40 +01:00
|
|
|
if (iterator) {
|
2013-02-02 23:28:55 +01:00
|
|
|
// assemble base url: http://host/path/
|
|
|
|
std::string base;
|
|
|
|
const static std::string HOST("HTTP_HOST");
|
|
|
|
tStrStrMap::const_iterator it = Headers().find(HOST);
|
|
|
|
if (it != Headers().end())
|
|
|
|
base = "http://" + it->second + "/";
|
|
|
|
else
|
|
|
|
base = (std::string) "http://" + LocalIp() + ":" +
|
|
|
|
(const char*) itoa(StreamdevServerSetup.HTTPServerPort) + "/";
|
|
|
|
base += Path;
|
|
|
|
|
2010-07-20 14:26:29 +02:00
|
|
|
if (Filebase.empty() || Fileext.compare(".htm") == 0 || Fileext.compare(".html") == 0) {
|
|
|
|
std::string self = Filebase + Fileext;
|
2013-02-02 23:28:55 +01:00
|
|
|
std::string rss = Filebase + ".rss";
|
2012-12-16 13:09:29 +01:00
|
|
|
tStrStrMap::const_iterator it = Headers().find("QUERY_STRING");
|
2013-02-02 23:28:55 +01:00
|
|
|
if (it != Headers().end() && !it->second.empty()) {
|
2012-12-16 13:09:29 +01:00
|
|
|
self += '?' + it->second;
|
2013-02-02 23:28:55 +01:00
|
|
|
rss += '?' + it->second;
|
|
|
|
}
|
2013-02-03 11:02:25 +01:00
|
|
|
return new cHtmlMenuList(iterator, m_StreamType, self.c_str(), rss.c_str(), groupTarget.c_str());
|
2010-07-20 14:26:29 +02:00
|
|
|
} else if (Fileext.compare(".m3u") == 0) {
|
2013-02-03 11:02:25 +01:00
|
|
|
return new cM3uMenuList(iterator, base.c_str());
|
2013-02-02 23:28:55 +01:00
|
|
|
} else if (Fileext.compare(".rss") == 0) {
|
|
|
|
std::string html = Filebase + ".html";
|
|
|
|
tStrStrMap::const_iterator it = Headers().find("QUERY_STRING");
|
|
|
|
if (it != Headers().end() && !it->second.empty()) {
|
|
|
|
html += '?' + it->second;
|
|
|
|
}
|
2013-02-03 11:02:25 +01:00
|
|
|
return new cRssMenuList(iterator, base.c_str(), html.c_str());
|
2008-03-28 16:11:40 +01:00
|
|
|
} else {
|
|
|
|
delete iterator;
|
|
|
|
}
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-10-01 23:47:25 +02:00
|
|
|
RecPlayer* cConnectionHTTP::RecPlayerFromString(const char *FileBase, const char *FileExt)
|
2013-02-02 22:34:47 +01:00
|
|
|
{
|
2013-10-01 23:47:25 +02:00
|
|
|
RecPlayer *recPlayer = NULL;
|
|
|
|
|
2013-02-02 22:34:47 +01:00
|
|
|
if (strcasecmp(FileExt, ".rec") != 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
char *p = NULL;
|
|
|
|
unsigned long l = strtoul(FileBase, &p, 0);
|
|
|
|
if (p != FileBase && l > 0L) {
|
|
|
|
if (*p == ':') {
|
|
|
|
// get recording by dev:inode
|
2013-07-16 13:07:58 +02:00
|
|
|
ino_t inode = (ino_t) strtoull(p + 1, &p, 0);
|
2013-02-02 22:34:47 +01:00
|
|
|
if (*p == 0 && inode > 0) {
|
|
|
|
struct stat st;
|
|
|
|
cThreadLock RecordingsLock(&Recordings);
|
|
|
|
for (cRecording *rec = Recordings.First(); rec; rec = Recordings.Next(rec)) {
|
2013-07-16 13:07:58 +02:00
|
|
|
if (stat(rec->FileName(), &st) == 0 && st.st_dev == (dev_t) l && st.st_ino == inode)
|
2013-10-01 23:47:25 +02:00
|
|
|
recPlayer = new RecPlayer(rec->FileName());
|
2013-02-02 22:34:47 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (*p == 0) {
|
|
|
|
// get recording by index
|
|
|
|
cThreadLock RecordingsLock(&Recordings);
|
2013-10-01 23:47:25 +02:00
|
|
|
cRecording *rec = Recordings.Get((int) l - 1);
|
2013-02-02 22:34:47 +01:00
|
|
|
if (rec)
|
2013-10-01 23:47:25 +02:00
|
|
|
recPlayer = new RecPlayer(rec->FileName());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (recPlayer) {
|
|
|
|
const char *pos = NULL;
|
|
|
|
tStrStrMap::const_iterator it = m_Params.begin();
|
|
|
|
while (it != m_Params.end()) {
|
|
|
|
if (it->first == "pos") {
|
|
|
|
pos = it->second.c_str();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
if (pos) {
|
|
|
|
// With prefix "full_" we try to fool players
|
|
|
|
// by replying with a content range starting
|
|
|
|
// at the requested position instead of 0.
|
|
|
|
// This is a heavy violation of standards.
|
|
|
|
// Use at your own risk!
|
|
|
|
if (strncasecmp(pos, "full_", 5) == 0) {
|
|
|
|
m_ReplayFakeRange = true;
|
|
|
|
pos += 5;
|
|
|
|
}
|
|
|
|
if (strncasecmp(pos, "resume", 6) == 0) {
|
|
|
|
int id = pos[6] == '.' ? atoi(pos + 7) : 0;
|
|
|
|
m_ReplayPos = recPlayer->positionFromResume(id);
|
|
|
|
}
|
|
|
|
else if (strncasecmp(pos, "mark.", 5) == 0) {
|
|
|
|
int index = atoi(pos + 5);
|
|
|
|
m_ReplayPos = recPlayer->positionFromMark(index);
|
|
|
|
}
|
|
|
|
else if (strncasecmp(pos, "time.", 5) == 0) {
|
|
|
|
int seconds = atoi(pos + 5);
|
|
|
|
m_ReplayPos = recPlayer->positionFromTime(seconds);
|
|
|
|
}
|
|
|
|
else if (strncasecmp(pos, "frame.", 6) == 0) {
|
|
|
|
int frame = atoi(pos + 6);
|
|
|
|
m_ReplayPos = recPlayer->positionFromFrameNumber(frame);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
m_ReplayPos = atol(pos);
|
|
|
|
if (m_ReplayPos > 0L && m_ReplayPos < 100L)
|
|
|
|
m_ReplayPos = recPlayer->positionFromPercent((int) m_ReplayPos);
|
|
|
|
}
|
|
|
|
}
|
2013-02-02 22:34:47 +01:00
|
|
|
}
|
|
|
|
}
|
2013-10-01 23:47:25 +02:00
|
|
|
return recPlayer;
|
2013-02-02 22:34:47 +01:00
|
|
|
}
|
|
|
|
|
2010-07-20 14:26:29 +02:00
|
|
|
bool cConnectionHTTP::ProcessURI(const std::string& PathInfo)
|
|
|
|
{
|
|
|
|
std::string filespec, fileext;
|
|
|
|
size_t file_pos = PathInfo.rfind('/');
|
|
|
|
|
|
|
|
if (file_pos != std::string::npos) {
|
|
|
|
size_t ext_pos = PathInfo.rfind('.');
|
2013-10-01 23:52:03 +02:00
|
|
|
if (ext_pos != std::string::npos) {
|
2010-07-20 14:26:29 +02:00
|
|
|
// file extension including leading .
|
|
|
|
fileext = PathInfo.substr(ext_pos);
|
2013-10-01 23:52:03 +02:00
|
|
|
const char *ext = fileext.c_str();
|
|
|
|
// ignore dummy file extensions
|
|
|
|
if (strcasecmp(ext, ".ts") == 0 ||
|
|
|
|
strcasecmp(ext, ".vdr") == 0 ||
|
|
|
|
strcasecmp(ext, ".vob") == 0) {
|
|
|
|
size_t ext_end = ext_pos;
|
|
|
|
if (ext_pos > 0)
|
|
|
|
ext_pos = PathInfo.rfind('.', ext_pos - 1);
|
|
|
|
if (ext_pos == std::string::npos)
|
|
|
|
ext_pos = ext_end;
|
|
|
|
fileext = PathInfo.substr(ext_pos, ext_end - ext_pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// file basename with leading / stripped off
|
|
|
|
filespec = PathInfo.substr(file_pos + 1, ext_pos - file_pos - 1);
|
2010-07-20 14:26:29 +02:00
|
|
|
}
|
|
|
|
if (fileext.length() > 5) {
|
|
|
|
//probably not an extension
|
|
|
|
filespec += fileext;
|
|
|
|
fileext.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Streamtype with leading / stripped off
|
|
|
|
std::string type = PathInfo.substr(1, PathInfo.find_first_of("/;", 1) - 1);
|
|
|
|
const char* pType = type.c_str();
|
2013-11-17 11:20:42 +01:00
|
|
|
if (strcasecmp(pType, "PES") == 0) {
|
2010-07-20 14:26:29 +02:00
|
|
|
m_StreamType = stPES;
|
2013-11-17 11:20:42 +01:00
|
|
|
#ifdef STREAMDEV_PS
|
|
|
|
} else if (strcasecmp(pType, "PS") == 0) {
|
|
|
|
m_StreamType = stPS;
|
|
|
|
#endif
|
2010-07-20 14:26:29 +02:00
|
|
|
} else if (strcasecmp(pType, "TS") == 0) {
|
|
|
|
m_StreamType = stTS;
|
|
|
|
} else if (strcasecmp(pType, "ES") == 0) {
|
|
|
|
m_StreamType = stES;
|
|
|
|
} else if (strcasecmp(pType, "EXT") == 0) {
|
|
|
|
m_StreamType = stEXT;
|
|
|
|
}
|
|
|
|
|
|
|
|
Dprintf("before channelfromstring: type(%s) filespec(%s) fileext(%s)\n", type.c_str(), filespec.c_str(), fileext.c_str());
|
|
|
|
|
2013-02-03 11:02:25 +01:00
|
|
|
if ((m_MenuList = MenuListFromString(PathInfo.substr(1, file_pos), filespec.c_str(), fileext.c_str())) != NULL) {
|
2010-07-20 14:26:29 +02:00
|
|
|
Dprintf("Channel list requested\n");
|
|
|
|
return true;
|
2013-10-01 23:47:25 +02:00
|
|
|
} else if ((m_RecPlayer = RecPlayerFromString(filespec.c_str(), fileext.c_str())) != NULL) {
|
|
|
|
Dprintf("Recording %s found\n", m_RecPlayer->getCurrentRecording()->Name());
|
2013-02-02 22:34:47 +01:00
|
|
|
return true;
|
2010-07-20 14:26:29 +02:00
|
|
|
} else if ((m_Channel = ChannelFromString(filespec.c_str(), &m_Apid[0], &m_Dpid[0])) != NULL) {
|
|
|
|
Dprintf("Channel found. Apid/Dpid is %d/%d\n", m_Apid[0], m_Dpid[0]);
|
|
|
|
return true;
|
2008-03-28 16:11:40 +01:00
|
|
|
} else
|
|
|
|
return false;
|
2004-12-30 23:43:55 +01:00
|
|
|
}
|
|
|
|
|
2014-11-07 23:01:08 +01:00
|
|
|
cString cConnectionHTTP::ToText(char Delimiter) const
|
2011-11-28 16:23:57 +01:00
|
|
|
{
|
2014-11-07 23:01:08 +01:00
|
|
|
cString str = cServerConnection::ToText(Delimiter);
|
|
|
|
return Streamer() ? cString::sprintf("%s%c%s", *str, Delimiter, *Streamer()->ToText()) : str;
|
2011-11-28 16:23:57 +01:00
|
|
|
}
|