blob: c7b8898f537bba70ac29db869919efcf2b1cf826 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/**
* Copyright (c) 2013-2017, Regents of the University of California.
*
* This file is part of ChronoShare, a decentralized file sharing application over NDN.
*
* ChronoShare 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 3 of the License, or (at your option) any later version.
*
* ChronoShare 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 copies of the GNU General Public License along with
* ChronoShare, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*
* See AUTHORS.md for complete list of ChronoShare authors and contributors.
*/
//
// request_handler.cpp
// ~~~~~~~~~~~~~~~~~~~
//
// Copyright (c) 2003-2012 Christopher M. Kohlhoff (chris at kohlhoff dot com)
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
//
#include "request_handler.hpp"
#include "mime_types.hpp"
#include "reply.hpp"
#include "request.hpp"
#include "core/logging.hpp"
#include <fstream>
#include <sstream>
#include <string>
#include <boost/lexical_cast.hpp>
#include <QDataStream>
#include <QFile>
#include <QIODevice>
#include <QString>
namespace http {
namespace server {
_LOG_INIT(HttpServer);
request_handler::request_handler(const std::string& doc_root)
: doc_root_(doc_root.c_str())
{
}
void
request_handler::handle_request(const request& req, reply& rep)
{
// Decode url to path.
std::string request_path;
if (!url_decode(req.uri, request_path)) {
rep = reply::stock_reply(reply::bad_request);
return;
}
// Request path must be absolute and not contain "..".
if (request_path.empty() || request_path[0] != '/' || request_path.find("..") != std::string::npos) {
rep = reply::stock_reply(reply::bad_request);
return;
}
// If path ends in slash (i.e. is a directory) then add "index.html".
if (request_path[request_path.size() - 1] == '/') {
request_path += "index.html";
}
// Determine the file extension.
std::size_t last_slash_pos = request_path.find_last_of("/");
std::size_t last_dot_pos = request_path.find_last_of(".");
std::string extension;
if (last_dot_pos != std::string::npos && last_dot_pos > last_slash_pos) {
extension = request_path.substr(last_dot_pos + 1);
}
// Open the file to send back.
// The following is a hack to make the server understands Qt's
// resource system, so that the html resources can be managed using Qt's
// resource system (e.g. no need to worry about the location of html)
// in Mac OS, it will be inside the bundle, in Linux, perhaps somewhere
// in /usr/local/share
QString full_path = doc_root_.absolutePath() + QString(request_path.c_str());
QFile file(full_path);
if (!file.exists() || !file.open(QIODevice::ReadOnly)) {
rep = reply::stock_reply(reply::not_found);
return;
}
_LOG_DEBUG("Serving file: " << request_path);
// Fill out the reply to be sent to the client.
rep.status = reply::ok;
char buf[512];
QDataStream in(&file);
while (true) {
int bytes = in.readRawData(buf, sizeof(buf));
if (bytes > 0) {
rep.content.append(buf, bytes);
}
else {
break;
}
}
rep.headers.resize(2);
rep.headers[0].name = "Content-Length";
rep.headers[0].value = boost::lexical_cast<std::string>(rep.content.size());
rep.headers[1].name = "Content-Type";
rep.headers[1].value = mime_types::extension_to_type(extension);
}
bool
request_handler::url_decode(const std::string& in, std::string& out)
{
out.clear();
out.reserve(in.size());
for (std::size_t i = 0; i < in.size(); ++i) {
if (in[i] == '%') {
if (i + 3 <= in.size()) {
int value = 0;
std::istringstream is(in.substr(i + 1, 2));
if (is >> std::hex >> value) {
out += static_cast<char>(value);
i += 2;
}
else {
return false;
}
}
else {
return false;
}
}
else if (in[i] == '+') {
out += ' ';
}
else {
out += in[i];
}
}
return true;
}
} // namespace server
} // namespace http