mirror of
https://github.com/tuxbox-neutrino/neutrino.git
synced 2025-08-31 17:31:20 +02:00
git-svn-id: file:///home/bas/coolstream_public_svn/THIRDPARTY/applications/neutrino-experimental@27 e54a6e83-5905-42d5-8d5c-058d10e6a962
358 lines
14 KiB
C++
358 lines
14 KiB
C++
//=============================================================================
|
|
// YHTTPD
|
|
// Hook and HookHandler
|
|
//=============================================================================
|
|
// C
|
|
#include <cstdarg>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
|
|
// yhttpd
|
|
#include "yhook.h"
|
|
#include "ylogging.h"
|
|
#include "helper.h"
|
|
|
|
//=============================================================================
|
|
// Initialization of static variables
|
|
//=============================================================================
|
|
THookList CyhookHandler::HookList;
|
|
|
|
//=============================================================================
|
|
// Hook Handling
|
|
//=============================================================================
|
|
//-----------------------------------------------------------------------------
|
|
// Hook Dispatcher for Session Hooks
|
|
// Execute every Hook in HookList until State change != HANDLED_NONE
|
|
//-----------------------------------------------------------------------------
|
|
THandleStatus CyhookHandler::Hooks_SendResponse()
|
|
{
|
|
log_level_printf(4,"Response Hook-List Start\n");
|
|
THandleStatus _status = HANDLED_NONE;
|
|
THookList::iterator i = HookList.begin();
|
|
for ( ; i!= HookList.end(); i++ )
|
|
{
|
|
log_level_printf(4,"Response Hook-List (%s) Start\n", ((*i)->getHookName()).c_str());
|
|
// response Hook
|
|
_status = (*i)->Hook_SendResponse(this);
|
|
log_level_printf(4,"Response Hook-List (%s) End. Status (%d)\n", ((*i)->getHookName()).c_str(), status);
|
|
if((_status != HANDLED_NONE) && (_status != HANDLED_CONTINUE))
|
|
break;
|
|
}
|
|
log_level_printf(4,"Response Hook-List End\n");
|
|
log_level_printf(8,"Response Hook-List Result:\n%s\n", yresult.c_str());
|
|
status = _status;
|
|
return _status;
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// Hook Dispatcher for Session Hooks
|
|
// Execute every Hook in HookList until State change != HANDLED_NONE
|
|
//-----------------------------------------------------------------------------
|
|
THandleStatus CyhookHandler::Hooks_PrepareResponse()
|
|
{
|
|
log_level_printf(4,"PrepareResponse Hook-List Start\n");
|
|
THandleStatus _status = HANDLED_NONE;
|
|
THookList::iterator i = HookList.begin();
|
|
for ( ; i!= HookList.end(); i++ )
|
|
{
|
|
log_level_printf(4,"PrepareResponse Hook-List (%s) Start\n", ((*i)->getHookName()).c_str());
|
|
// response Hook
|
|
_status = (*i)->Hook_PrepareResponse(this);
|
|
log_level_printf(4,"PrepareResponse Hook-List (%s) End. Status (%d) HTTP Status (%d)\n", ((*i)->getHookName()).c_str(), status, httpStatus);
|
|
if((_status != HANDLED_NONE) && (_status != HANDLED_CONTINUE))
|
|
break;
|
|
}
|
|
log_level_printf(4,"PrepareResponse Hook-List End\n");
|
|
log_level_printf(8,"PrepareResponse Hook-List Result:\n%s\n", yresult.c_str());
|
|
status = _status;
|
|
return _status;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Hook Dispatcher for Server based Hooks
|
|
// Execute every Hook in HookList until State change != HANDLED_NONE and
|
|
// != HANDLED_CONTINUE
|
|
//-----------------------------------------------------------------------------
|
|
THandleStatus CyhookHandler::Hooks_ReadConfig(CConfigFile *Config, CStringList &ConfigList)
|
|
{
|
|
log_level_printf(4,"ReadConfig Hook-List Start\n");
|
|
THandleStatus _status = HANDLED_NONE;
|
|
THookList::iterator i = HookList.begin();
|
|
for ( ; i!= HookList.end(); i++ )
|
|
{
|
|
// log_level_printf(4,"ReadConfig Hook-List (%s) Start\n", ((*i)->getHookName()).c_str());
|
|
// response Hook
|
|
_status = (*i)->Hook_ReadConfig(Config, ConfigList);
|
|
log_level_printf(4,"ReadConfig Hook-List (%s) Status (%d)\n", ((*i)->getHookName()).c_str(), _status);
|
|
if((_status != HANDLED_NONE) && (_status != HANDLED_CONTINUE))
|
|
break;
|
|
}
|
|
log_level_printf(4,"ReadConfig Hook-List End\n");
|
|
return _status;
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// Hook Dispatcher for EndConnection
|
|
//-----------------------------------------------------------------------------
|
|
THandleStatus CyhookHandler::Hooks_EndConnection()
|
|
{
|
|
log_level_printf(4,"EndConnection Hook-List Start\n");
|
|
THandleStatus _status = HANDLED_NONE;
|
|
THookList::iterator i = HookList.begin();
|
|
for ( ; i!= HookList.end(); i++ )
|
|
{
|
|
log_level_printf(4,"EndConnection Hook-List (%s) Start\n", ((*i)->getHookName()).c_str());
|
|
// response Hook
|
|
_status = (*i)->Hook_EndConnection(this);
|
|
log_level_printf(4,"EndConnection Hook-List (%s) End. Status (%d)\n", ((*i)->getHookName()).c_str(), _status);
|
|
if((_status != HANDLED_NONE) && (_status != HANDLED_CONTINUE))
|
|
break;
|
|
}
|
|
log_level_printf(4,"EndConnection Hook-List End\n");
|
|
status = _status;
|
|
return _status;
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// Hook Dispatcher for UploadSetFilename
|
|
//-----------------------------------------------------------------------------
|
|
THandleStatus CyhookHandler::Hooks_UploadSetFilename(std::string &Filename)
|
|
{
|
|
log_level_printf(4,"UploadSetFilename Hook-List Start. Filename:(%s)\n", Filename.c_str());
|
|
THandleStatus _status = HANDLED_NONE;
|
|
THookList::iterator i = HookList.begin();
|
|
for ( ; i!= HookList.end(); i++ )
|
|
{
|
|
log_level_printf(4,"UploadSetFilename Hook-List (%s) Start\n", ((*i)->getHookName()).c_str());
|
|
// response Hook
|
|
_status = (*i)->Hook_UploadSetFilename(this, Filename);
|
|
log_level_printf(4,"UploadSetFilename Hook-List (%s) End. Status (%d)\n", ((*i)->getHookName()).c_str(), _status);
|
|
if((_status != HANDLED_NONE) && (_status != HANDLED_CONTINUE))
|
|
break;
|
|
}
|
|
log_level_printf(4,"UploadSetFilename Hook-List End\n");
|
|
status = _status;
|
|
return _status;
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
// Hook Dispatcher for UploadSetFilename
|
|
//-----------------------------------------------------------------------------
|
|
THandleStatus CyhookHandler::Hooks_UploadReady(const std::string& Filename)
|
|
{
|
|
log_level_printf(4,"UploadReady Hook-List Start. Filename:(%s)\n", Filename.c_str());
|
|
THandleStatus _status = HANDLED_NONE;
|
|
THookList::iterator i = HookList.begin();
|
|
for ( ; i!= HookList.end(); i++ )
|
|
{
|
|
log_level_printf(4,"UploadReady Hook-List (%s) Start\n", ((*i)->getHookName()).c_str());
|
|
// response Hook
|
|
_status = (*i)->Hook_UploadReady(this, Filename);
|
|
log_level_printf(4,"UploadReady Hook-List (%s) End. Status (%d)\n", ((*i)->getHookName()).c_str(), _status);
|
|
if((_status != HANDLED_NONE) && (_status != HANDLED_CONTINUE))
|
|
break;
|
|
}
|
|
log_level_printf(4,"UploadReady Hook-List End\n");
|
|
status = _status;
|
|
return _status;
|
|
}
|
|
|
|
//=============================================================================
|
|
// Output helpers
|
|
//=============================================================================
|
|
void CyhookHandler::session_init(CStringList _ParamList, CStringList _UrlData, CStringList _HeaderList,
|
|
CStringList& _ConfigList, THttp_Method _Method, bool _keep_alive)
|
|
{
|
|
ParamList = _ParamList;
|
|
UrlData = _UrlData;
|
|
HeaderList = _HeaderList;
|
|
WebserverConfigList = _ConfigList;
|
|
Method = _Method;
|
|
status = HANDLED_NONE;
|
|
httpStatus = HTTP_OK;
|
|
ContentLength = 0;
|
|
LastModified = (time_t)-1;
|
|
keep_alive = _keep_alive;
|
|
HookVarList.clear();
|
|
}
|
|
|
|
//=============================================================================
|
|
// Build Header
|
|
//-----------------------------------------------------------------------------
|
|
// RFC 2616 / 6 Response (Header)
|
|
//
|
|
// The first line of a Response message is the Status-Line, consisting
|
|
// of the protocol version followed by a numeric status code and its
|
|
// associated textual phrase, with each element separated by SP
|
|
// characters. No CR or LF is allowed except in the final CRLF sequence.
|
|
//
|
|
// Response =Status-Line ; generated by SendHeader
|
|
// *(( general-header
|
|
// | response-header
|
|
// | entity-header ) CRLF)
|
|
// CRLF ; generated by SendHeader
|
|
// [ message-body ] ; by HOOK Handling Loop or Sendfile
|
|
//
|
|
// Status-Line = HTTP-Version SP Status-Code SP Reason-Phrase CRLF ; generated by SendHeader
|
|
//
|
|
// general-header = Cache-Control ; not implemented
|
|
// | Connection ; implemented
|
|
// | Date ; implemented
|
|
// | Pragma ; not implemented
|
|
// | Trailer ; not implemented
|
|
// | Transfer-Encoding ; not implemented
|
|
// | Upgrade ; not implemented
|
|
// | Via ; not implemented
|
|
// | Warning ; not implemented
|
|
//
|
|
// response-header = Accept-Ranges ; not implemented
|
|
// | Age ; not implemented
|
|
// | ETag ; not implemented
|
|
// | Location ; implemented (redirection / Object moved)
|
|
// | Proxy-Authenticate ; not implemented
|
|
// | Retry-After ; not implemented
|
|
// | Server ; implemented
|
|
// | Vary ; not implemented
|
|
// | WWW-Authenticate ; implemented (by mod_auth and SendHeader)
|
|
//
|
|
// entity-header = Allow ; not implemented
|
|
// | Content-Encoding ; not implemented
|
|
// | Content-Language ; not implemented
|
|
// | Content-Length ; implemented
|
|
// | Content-Location ; not implemented
|
|
// | Content-MD5 ; not implemented
|
|
// | Content-Range ; not implemented
|
|
// | Content-Type ; implemented
|
|
// | Expires ; not implemented
|
|
// | Last-Modified ; implemented for static files
|
|
// | extension-header
|
|
//
|
|
// extension-header = message-header
|
|
//=============================================================================
|
|
std::string CyhookHandler::BuildHeader(bool cache)
|
|
{
|
|
std::string result="";
|
|
|
|
const char *responseString = "";
|
|
const char *infoString = 0;
|
|
|
|
// get Info Index
|
|
for (unsigned int i = 0;i < (sizeof(httpResponseNames)/sizeof(httpResponseNames[0])); i++)
|
|
if (httpResponseNames[i].type == httpStatus)
|
|
{
|
|
responseString = httpResponseNames[i].name;
|
|
infoString = httpResponseNames[i].info;
|
|
break;
|
|
}
|
|
// print Status-line
|
|
result = string_printf(HTTP_PROTOCOL " %d %s\r\nContent-Type: %s\r\n",httpStatus, responseString, ResponseMimeType.c_str());
|
|
log_level_printf(2,"Respose: HTTP/1.1 %d %s\r\nContent-Type: %s\r\n",httpStatus, responseString, ResponseMimeType.c_str());
|
|
|
|
switch (httpStatus)
|
|
{
|
|
case HTTP_UNAUTHORIZED:
|
|
result += "WWW-Authenticate: Basic realm=\"";
|
|
result += AUTH_NAME_MSG "\r\n";
|
|
break;
|
|
|
|
case HTTP_MOVED_TEMPORARILY:
|
|
case HTTP_MOVED_PERMANENTLY:
|
|
// Status HTTP_*_TEMPORARILY (redirection)
|
|
result += string_printf("Location: %s\r\n",NewURL.c_str());
|
|
// NO break HERE !!!
|
|
|
|
default:
|
|
time_t timer = time(0);
|
|
char timeStr[80];
|
|
// cache
|
|
if(!cache && (HookVarList["CacheCategory"]).empty() )
|
|
result += "Cache-Control: no-cache\r\n";
|
|
else
|
|
{
|
|
time_t x_time = time(NULL);
|
|
struct tm *ptm = gmtime(&x_time);
|
|
ptm->tm_mday+=1;
|
|
x_time = mktime(ptm);
|
|
strftime(timeStr, sizeof(timeStr), RFC1123FMT, gmtime(&x_time));
|
|
result += string_printf("Expires: %s\r\n", timeStr);
|
|
}
|
|
result += "Server: " WEBSERVERNAME "\r\n";
|
|
// actual date
|
|
strftime(timeStr, sizeof(timeStr), RFC1123FMT, gmtime(&timer));
|
|
result += string_printf("Date: %s\r\n", timeStr);
|
|
// connection type
|
|
#ifdef Y_CONFIG_FEATURE_KEEP_ALIVE
|
|
if(keep_alive)
|
|
result += "Connection: keep-alive\r\n";
|
|
else
|
|
result += "Connection: close\r\n";
|
|
#else
|
|
result += "Connection: close\r\n";
|
|
#endif
|
|
// content-len, last-modified
|
|
if(httpStatus == HTTP_NOT_MODIFIED ||httpStatus == HTTP_NOT_FOUND)
|
|
result += "Content-Length: 0\r\n";
|
|
else if(GetContentLength() >0)
|
|
{
|
|
time_t mod_time = time(NULL);
|
|
if(LastModified != (time_t)-1)
|
|
mod_time = LastModified;
|
|
|
|
strftime(timeStr, sizeof(timeStr), RFC1123FMT, gmtime(&mod_time));
|
|
result += string_printf("Last-Modified: %s\r\nContent-Length: %ld\r\n", timeStr, GetContentLength());
|
|
}
|
|
result += "\r\n"; // End of Header
|
|
break;
|
|
}
|
|
|
|
// Body
|
|
if (Method != M_HEAD)
|
|
switch (httpStatus)
|
|
{
|
|
case HTTP_OK:
|
|
case HTTP_NOT_MODIFIED:
|
|
case HTTP_CONTINUE:
|
|
case HTTP_ACCEPTED:
|
|
case HTTP_NO_CONTENT:
|
|
case HTTP_NOT_FOUND:
|
|
case HTTP_INTERNAL_SERVER_ERROR:
|
|
break;
|
|
|
|
case HTTP_MOVED_TEMPORARILY:
|
|
case HTTP_MOVED_PERMANENTLY:
|
|
result += "<html><head><title>Object moved</title></head><body>";
|
|
result += string_printf("302 : Object moved.<br/>If you dont get redirected click <a href=\"%s\">here</a></body></html>\n",NewURL.c_str());
|
|
break;
|
|
|
|
default:
|
|
// Error pages
|
|
break;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
//=============================================================================
|
|
// Output helpers
|
|
//=============================================================================
|
|
//-----------------------------------------------------------------------------
|
|
void CyhookHandler::SendHTMLHeader(const std::string& Titel)
|
|
{
|
|
WriteLn("<html>\n<head><title>" + Titel + "</title>\n");
|
|
WriteLn("<meta http-equiv=\"cache-control\" content=\"no-cache\" />");
|
|
WriteLn("<meta http-equiv=\"expires\" content=\"0\" />\n</head>\n<body>\n");
|
|
}
|
|
//-----------------------------------------------------------------------------
|
|
void CyhookHandler::SendHTMLFooter(void)
|
|
{
|
|
WriteLn("</body>\n</html>\n\n");
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
#define OUTBUFSIZE 4096
|
|
void CyhookHandler::printf ( const char *fmt, ... )
|
|
{
|
|
char outbuf[OUTBUFSIZE];
|
|
bzero(outbuf,OUTBUFSIZE);
|
|
va_list arglist;
|
|
va_start( arglist, fmt );
|
|
vsnprintf( outbuf,OUTBUFSIZE, fmt, arglist );
|
|
va_end(arglist);
|
|
Write(outbuf);
|
|
}
|