You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

127 lines
3.2 KiB
C++

#ifndef WEBCC_HTTP_MESSAGE_H_
#define WEBCC_HTTP_MESSAGE_H_
#include <cassert>
#include <string>
#include <utility> // for move()
#include <vector>
#include "boost/asio/buffer.hpp" // for const_buffer
#include "webcc/common.h"
#include "webcc/http_file.h"
#include "webcc/globals.h"
namespace webcc {
class HttpMessage;
std::ostream& operator<<(std::ostream& os, const HttpMessage& message);
// Base class for HTTP request and response messages.
class HttpMessage {
public:
HttpMessage() : content_length_(kInvalidLength) {
}
virtual ~HttpMessage() = default;
const std::string& start_line() const {
return start_line_;
}
void set_start_line(const std::string& start_line) {
start_line_ = start_line;
}
std::size_t content_length() const {
return content_length_;
}
const std::string& content() const {
return content_;
}
bool IsConnectionKeepAlive() const;
void SetHeader(HttpHeader&& header) {
headers_.Set(std::move(header.first), std::move(header.second));
}
void SetHeader(const std::string& key, const std::string& value) {
headers_.Set(key, value);
}
const std::string& GetHeader(const std::string& key,
bool* existed = nullptr) const {
return headers_.Get(key, existed);
}
bool HasHeader(const std::string& key) const {
return headers_.Has(key);
}
http::ContentEncoding GetContentEncoding() const;
// Return true if header Accept-Encoding contains "gzip".
bool AcceptEncodingGzip() const;
const ContentType& content_type() const {
return content_type_;
}
// TODO: Set header?
void SetContentType(const ContentType& content_type) {
content_type_ = content_type;
}
void SetContentType(const std::string& content_type) {
SetHeader(http::headers::kContentType, content_type);
}
// Example: SetContentType("application/json", "utf-8")
void SetContentType(const std::string& media_type,
const std::string& charset);
void SetContent(std::string&& content, bool set_length);
// Make the message (e.g., update start line).
// Must be called before ToBuffers()!
virtual bool Prepare() = 0;
// Convert the message into a vector of buffers. The buffers do not own the
// underlying memory blocks, therefore the message object must remain valid
// and not be changed until the write operation has completed.
std::vector<boost::asio::const_buffer> ToBuffers() const;
// Dump to output stream.
void Dump(std::ostream& os, std::size_t indent = 0,
const std::string& prefix = "") const;
// Dump to string, only used by logger.
std::string Dump(std::size_t indent = 0,
const std::string& prefix = "") const;
protected:
void SetContentLength(std::size_t content_length) {
content_length_ = content_length;
SetHeader(http::headers::kContentLength, std::to_string(content_length));
}
protected:
// Start line with trailing CRLF.
// TODO: Don't include trailing CRLF since it's confusing.
std::string start_line_;
std::string content_;
ContentType content_type_;
std::size_t content_length_;
HttpHeaders headers_;
};
} // namespace webcc
#endif // WEBCC_HTTP_MESSAGE_H_