mirror of
https://github.com/f4exb/sdrangel.git
synced 2025-10-24 09:30:22 -04:00
188 lines
6.4 KiB
C++
188 lines
6.4 KiB
C++
/**
|
|
@file
|
|
@author Stefan Frings
|
|
*/
|
|
|
|
#include "staticfilecontroller.h"
|
|
#include <QFileInfo>
|
|
#include <QDir>
|
|
#include <QDateTime>
|
|
|
|
using namespace stefanfrings;
|
|
|
|
StaticFileController::StaticFileController(QSettings* settings, QObject* parent)
|
|
:HttpRequestHandler(parent)
|
|
{
|
|
maxAge=settings->value("maxAge","60000").toInt();
|
|
encoding=settings->value("encoding","UTF-8").toString();
|
|
docroot=settings->value("path",".").toString();
|
|
if(!(docroot.startsWith(":/") || docroot.startsWith("qrc://")))
|
|
{
|
|
// Convert relative path to absolute, based on the directory of the config file.
|
|
#ifdef Q_OS_WIN32
|
|
if (QDir::isRelativePath(docroot) && settings->format()!=QSettings::NativeFormat)
|
|
#else
|
|
if (QDir::isRelativePath(docroot))
|
|
#endif
|
|
{
|
|
QFileInfo configFile(settings->fileName());
|
|
docroot=QFileInfo(configFile.absolutePath(),docroot).absoluteFilePath();
|
|
}
|
|
}
|
|
qDebug("StaticFileController: docroot=%s, encoding=%s, maxAge=%i",qPrintable(docroot),qPrintable(encoding),maxAge);
|
|
maxCachedFileSize=settings->value("maxCachedFileSize","65536").toInt();
|
|
cache.setMaxCost(settings->value("cacheSize","1000000").toInt());
|
|
cacheTimeout=settings->value("cacheTime","60000").toInt();
|
|
qDebug("StaticFileController: cache timeout=%i, size=%i",cacheTimeout,cache.maxCost());
|
|
}
|
|
|
|
|
|
void StaticFileController::service(HttpRequest& request, HttpResponse& response)
|
|
{
|
|
QByteArray path=request.getPath();
|
|
// Check if we have the file in cache
|
|
qint64 now=QDateTime::currentMSecsSinceEpoch();
|
|
mutex.lock();
|
|
CacheEntry* entry=cache.object(path);
|
|
if (entry && (cacheTimeout==0 || entry->created>now-cacheTimeout))
|
|
{
|
|
QByteArray document=entry->document; //copy the cached document, because other threads may destroy the cached entry immediately after mutex unlock.
|
|
QByteArray filename=entry->filename;
|
|
mutex.unlock();
|
|
qDebug("StaticFileController: Cache hit for %s",path.data());
|
|
setContentType(filename,response);
|
|
response.setHeader("Cache-Control","max-age="+QByteArray::number(maxAge/1000));
|
|
response.write(document);
|
|
}
|
|
else
|
|
{
|
|
mutex.unlock();
|
|
// The file is not in cache.
|
|
qDebug("StaticFileController: Cache miss for %s",path.data());
|
|
// Forbid access to files outside the docroot directory
|
|
if (path.contains("/.."))
|
|
{
|
|
qWarning("StaticFileController: detected forbidden characters in path %s",path.data());
|
|
response.setStatus(403,"forbidden");
|
|
response.write("403 forbidden",true);
|
|
return;
|
|
}
|
|
// If the filename is a directory, append index.html.
|
|
if (QFileInfo(docroot+path).isDir())
|
|
{
|
|
path+="/index.html";
|
|
}
|
|
// Try to open the file
|
|
QFile file(docroot+path);
|
|
qDebug("StaticFileController: Open file %s",qPrintable(file.fileName()));
|
|
if (file.open(QIODevice::ReadOnly))
|
|
{
|
|
setContentType(path,response);
|
|
response.setHeader("Cache-Control","max-age="+QByteArray::number(maxAge/1000));
|
|
if (file.size()<=maxCachedFileSize)
|
|
{
|
|
// Return the file content and store it also in the cache
|
|
entry=new CacheEntry();
|
|
while (!file.atEnd() && !file.error())
|
|
{
|
|
QByteArray buffer=file.read(65536);
|
|
response.write(buffer);
|
|
entry->document.append(buffer);
|
|
}
|
|
entry->created=now;
|
|
entry->filename=path;
|
|
mutex.lock();
|
|
cache.insert(request.getPath(),entry,entry->document.size());
|
|
mutex.unlock();
|
|
}
|
|
else
|
|
{
|
|
// Return the file content, do not store in cache
|
|
while (!file.atEnd() && !file.error())
|
|
{
|
|
response.write(file.read(65536));
|
|
}
|
|
}
|
|
file.close();
|
|
}
|
|
else {
|
|
if (file.exists())
|
|
{
|
|
qWarning("StaticFileController: Cannot open existing file %s for reading",qPrintable(file.fileName()));
|
|
response.setStatus(403,"forbidden");
|
|
response.write("403 forbidden",true);
|
|
}
|
|
else
|
|
{
|
|
response.setStatus(404,"not found");
|
|
response.write("404 not found",true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void StaticFileController::setContentType(QString fileName, HttpResponse& response) const
|
|
{
|
|
if (fileName.endsWith(".png"))
|
|
{
|
|
response.setHeader("Content-Type", "image/png");
|
|
}
|
|
else if (fileName.endsWith(".jpg"))
|
|
{
|
|
response.setHeader("Content-Type", "image/jpeg");
|
|
}
|
|
else if (fileName.endsWith(".gif"))
|
|
{
|
|
response.setHeader("Content-Type", "image/gif");
|
|
}
|
|
else if (fileName.endsWith(".pdf"))
|
|
{
|
|
response.setHeader("Content-Type", "application/pdf");
|
|
}
|
|
else if (fileName.endsWith(".txt"))
|
|
{
|
|
response.setHeader("Content-Type", qPrintable("text/plain; charset="+encoding));
|
|
}
|
|
else if (fileName.endsWith(".html") || fileName.endsWith(".htm"))
|
|
{
|
|
response.setHeader("Content-Type", qPrintable("text/html; charset="+encoding));
|
|
}
|
|
else if (fileName.endsWith(".css"))
|
|
{
|
|
response.setHeader("Content-Type", "text/css");
|
|
}
|
|
else if (fileName.endsWith(".js"))
|
|
{
|
|
response.setHeader("Content-Type", "text/javascript");
|
|
}
|
|
else if (fileName.endsWith(".svg"))
|
|
{
|
|
response.setHeader("Content-Type", "image/svg+xml");
|
|
}
|
|
else if (fileName.endsWith(".woff"))
|
|
{
|
|
response.setHeader("Content-Type", "font/woff");
|
|
}
|
|
else if (fileName.endsWith(".woff2"))
|
|
{
|
|
response.setHeader("Content-Type", "font/woff2");
|
|
}
|
|
else if (fileName.endsWith(".ttf"))
|
|
{
|
|
response.setHeader("Content-Type", "application/x-font-ttf");
|
|
}
|
|
else if (fileName.endsWith(".eot"))
|
|
{
|
|
response.setHeader("Content-Type", "application/vnd.ms-fontobject");
|
|
}
|
|
else if (fileName.endsWith(".otf"))
|
|
{
|
|
response.setHeader("Content-Type", "application/font-otf");
|
|
}
|
|
// Todo: add all of your content types
|
|
else
|
|
{
|
|
qDebug("StaticFileController: unknown MIME type for filename '%s'", qPrintable(fileName));
|
|
}
|
|
}
|