feat: cgi headers: enhance CGI handling with output parsing and buffer management

This commit is contained in:
whaffman 2025-10-19 13:54:19 +02:00
parent 60405f03d5
commit 0c081bbb2c
3 changed files with 72 additions and 4 deletions

View File

@ -1,6 +1,10 @@
<?php
// CGI Capabilities Demonstration Script
header("Content-Type: text/html; charset=UTF-8");
header("Cache-Control: no-cache, no-store, must-revalidate");
header("Pragma: no-cache");
header("Expires: 0");
header("X-Content-Type-Options: nosniff");
?>
<!DOCTYPE html>
<html lang="en">

View File

@ -5,6 +5,7 @@
#include <webserv/http/HttpResponse.hpp> // for HttpResponse
#include <webserv/log/Log.hpp> // for Log
#include <webserv/socket/CgiSocket.hpp> // for CgiSocket
#include <webserv/utils/utils.hpp> // for stoul
CgiHandler::CgiHandler(const HttpRequest &request, HttpResponse &response)
: AHandler(request, response), cgiProcess_(nullptr), cgiStdIn_(nullptr), cgiStdOut_(nullptr)
@ -19,7 +20,6 @@ void CgiHandler::handle()
// Initialize CGI process
cgiProcess_ = std::make_unique<CgiProcess>(request_, *this);
Log::info("CGI process started and sockets registered");
}
@ -72,14 +72,13 @@ void CgiHandler::read()
Log::info("CGI process closed stdout, fd: " + std::to_string(cgiStdOut_->getFd()));
request_.getClient().removeCgiSocket(cgiStdOut_.get());
cgiStdOut_ = nullptr;
response_.addHeader("Content-Type", "text/html");
response_.setComplete();
parseCgiOutput();
return;
}
else
{
buffer[bytesRead] = '\0'; // NOLINT(cppcoreguidelines-pro-bounds-constant-array-index)
response_.appendBody(std::string(buffer, static_cast<size_t>(bytesRead)));
appendToBuffer(buffer, static_cast<size_t>(bytesRead));
Log::debug("Read " + std::to_string(bytesRead)
+ " bytes from CGI stdout, fd: " + std::to_string(cgiStdOut_->getFd()));
}
@ -109,5 +108,63 @@ void CgiHandler::wait() noexcept
void CgiHandler::setPid(int pid)
{
pid_ = pid;
}
void CgiHandler::parseCgiOutput()
{
Log::trace(LOCATION);
// Parse the headers from the buffer
size_t headerEnd = std::string(buffer_.begin(), buffer_.end()).find("\r\n\r\n");
if (headerEnd == std::string::npos)
{
Log::debug("CGI output headers not complete yet");
return;
}
// Parse the headers
std::string headers(buffer_.begin(), buffer_.begin() + static_cast<long>(headerEnd));
Log::debug("CGI output headers: " + headers);
parseCgiHeaders(headers);
buffer_.erase(buffer_.begin(), buffer_.begin() + static_cast<long>(headerEnd) + 4);
parseCgiBody();
}
void CgiHandler::parseCgiHeaders(std::string &headers)
{
Log::trace(LOCATION);
size_t start = 0;
size_t end = headers.find("\r\n");
while (end != std::string::npos)
{
std::string header = headers.substr(start, end - start);
Log::debug("CGI header: " + header);
size_t colonPos = header.find(':');
if (colonPos != std::string::npos)
{
std::string name = header.substr(0, colonPos);
std::string value = header.substr(colonPos + 1);
name = utils::trim(name);
value = utils::trim(value);
response_.addHeader(name, value);
}
start = end + 2;
end = headers.find("\r\n", start);
}
}
void CgiHandler::parseCgiBody()
{
Log::trace(LOCATION);
// Append the body to the response
response_.appendBody(buffer_);
response_.setComplete();
buffer_.clear();
}
void CgiHandler::appendToBuffer(const char *data, size_t length)
{
Log::trace(LOCATION);
buffer_.insert(buffer_.end(), data, data + length);
}

View File

@ -26,9 +26,16 @@ class CgiHandler : public AHandler
private:
constexpr static size_t bufferSize_ = 8192; // TODO: remove duplicate definition and move to configmanager
std::vector<uint8_t> buffer_;
std::unique_ptr<CgiProcess> cgiProcess_;
std::unique_ptr<CgiSocket> cgiStdIn_;
std::unique_ptr<CgiSocket> cgiStdOut_;
void parseCgiOutput();
void parseCgiHeaders(std::string &headers);
void parseCgiBody();
void appendToBuffer(const char *data, size_t length);
int pid_ = -1;