//This file is part of HTTPServer a Miranda IM plugin
//Copyright (C)2002 Kennet Nielsen
//
//This program is free software; you can redistribute it and/or
//modify it under the terms of the GNU General Public License
//as published by the Free Software Foundation; either
//version 2 of the License, or (at your option) any later version.
//
//This program is distributed in the hope that it will be useful,
//but WITHOUT ANY WARRANTY; without even the implied warranty of
//MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
//GNU General Public License for more details.
//
//You should have received a copy of the GNU General Public License
//along with this program; if not, write to the Free Software
//Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#include "Glob.h"
/////////////////////////////////////////////////////////////////////
// Member Function : FileTimeToUnixTime
// Type : Global
// Parameters : pft - ?
// t - ?
// Returns : void
// Description :
//
// References : -
// Remarks : -
// Created : 030826, 26 august 2003
// Developer : KN
/////////////////////////////////////////////////////////////////////
void FileTimeToUnixTime(LPFILETIME pft, time_t* t) {
LONGLONG ll = ((LONGLONG)pft->dwHighDateTime) << 32;
ll = ll + pft->dwLowDateTime - 116444736000000000;
*t = (time_t)(ll / 10000000);
}
/////////////////////////////////////////////////////////////////////
// Function : nUnescapedURI
// Type : Local
// Parameters : pszURI - URI to convert buffer is modifyed !!
// Returns : length of the Unescaped URI
// Description :
//
// References : -
// Remarks : -
// Created : 030911, 11 september 2003
// Developer : KN
/////////////////////////////////////////////////////////////////////
static int nUnescapedURI(TCHAR *pszURI) {
if (! pszURI)
return 0;
TCHAR *pszOrigURI = pszURI;
int sumb;
int more = -1;
TCHAR *pszCurInsert = pszURI;
for (; *pszURI && pszURI[0] != ' ' ; pszURI++) {
int nNewChar;
if (pszURI[0] == '%') {
// we need to unescape the char
if (pszURI[1] == NULL || pszURI[2] == NULL || pszURI[1] == ' ' || pszURI[2] == ' ')
break; // handles error like "%2 " or "%a"
if (_stscanf(&pszURI[1], _T("%2X"), &nNewChar) != 1) {
// can't convert to number
pszURI += 2;
continue; // we skip it !!!
}
pszURI += 2;
}
/*else if( pszURI[0] == '+' )
{
nNewChar = ' ';
}*/
else
nNewChar = (int)tolower(pszURI[0]); // also make the request lowercase
if (nNewChar == '\\')
nNewChar = '/';
if ((nNewChar & 0x80) == 0x00) { // 0xxxxxxx (yields 7 bits)
more = -1; // this is to avoid sequence like %C3k%A6 will end up beeing "kę" insted it will be "k"
*pszCurInsert = (char)nNewChar;
pszCurInsert++;
continue;
}
if ((nNewChar & 0xc0) == 0x80) { // 10xxxxxx (continuation byte)
sumb = (sumb << 6) | (nNewChar & 0x3f) ; // Add 6 bits to sumb
more--;
if (more == 0) {
*pszCurInsert = (char)sumb; // here we just throw away all the fine UTF-8 encoding
pszCurInsert++;
}
} else if ((nNewChar & 0xe0) == 0xc0) { // 110xxxxx (yields 5 bits)
sumb = nNewChar & 0x1f;
more = 1; // Expect 1 more byte
} else if ((nNewChar & 0xf0) == 0xe0) { // 1110xxxx (yields 4 bits)
sumb = nNewChar & 0x0f;
more = 2; // Expect 2 more bytes
} else if ((nNewChar & 0xf8) == 0xf0) { // 11110xxx (yields 3 bits)
sumb = nNewChar & 0x07;
more = 3; // Expect 3 more bytes
} else if ((nNewChar & 0xfc) == 0xf8) { // 111110xx (yields 2 bits)
sumb = nNewChar & 0x03;
more = 4; // Expect 4 more bytes
} else /*if ((nNewChar & 0xfe) == 0xfc)*/
{ // 1111110x (yields 1 bit)
sumb = nNewChar & 0x01;
more = 5; // Expect 5 more bytes
}
}
return (pszCurInsert - pszOrigURI);
}
/////////////////////////////////////////////////////////////////////
// Member Function : CLHttpUser
// Type : Private / Public / Protected
// Parameters : hCon - ?
// stAdd - ?
// pcl - ?
// Returns :
// Description :
//
// References : -
// Remarks : -
// Created : 030918, 18 september 2003
// Developer : KN
/////////////////////////////////////////////////////////////////////
CLHttpUser::CLHttpUser(HANDLE hCon, in_addr stAdd) : CLShareUser(hCon, stAdd) {
memset(apszParam, 0, sizeof(apszParam));
hFile = INVALID_HANDLE_VALUE;
}
/////////////////////////////////////////////////////////////////////
// Member Function : CLHttpUser
// Type : Private / Public / Protected
// Parameters : None
// Returns :
// Description :
//
// References : -
// Remarks : -
// Created : 031124, 24 november 2003
// Developer : KN
/////////////////////////////////////////////////////////////////////
CLHttpUser::~CLHttpUser() {
if (hFile != INVALID_HANDLE_VALUE)
CloseHandle(hFile);
}
/////////////////////////////////////////////////////////////////////
// Member Function : bReadGetParameters
// Type : Private / Public / Protected
// Parameters : pszRequest - ?
// Returns : Returns true if
// Description :
//
// References : -
// Remarks : -
// Created : 030928, 28 september 2003
// Developer : KN
/////////////////////////////////////////////////////////////////////
bool CLHttpUser::bReadGetParameters(TCHAR *pszRequest) {
bool bRet = true;
for (; *pszRequest ; pszRequest++) {
if (pszRequest[0] != '\n') {
if (pszRequest[0] == '\r')
pszRequest[0] = 0;
continue;
}
pszRequest[0] = 0;
pszRequest++;
for (int nCur = 0; nCur < eLastParam ; nCur++) {
int nLen = (int)_tcslen(szParmStr[nCur]);
if (_tcsncmp(pszRequest, szParmStr[nCur], nLen) == 0) {
if (apszParam[nCur]) {
bRet = false;
// already set !!
} else {
pszRequest += nLen;
apszParam[nCur] = pszRequest;
pszRequest += _tcscspn(pszRequest, _T("\r\n")) - 1;
TCHAR *psz = pszRequest;
while (*psz == ' ') {
psz[0] = 0;// owerwrite ' ' or '\r' or '\n'
psz--;
}
}
break;
}
}
}
return bRet;
}
/////////////////////////////////////////////////////////////////////
// Member Function : SendError
// Type : Private / Public / Protected
// Parameters : iErrorCode - ?
// pszError - ?
// pszDescription - ?
// Returns : void
// Description :
//
// References : -
// Remarks : -
// Created : 031124, 24 november 2003
// Developer : KN, Houdini
// Changed : 21 January 2006 by Vampik
/////////////////////////////////////////////////////////////////////
void CLHttpUser::SendError(int iErrorCode, const TCHAR *pszError, const TCHAR *pszDescription)
{
TCHAR szCurTime[100];
time_t ltime;
time(<ime);
_tcsftime(szCurTime, SIZEOF(szCurTime), _T("%a, %d %b %Y %H:%M:%S GMT"), gmtime(<ime));
if (!pszDescription)
pszDescription = pszError;
TCHAR szBuf[1000];
DWORD dwBytesToWrite = mir_sntprintf(szBuf, SIZEOF(szBuf),
_T("HTTP/1.1 %i %s\r\n")
_T("Date: %s\r\n")
_T("Server: MirandaWeb/%s\r\n")
_T("Transfer-Encoding: chunked\r\n")
_T("Content-Type: text/html; charset=iso-8859-1\r\n")
_T("\r\n")
_T("10f\r\n")
_T("\n")
_T("
\n")
_T("%i %s\n")
_T("\n")
_T("%s
\n")
_T("%s\n")
_T("
\n")
_T("MirandaWeb/%s\n")
_T("\n")
_T("\r\n")
_T("\r\n"),
iErrorCode, pszError, szCurTime, PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM), iErrorCode, pszError, pszError, pszDescription, PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM));
Netlib_Send(hConnection, _T2A(szBuf), dwBytesToWrite, 0);
}
/////////////////////////////////////////////////////////////////////
// Member Function : SendRedir
// Type : Private / Public / Protected
// Parameters : iErrorCode - ?
// pszError - ?
// pszDescription - ?
// pszRedirect - ?
// Returns : void
// Description :
//
// References : -
// Remarks : -
// Created : 21 january 2006
// Developer : KN, Houdini, Vampik
/////////////////////////////////////////////////////////////////////
void CLHttpUser::SendRedir(int iErrorCode, const TCHAR *pszError, const TCHAR *pszDescription, const TCHAR *pszRedirect)
{
TCHAR szCurrTime[100];
time_t ltime;
time(<ime);
_tcsftime(szCurrTime, SIZEOF(szCurrTime), _T("%a, %d %b %Y %H:%M:%S GMT"), gmtime(<ime));
if (!pszDescription)
pszDescription = pszError;
TCHAR szBuff[1000];
DWORD dwBytesToWrite = mir_sntprintf(szBuff, SIZEOF(szBuff),
_T("HTTP/1.1 %i %s\r\n")
_T("Date: %s\r\n")
_T("Server: MirandaWeb/%s\r\n")
_T("Location: %s/\r\n")
_T("Transfer-Encoding: chunked\r\n")
_T("Content-Type: text/html; charset=iso-8859-1\r\n")
_T("\r\n")
_T("10f\r\n")
_T("\n")
_T("\n")
_T("%i %s\n")
_T("\n")
_T("%s
\n")
_T("%s\n")
_T("
\n")
_T("MirandaWeb/%s\n")
_T("\n")
_T("\r\n")
_T("\r\n"),
iErrorCode, pszError, szCurrTime, PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM), pszRedirect, iErrorCode, pszError, pszError, pszDescription, PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM));
Netlib_Send(hConnection, _T2A(szBuff), dwBytesToWrite, 0);
}
/////////////////////////////////////////////////////////////////////
// Function : strmcat
// Type : Local
// Parameters : pszDest - ?
// pszSrc - ?
// MaxLength - ?
// Returns :
// Description : Fills pszDest until it is MaxLength long or
// pszSrc is finished - always appends a \0
// References : -
// Remarks : -
// Created :
// Developer : Houdini
/////////////////////////////////////////////////////////////////////
static void _tcsmcat(TCHAR *pszDest, const TCHAR *pszSrc, int iMaxLength)
{
int iLength = 0;
while (*pszDest != '\0') {
pszDest++;
iLength++;
}
while (iLength < iMaxLength - 1 && *pszSrc != '\0') {
*pszDest = *pszSrc;
pszDest++;
pszSrc++;
iLength++;
}
*pszDest = '\0';
}
/////////////////////////////////////////////////////////////////////
// Member Function : bProcessGetRequest
// Type : Global
// Parameters : hNewConnection - ?
// dwRemoteIP - ?
// pszRequest - ?
// Returns : Returns true if a file was send
// Description :
//
// References : -
// Remarks : -
// Created : 030813, 13 august 2003
// Developer : KN, Houdini
// Changed : 27 January 2005 by Vampik
// Changed : 21 January 2006 by Vampik
/////////////////////////////////////////////////////////////////////
bool CLHttpUser::bProcessGetRequest(TCHAR *pszRequest, bool bIsGetCommand) {
//LogEvent("Request", pszRequest);
int nUriLength = nUnescapedURI(pszRequest);
if (nUriLength <= 0)
return false;
CLFileShareListAccess clCritSection;
if (bShutdownInProgress)
return false;
static TCHAR szTempfile[MAX_PATH+1];
szTempfile[0] = '\0';
if (!bReadGetParameters(pszRequest)) {
SendError(400, _T("Bad Request"));
return false;
}
DWORD dwRemoteIP = ntohl(stAddr.S_un.S_addr);
for (CLFileShareNode * pclCur = pclFirstNode; pclCur ; pclCur = pclCur->pclNext) {
if ((pclCur->st.dwAllowedIP ^ dwRemoteIP) & pclCur->st.dwAllowedMask)
continue; // Not an allowed IP address
if (!pclCur->bIsDirectory() && pclCur->nGetSrvPathLen() != nUriLength)
continue; // not the right length, quickly move on to the next.
if (pclCur->bIsDirectory() ?
(_tcsncmp(pclCur->st.pszSrvPath, pszRequest, pclCur->nGetSrvPathLen() - 1) == 0) :
(_tcsncmp(pclCur->st.pszSrvPath, pszRequest, pclCur->nGetSrvPathLen()) == 0)) {
/*OutputDebugString( "Request for file OK : ");
OutputDebugString( pclCur->st.pszSrvPath );
OutputDebugString( "\n" );*/
static TCHAR szSrvPath[MAX_PATH+1];
static TCHAR szRealPath[MAX_PATH+1];
TCHAR *pszSrvPath = pclCur->st.pszSrvPath;
TCHAR *pszRealPath = pclCur->st.pszRealPath;
if (pclCur->bIsDirectory()) {
_tcscpy(szRealPath, pclCur->st.pszRealPath);
_tcscpy(szSrvPath, pclCur->st.pszSrvPath);
pszRealPath = szRealPath;
pszSrvPath = szSrvPath;
if (nUriLength > MAX_PATH)
nUriLength = MAX_PATH;
pszRequest[nUriLength] = '\0';
if (pclCur->nGetSrvPathLen() - nUriLength == 1) {
SendRedir(302, _T("Found"), _T("The document has moved"), pszRequest);
return false;
} else {
_tcsmcat(pszRealPath, &pszRequest[pclCur->nGetSrvPathLen()], MAX_PATH);
_tcsmcat(pszSrvPath, &pszRequest[pclCur->nGetSrvPathLen()], MAX_PATH);
}
pszRequest[nUriLength] = ' ';
// hacker protection - should be removed by the browser
if (_tcsstr(pszRealPath, _T(".."))) {
SendError(404, _T("Not Found"), _T("The requested URL was not found on this server."));
return false;
}
TCHAR *pszTmp = pszRealPath;
while (pszTmp = _tcschr(pszTmp, '/'))
* pszTmp = '\\';
hFile = CreateFile(pszRealPath, GENERIC_READ ,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (hFile == INVALID_HANDLE_VALUE) {
if (pszSrvPath[_tcslen(pszSrvPath)-1] != '/') {
_tcsmcat(pszRealPath, _T("\\"), MAX_PATH);
_tcsmcat(pszSrvPath, _T("/"), MAX_PATH);
}
// a directory with index.htm
_tcsmcat(szRealPath, _T("index.htm"), MAX_PATH);
hFile = CreateFile(pszRealPath, GENERIC_READ ,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_HIDDEN, NULL);
if (hFile == INVALID_HANDLE_VALUE) {
// a directory with index.html
_tcsmcat(szRealPath, _T("l"), MAX_PATH);
hFile = CreateFile(pszRealPath, GENERIC_READ ,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_HIDDEN, NULL);
if (hFile == INVALID_HANDLE_VALUE) {
// generate directory index in temporary file
if (*szTempfile == '\0') {
GetTempPath(MAX_PATH, szTempfile);
_tcsmcat(szTempfile, _T("\\HttpServerTemp"), MAX_PATH);
_tcsmcat(szTempfile, pszSrvPath, MAX_PATH);
TCHAR *pszTmp = szTempfile;
while (pszTmp = _tcschr(pszTmp, '/'))
* pszTmp = '~';
}
pszRealPath[_tcslen(pszRealPath) - 10] = '\0';
// detecting browser function removed
// every browser should support it by now
bool BrowserSupportsXML = true;
//(apszParam[eUserAgent] != NULL) &&
// (strstr(apszParam[eUserAgent], "Firefox") ||
// (strstr(apszParam[eUserAgent], "MSIE") && !strstr(apszParam[eUserAgent], "Opera")));
if ((indexCreationMode == INDEX_CREATION_XML ||
(indexCreationMode == INDEX_CREATION_DETECT && BrowserSupportsXML)) &&
bCreateIndexXML(pszRealPath, szTempfile, pszSrvPath, dwRemoteIP)) {
hFile = CreateFile(szTempfile, GENERIC_READ ,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
_tcscpy(szRealPath, _T("a.xml")); // restore .xml for mime type
} else if ((indexCreationMode == INDEX_CREATION_HTML ||
indexCreationMode == INDEX_CREATION_DETECT) &&
bCreateIndexHTML(pszRealPath, szTempfile, pszSrvPath, dwRemoteIP)) {
hFile = CreateFile(szTempfile, GENERIC_READ,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
_tcscpy(szRealPath, _T("a.html")); // restore .html for mime type
} else {
continue;
}
} else {
_tcsmcat(pszSrvPath, _T("index.html"), MAX_PATH);
szTempfile[0] = '\0';
}
} else {
_tcsmcat(pszSrvPath, _T("index.htm"), MAX_PATH);
szTempfile[0] = '\0';
}
} else {
szTempfile[0] = '\0';
}
} else {
hFile = CreateFile(pszRealPath, GENERIC_READ ,
FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (hFile == INVALID_HANDLE_VALUE) {
SendError(404, _T("Not Found"), _T("HTTP server failed to open local file"));
return false;
}
}
_tcscpy(this->szCurrentDLSrvPath, pszSrvPath);
DWORD nDataSize = GetFileSize(hFile, NULL);
dwTotalSize = nDataSize;
FILETIME stFileTime;
GetFileTime(hFile, NULL, NULL, &stFileTime);
TCHAR szCurTime[100];
time_t ltime;
time(<ime);
_tcsftime(szCurTime, SIZEOF(szCurTime), _T("%a, %d %b %Y %H:%M:%S GMT"), gmtime(<ime));
TCHAR szFileTime[100];
FileTimeToUnixTime(&stFileTime, <ime);
_tcsftime(szFileTime, SIZEOF(szFileTime), _T("%a, %d %b %Y %H:%M:%S GMT"), gmtime(<ime));
if (apszParam[eIfModifiedSince] && _tcscmp(apszParam[eIfModifiedSince], szFileTime) == 0) {
SendError(304, _T("Not Modified"));
return true;
}
// we found match send file !!
if (bIsGetCommand) {
if (! pclCur->bAddUser(this)) {
SendError(403, _T("Forbidden"), _T("Access has been denied because there are too many connections"));
return false;
}
}
if (*(ULONG*)(&stAddr) != 0x0100007F && // do not show popup of 127.0.0.1
_tcsstr(pszRealPath, _T("\\@")) == NULL) { // and of shares which start with an @
ShowPopupWindow(_A2T(inet_ntoa(stAddr)), pszSrvPath);
}
clCritSection.Unlock();
DWORD dwFileStart = 0;
DWORD dwDataToSend = nDataSize;
TCHAR szETag[50];
{
int nETagLen = mir_sntprintf(szETag, SIZEOF(szETag), _T("\"%x-%x-%x\""),
nDataSize, stFileTime.dwHighDateTime, stFileTime.dwLowDateTime);
if (!apszParam[eIfRange] || (_tcsncmp(szETag, apszParam[eIfRange], nETagLen) == 0)) {
TCHAR *pszRange = apszParam[eRange];
if (pszRange) {
if (_tcsncmp(pszRange, _T("bytes="), _tcslen(_T("bytes="))) == 0) {
pszRange += 6;
// Do resume !!!
TCHAR *pszEnd;
if (pszRange[0] == '-') {
// its a suffix-byte-range-spec
DWORD dwLen = _tcstol(pszRange + 1, &pszEnd, 10);
if (dwLen < nDataSize)
dwFileStart = nDataSize - dwLen;
} else {
DWORD dwLen = _tcstol(pszRange, &pszEnd, 10);
if (*pszEnd == '-' && dwLen < nDataSize) {
dwFileStart = dwLen;
pszRange = pszEnd + 1;
if (*pszRange != 0) {
dwLen = _tcstol(pszEnd + 1, &pszEnd, 10);
if (dwLen > dwFileStart)
dwDataToSend = (dwLen + 1) - dwFileStart;
else
dwFileStart = 0;
}
} else {
SendError(400, _T("Bad Request"));
return false;
}
}
}
}
}
}
if (dwFileStart >= nDataSize)
dwFileStart = 0;
if (dwFileStart + dwDataToSend >= nDataSize)
dwDataToSend = nDataSize - dwFileStart;
DWORD dwBytesToWrite = 0;
// To optimize send speed it it ideal to always send larges size packages
// But this size depended on network media but on Ethernet it is 1518 bytes.
// Ethernet, IP and TCP headers use some of this space and leaves 1460 bytes
// for data transfer.
// We will use a multiply of this to always send optimal sized packages.
TCHAR szBuf[1460 * 4];
if (dwFileStart > 0 || dwDataToSend != nDataSize) {
if (SetFilePointer(hFile, dwFileStart, NULL, FILE_BEGIN) == INVALID_SET_FILE_POINTER) {
SendError(416, _T("Requested Range Not Satisfiable"));
return true;
}
const TCHAR szHttpPartial[] = _T("HTTP/1.1 206 Partial Content\r\n")
_T("Connection: Keep-Alive\r\n")
_T("Date: %s\r\n")
_T("Server: MirandaWeb/%s\r\n")
_T("Accept-Ranges: bytes\r\n")
_T("ETag: %s\r\n")
_T("Content-Length: %d\r\n")
_T("Content-Type: %s\r\n")
_T("Content-Range: bytes %d-%d/%d\r\n")
_T("Last-Modified: %s\r\n")
_T("\r\n");
dwBytesToWrite = mir_sntprintf(szBuf, SIZEOF(szBuf), szHttpPartial,
szCurTime,
PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM),
szETag,
dwDataToSend,
pszGetMimeType(pszRealPath),
dwFileStart,
(dwFileStart + dwDataToSend - 1),
nDataSize,
szFileTime);
} else {
const TCHAR szHttpOk[] = _T("HTTP/1.1 200 OK\r\n")
_T("Connection: Keep-Alive\r\n")
_T("Date: %s\r\n")
_T("Server: MirandaWeb/%s\r\n")
_T("Accept-Ranges: bytes\r\n")
_T("ETag: %s\r\n")
_T("Content-Length: %d\r\n")
_T("Content-Type: %s\r\n")
_T("Last-Modified: %s\r\n")
_T("\r\n");
dwBytesToWrite = mir_sntprintf(szBuf, SIZEOF(szBuf), szHttpOk,
szCurTime,
PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM),
szETag,
nDataSize,
pszGetMimeType(pszRealPath),
szFileTime);
}
Netlib_Send(hConnection, _T2A(szBuf), dwBytesToWrite, 0);
if (bIsGetCommand) {
static int nThreadCount = 0;
nThreadCount++;
DWORD dwLastUpdate = GetTickCount();
DWORD dwLastCurrentDL = 0;
/*
dwLastDLSTickCount = dwCurTick;
dwCurrentDL;*/
//DWORD dwMaxSpeed = 8192;// Byte pr Sek
//DWORD dwMaxSpeed = 20000;// Byte pr Sek
//DWORD dwMaxSpeed = 163840;// Byte pr Sek
//DWORD dwMaxSpeed = 4096;// Byte pr Sek
DWORD dwLastResetTime = GetTickCount();
int nMaxBytesToSend = nMaxUploadSpeed / nThreadCount;
for (;;) {
{
DWORD dwCurTick = GetTickCount();
if (dwCurTick - dwLastUpdate >= 1000) {
dwSpeed = ((dwCurrentDL - dwLastCurrentDL) * 1000) / (dwCurTick - dwLastUpdate);
dwLastUpdate = dwCurTick;
dwLastCurrentDL = dwCurrentDL;
}
}
if (nMaxUploadSpeed == 0) {
Sleep(1000);
continue;
}
bool bSpeedLimit = (nMaxUploadSpeed >= 0) && (bIsOnline || !bLimitOnlyWhenOnline);
DWORD dwCurOpr = SIZEOF(szBuf);
if (bSpeedLimit)
dwCurOpr = min(nMaxBytesToSend, SIZEOF(szBuf));
if (!ReadFile(hFile, szBuf, dwCurOpr, &dwBytesToWrite, NULL))
break;
if (dwBytesToWrite <= 0)
break;
if (dwCurrentDL + dwBytesToWrite > dwDataToSend)
dwBytesToWrite = dwDataToSend - dwCurrentDL;
if (bSpeedLimit)
nMaxBytesToSend -= dwBytesToWrite;
DWORD dwSend = Netlib_Send(hConnection, _T2A(szBuf), dwBytesToWrite, MSG_NODUMP);
if (dwSend == SOCKET_ERROR)
break;
dwCurrentDL += dwSend;
if (dwSend != dwBytesToWrite)
break;
if (dwCurrentDL >= dwDataToSend)
break;
if (bSpeedLimit && nMaxBytesToSend <= 0) {
// we have reached the limmit
DWORD dwTimeUsed = GetTickCount() - dwLastResetTime;
if (dwTimeUsed < 1000)
Sleep(1000 - dwTimeUsed);
dwLastResetTime = GetTickCount();
nMaxBytesToSend = nMaxUploadSpeed / nThreadCount;
}
}
// file is always closed in destructor
if (szTempfile[0] != '\0') {
// and here - since it is a temporary index which as to be deleted
CloseHandle(hFile);
hFile = INVALID_HANDLE_VALUE;
DeleteFile(szTempfile);
}
clCritSection.Lock();
nThreadCount--;
bool bNeedToWriteConfig = false;
if (dwCurrentDL == nDataSize) {
if (pclCur->st.nMaxDownloads > 0) {
pclCur->st.nMaxDownloads--;
bNeedToWriteConfig = true;
}
}
pclCur->bRemoveUser(this);
// nMaxDownloads can have been decreesed by another thread.
// Therefore we test it even if we did'en decreese it
if (pclCur->st.nMaxDownloads == 0 && !pclCur->bAnyUsers()) {
CLFileShareNode **pclPrev = &pclFirstNode;
for (CLFileShareNode * pcl = pclFirstNode ; pcl ; pcl = pcl->pclNext) {
if (pcl == pclCur) {
*pclPrev = pclCur->pclNext;
ShowPopupWindow(TranslateT("Share removed"), pclCur->st.pszSrvPath, RGB(255, 189, 189));
delete pclCur;
bNeedToWriteConfig = true;
break;
}
pclPrev = &pcl->pclNext;
}
}
if (bNeedToWriteConfig) {
bWriteConfigurationFile();
}
}
return true;
}
}
#ifdef _DEBUG
OutputDebugString(_T("########### Request Failed ###########\n"));
OutputDebugString(pszRequest);
#endif
pszRequest[nUriLength] = 0;
if ((nUriLength != 12 || _tcsncmp(pszRequest, _T("/favicon.ico"), nUriLength)) && // do not show popup of favicon.ico
*(ULONG*)(&stAddr) != 0x0100007F) { // do not show popup of 127.0.0.1
ShowPopupWindow(_A2T(inet_ntoa(stAddr)), pszRequest, RGB(255, 189, 189));
}
SendError(404, _T("Not Found"), _T("The requested URL was not found on this server."));
return false;
}
void CLHttpUser::HandleNewConnection()
{
/*
{
SOCKET s = CallService(MS_NETLIB_GETSOCKET, (WPARAM) hConnection, 0);
sockaddr_in MyAddr;
int nSize = sizeof( MyAddr );
getsockname( s, (sockaddr*)&MyAddr, &nSize );
ShowPopupWindow( "My IP address", inet_ntoa( MyAddr.sin_addr ) );
//OutputDebugString( );
}*/
/*
{
LINGER li;
int nLenght = sizeof( li );
int ret = getsockopt( s, IPPROTO_TCP, SO_LINGER, (char *)&li, &nLenght );
if( ret )
{
DWORD error = WSAGetLastError();
if( error
WSANOTINITIALISED
WSAENETDOWN The network subsystem has failed.
WSAEFAULT One of the optval or the optlen parameters is not a valid part of the user address space, or the optlen parameter is too small.
WSAEINPROGRESS A blocking Windows Sockets 1.1 call is in progress, or the service provider is still processing a callback function.
WSAEINVAL The level parameter is unknown or invalid.
WSAENOPROTOOPT The option is unknown or unsupported by the indicated protocol family.
WSAENOTSOCK
}
nLenght = sizeof( li );
li.l_onoff = 1;
li.l_linger = 0;// time is default
ret = setsockopt( s, IPPROTO_TCP, SO_LINGER, (const char *)&li, nLenght );
if( ret )
{
// error
}
int nLenght = sizeof( li );
int ret = getsockopt( s, IPPROTO_TCP, SO_LINGER, (char *)&li, &nLenght );
}
*/
TCHAR szBuf[1000];
int nCurPos = 0;
while (SIZEOF(szBuf) - nCurPos > 10 && !bShutdownInProgress) {
int nBytesRead = Netlib_Recv(hConnection, _T2A(&szBuf[nCurPos]), SIZEOF(szBuf) - nCurPos, 0);
if (! nBytesRead) {
// socket closed gracefully
break;
}
if (nBytesRead == SOCKET_ERROR) {
// socket closed with error
// WSAGetLastError();
break;
}
int nOldCurPos = nCurPos;
nCurPos += nBytesRead;
if (nCurPos <= 5)
continue;
bool bIsGetCommand = _tcsncmp(szBuf, _T("GET "), _tcslen(_T("GET "))) == 0;
if (!bIsGetCommand && _tcsncmp(szBuf, _T("HEAD "), _tcslen(_T("HEAD "))) != 0) {
SendError(501, _T("Not Implemented"));
break; // We only support GET and HEAD commands !!
}
if (nOldCurPos < 4)
nOldCurPos = 4; // scan forward from end of "GET " to locate the next space ' '
bool bBreakWhile = false;
for (; nOldCurPos < nCurPos; nOldCurPos++) {
if (szBuf[nOldCurPos-2] == '\n' && szBuf[nOldCurPos-1] == '\r' && szBuf[nOldCurPos] == '\n') {
// we have a walid request !!! scan to see if we have this file
szBuf[nOldCurPos] = NULL;
bProcessGetRequest(&szBuf[bIsGetCommand?4:5], bIsGetCommand);
bBreakWhile = true;
break;
}
}
if (bBreakWhile)
break;
}
}