mirror of
https://gitee.com/acl-dev/acl.git
synced 2024-12-14 00:40:55 +08:00
ad62afa90b
修改了一些已知问题,完善了服务器生成向导程序,增加了目录拷贝程序dircopy
210 lines
6.0 KiB
C++
210 lines
6.0 KiB
C++
#pragma once
|
||
#include "acl_cpp/acl_cpp_define.hpp"
|
||
|
||
namespace acl {
|
||
|
||
class string;
|
||
class ostream;
|
||
class socket_stream;
|
||
class http_header;
|
||
class HttpCookie;
|
||
class HttpServlet;
|
||
|
||
/**
|
||
* 与 HTTP 客户端响应相关的类,该类不应被继承,用户也不需要
|
||
* 定义或创建该类对象
|
||
*/
|
||
class ACL_CPP_API HttpServletResponse
|
||
{
|
||
public:
|
||
/**
|
||
* 构造函数
|
||
* @param stream {socket_stream&} 数据流,内部不会自动关闭流
|
||
*/
|
||
HttpServletResponse(socket_stream& stream);
|
||
~HttpServletResponse(void);
|
||
|
||
/**
|
||
* 设置 HTTP 响应数据体的长度
|
||
* @param n {acl_int64} 数据体长度
|
||
*/
|
||
#ifdef WIN32
|
||
HttpServletResponse& setContentLength(__int64 n);
|
||
#else
|
||
HttpServletResponse& setContentLength(long long int n);
|
||
#endif
|
||
|
||
/**
|
||
* 设置 HTTP chunked 传输模式
|
||
* @param on {bool} 如果为 true,即使设置了 setContentLength,
|
||
* 则内部也会采用 chunked 传输方式,根据 HTTP RFC 规范要求,
|
||
* chunked 传输的优先级高级 conteng-length 方式
|
||
* @return {HttpServletResponse&}
|
||
*/
|
||
HttpServletResponse& setChunkedTransferEncoding(bool on);
|
||
|
||
/**
|
||
* 设置与 HTTP 客户端保持联系长连接
|
||
* @param on {bool}
|
||
*/
|
||
HttpServletResponse& setKeepAlive(bool on);
|
||
|
||
/**
|
||
* 设置 HTTP 响应数据体的 Content-Type 字段值,可字段值可以为:
|
||
* text/html 或 text/html; charset=utf8 格式
|
||
* @param value {const char*} 字段值
|
||
*/
|
||
HttpServletResponse& setContentType(const char* value);
|
||
|
||
/**
|
||
* 设置 HTTP 响应数据体中字符集,当已经在 setContentType 设置
|
||
* 了字符集,则就不必再调用本函数设置字符集
|
||
* @param charset {const char*} 响应体数据的字符集
|
||
*/
|
||
HttpServletResponse& setCharacterEncoding(const char* charset);
|
||
|
||
/**
|
||
* 设置 HTTP 响应头中的日期格式的字段
|
||
* @param name {const char*} HTTP 响应头中的字段名
|
||
* @param value {time_t} 时间值
|
||
*/
|
||
HttpServletResponse& setDateHeader(const char* name, time_t value);
|
||
|
||
/**
|
||
* 设置 HTTP 响应头中的字符串格式字段
|
||
* @param name {const char*} HTTP 响应头中的字段名
|
||
* @param value {const char*} 字段值
|
||
*/
|
||
HttpServletResponse& setHeader(const char* name, const char* value);
|
||
|
||
/**
|
||
* 设置 HTTP 响应头中的整数格式字段
|
||
* @param name {const char*} HTTP 响应头中的字段名
|
||
* @param value {int} 字段值
|
||
*/
|
||
HttpServletResponse& setHeader(const char* name, int value);
|
||
|
||
/**
|
||
* 设置 HTTP 响应头中的状态码:1xx, 2xx, 3xx, 4xx, 5xx
|
||
* @param status {int} HTTP 响应状态码, 如:200
|
||
*/
|
||
HttpServletResponse& setStatus(int status);
|
||
|
||
/**
|
||
* 设置为 CGI 模式,用户一般不需手工设置,因为 HttpServlet 类
|
||
* 会自动设置是否是 CGI 模式
|
||
* @param on {bool} 是否是 CGI 模式
|
||
*/
|
||
HttpServletResponse& setCgiMode(bool on);
|
||
|
||
/**
|
||
* 设置 HTTP 响应头中的重定向 location 字段
|
||
* @param location {const char*} URL,非空
|
||
* @param status {int} HTTP 响应状态码,一般为 3xx 类
|
||
*/
|
||
HttpServletResponse& setRedirect(const char* location, int status = 302);
|
||
|
||
/**
|
||
* 添加 cookie 对象,该对象必须是动态分配的,且用户自己不能
|
||
* 再显示释放该对象,因为内部会自动释放
|
||
* @param cookie {HttpCookie*}
|
||
*/
|
||
HttpServletResponse& addCookie(HttpCookie* cookie);
|
||
|
||
/**
|
||
* 添加 cookie
|
||
* @param name {const char*} cookie 名
|
||
* @param value {const char*} cookie 值
|
||
* @param domain {const char*} cookie 存储域
|
||
* @param path {const char*} cookie 存储路径
|
||
* @param expires {time_t} cookie 过期时间间隔,当当前时间加
|
||
* 该值为 cookie 的过期时间截(秒)
|
||
*/
|
||
HttpServletResponse& addCookie(const char* name, const char* value,
|
||
const char* domain = NULL, const char* path = NULL,
|
||
time_t expires = 0);
|
||
|
||
/**
|
||
* 将 url 进行 url 编码
|
||
* @param out {string&} 存储编码后的结果
|
||
* @param url {const char*} 未编码前原始的 url
|
||
*/
|
||
void encodeUrl(string& out, const char* url);
|
||
|
||
/**
|
||
* 获得 HTTP 响应头
|
||
* @return {http_header&}
|
||
*/
|
||
http_header& getHttpHeader(void) const;
|
||
|
||
/**
|
||
* 向客户端发送 HTTP 数据体响应数据,可以循环调用此函数,
|
||
* 当通过 setChunkedTransferEncoding 设置了 chunked 传输方式后,
|
||
* 内部自动采用 chunked 传输方式;调用此函数不必显式调用
|
||
* sendHeader 函数来发送 HTTP 响应头,因为内部会自动在第一次
|
||
* 写时发送 HTTP 响应头;另外,在使用 chunked 方式传输数据时,
|
||
* 应该应该最后再调用一次本函数,且参数均设为 0 表示数据结束
|
||
* @param data {const void*} 数据地址
|
||
* @param len {size_t} data 数据长度
|
||
* @return {bool} 发送是否成功,如果返回 false 表示连接中断
|
||
*/
|
||
bool write(const void* data, size_t len);
|
||
|
||
/**
|
||
* 向客户端发送 HTTP 数据体响应数据,可以循环调用此函数,该函数
|
||
* 内部调用 HttpServletResponse::write(const void*, size_t) 过程,
|
||
* 另外,在使用 chunked 方式传输数据时,应该应该最后再调用一次本函数,
|
||
* 且参数均设为 0 表示数据结束
|
||
* @param buf {const string&} 数据缓冲区
|
||
* @return {bool} 发送是否成功,如果返回 false 表示连接中断
|
||
*/
|
||
bool write(const string& buf);
|
||
|
||
/**
|
||
* 带格式方式向 HTTP 客户端发送响应数据,内部自动调用
|
||
* HttpServletResponse::write(const void*, size_t) 过程,
|
||
* 在使用 chunked 方式传输数据时,应该应该最后再调用 write(NULL, 0) 表示数据结束
|
||
* @param fmt {const char*} 变参格式字符串
|
||
* @return {int} 成功则返回值 > 0,否则返回 -1
|
||
*/
|
||
int format(const char* fmt, ...) ACL_CPP_PRINTF(2, 3);
|
||
|
||
/**
|
||
* 带格式方式向 HTTP 客户端发送响应数据,内部自动调用
|
||
* HttpServletResponse::write(const string&) 过程,
|
||
* 在使用 chunked 方式传输数据时,应该应该最后再调用 write(NULL, 0) 表示数据结束
|
||
* @param fmt {const char*} 变参格式字符串
|
||
* @param ap {va_list} 变参列表
|
||
* @return {int} 成功则返回值 > 0,否则返回 -1
|
||
*/
|
||
int vformat(const char* fmt, va_list ap);
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
/**
|
||
* 发送 HTTP 响应头,用户应该发送数据体前调用此函数将 HTTP
|
||
* 响应头发送给客户端
|
||
* @return {bool} 发送是否成功,若返回 false 则表示连接中断,
|
||
* 当调用以上几个写的函数时,本函数不必显式被调用,如果是
|
||
* 通过从 getOutputStream 获得的 socket 流写数据时,则本函数
|
||
* 必须显式被调用
|
||
*/
|
||
bool sendHeader(void);
|
||
|
||
/**
|
||
* 获得 HTTP 响应对象的输出流对象,用户在调用 sendHeader 发送
|
||
* 完 HTTP 响应头后,通过该输出流来发送 HTTP 数据体
|
||
* @return {ostream&}
|
||
*/
|
||
ostream& getOutputStream(void) const;
|
||
|
||
private:
|
||
socket_stream& stream_; // 客户端连接流
|
||
http_header* header_; // http 响应头
|
||
char charset_[32]; // 字符集
|
||
char content_type_[32]; // content-type 类型
|
||
bool head_sent_; // 是否已经发送了 HTTP 响应头
|
||
};
|
||
|
||
} // namespace acl
|