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.
251 lines
5.7 KiB
C++
251 lines
5.7 KiB
C++
#ifndef WEBCC_COMMON_H_
|
|
#define WEBCC_COMMON_H_
|
|
|
|
#include <cassert>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "boost/asio/buffer.hpp" // for const_buffer
|
|
#include "boost/filesystem/path.hpp"
|
|
|
|
#include "webcc/globals.h"
|
|
|
|
namespace webcc {
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
using Path = boost::filesystem::path;
|
|
|
|
using Payload = std::vector<boost::asio::const_buffer>;
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// Split a string to two parts by the given token.
|
|
bool Split2(const std::string& str, char token, std::string* part1,
|
|
std::string* part2);
|
|
|
|
// Read entire file into string.
|
|
bool ReadFile(const Path& path, std::string* output);
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
using Header = std::pair<std::string, std::string>;
|
|
|
|
class Headers {
|
|
public:
|
|
std::size_t size() const {
|
|
return headers_.size();
|
|
}
|
|
|
|
bool empty() const {
|
|
return headers_.empty();
|
|
}
|
|
|
|
const std::vector<Header>& data() const {
|
|
return headers_;
|
|
}
|
|
|
|
void Set(const std::string& key, const std::string& value);
|
|
|
|
void Set(std::string&& key, std::string&& value);
|
|
|
|
bool Has(const std::string& key) const;
|
|
|
|
// Get header by index.
|
|
const Header& Get(std::size_t index) const {
|
|
assert(index < size());
|
|
return headers_[index];
|
|
}
|
|
|
|
// Get header value by key.
|
|
// If there's no such header with the given key, besides return empty, the
|
|
// optional |existed| parameter will be set to false.
|
|
const std::string& Get(const std::string& key, bool* existed = nullptr) const;
|
|
|
|
void Clear() {
|
|
headers_.clear();
|
|
}
|
|
|
|
private:
|
|
std::vector<Header>::iterator Find(const std::string& key);
|
|
|
|
std::vector<Header> headers_;
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// Content-Type header.
|
|
// Syntax:
|
|
// Content-Type: text/html; charset=utf-8
|
|
// Content-Type: multipart/form-data; boundary=something
|
|
class ContentType {
|
|
public:
|
|
explicit ContentType(const std::string& str = "");
|
|
|
|
void Parse(const std::string& str);
|
|
|
|
bool Valid() const;
|
|
|
|
bool multipart() const {
|
|
return multipart_;
|
|
}
|
|
|
|
const std::string& media_type() const {
|
|
return media_type_;
|
|
}
|
|
|
|
const std::string& charset() const {
|
|
assert(!multipart_);
|
|
return additional_;
|
|
}
|
|
|
|
const std::string& boundary() const {
|
|
assert(multipart_);
|
|
return additional_;
|
|
}
|
|
|
|
private:
|
|
void Init(const std::string& str);
|
|
|
|
private:
|
|
std::string media_type_;
|
|
std::string additional_;
|
|
bool multipart_ = false;
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// Content-Disposition header.
|
|
// Syntax:
|
|
// Content-Disposition: form-data
|
|
// Content-Disposition: form-data; name="fieldName"
|
|
// Content-Disposition: form-data; name="fieldName"; filename="filename.jpg"
|
|
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Disposition
|
|
class ContentDisposition {
|
|
public:
|
|
explicit ContentDisposition(const std::string& str) {
|
|
valid_ = Init(str);
|
|
}
|
|
|
|
bool valid() const {
|
|
return valid_;
|
|
}
|
|
|
|
const std::string& name() const {
|
|
return name_;
|
|
}
|
|
|
|
const std::string& file_name() const {
|
|
return file_name_;
|
|
}
|
|
|
|
private:
|
|
bool Init(const std::string& str);
|
|
|
|
private:
|
|
std::string name_;
|
|
std::string file_name_;
|
|
bool valid_ = false;
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// A part of the multipart form data.
|
|
class FormPart {
|
|
public:
|
|
FormPart() = default;
|
|
|
|
// Construct a file part.
|
|
// The file name will be extracted from path.
|
|
// The media type, if not provided, will be inferred from file extension.
|
|
FormPart(const std::string& name, const Path& path,
|
|
const std::string& media_type = "");
|
|
|
|
// Construct a non-file part.
|
|
// The data will be moved, no file name is needed.
|
|
// The media type is optional. If the data is a JSON string, you can specify
|
|
// media type as "application/json".
|
|
FormPart(const std::string& name, std::string&& data,
|
|
const std::string& media_type = "");
|
|
|
|
FormPart(const FormPart&) = delete;
|
|
FormPart& operator=(const FormPart&) = delete;
|
|
|
|
// API: SERVER
|
|
const std::string& name() const {
|
|
return name_;
|
|
}
|
|
|
|
// API: SERVER/PARSER
|
|
void set_name(const std::string& name) {
|
|
name_ = name;
|
|
}
|
|
|
|
// API: SERVER
|
|
const std::string& file_name() const {
|
|
return file_name_;
|
|
}
|
|
|
|
// API: SERVER/PARSER
|
|
void set_file_name(const std::string& file_name) {
|
|
file_name_ = file_name;
|
|
}
|
|
|
|
// API: SERVER
|
|
const std::string& media_type() const {
|
|
return media_type_;
|
|
}
|
|
|
|
// API: SERVER
|
|
const std::string& data() const {
|
|
return data_;
|
|
}
|
|
|
|
// API: SERVER/PARSER
|
|
void AppendData(const std::string& data) {
|
|
data_.append(data);
|
|
}
|
|
|
|
// API: SERVER/PARSER
|
|
void AppendData(const char* data, std::size_t count) {
|
|
data_.append(data, count);
|
|
}
|
|
|
|
// API: CLIENT
|
|
void Prepare(Payload* payload);
|
|
|
|
private:
|
|
// Generate headers from properties.
|
|
void SetHeaders();
|
|
|
|
private:
|
|
// The <input> name within the original HTML form.
|
|
// E.g., given HTML form:
|
|
// <input name="file1" type="file">
|
|
// the name will be "file1".
|
|
std::string name_;
|
|
|
|
// The original local file name.
|
|
// E.g., "baby.jpg".
|
|
std::string file_name_;
|
|
|
|
// The content-type if the media type is known (e.g., inferred from the file
|
|
// extension or operating system typing information) or as
|
|
// application/octet-stream.
|
|
// E.g., "image/jpeg".
|
|
std::string media_type_;
|
|
|
|
// Headers generated from the above properties.
|
|
// Only Used to prepare payload.
|
|
Headers headers_;
|
|
|
|
std::string data_;
|
|
};
|
|
|
|
using FormPartPtr = std::shared_ptr<FormPart>;
|
|
|
|
} // namespace webcc
|
|
|
|
#endif // WEBCC_COMMON_H_
|