acl/lib_acl_cpp/include/acl_cpp/http/http_aclient.hpp

374 lines
9.3 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#pragma once
#include "../acl_cpp_define.hpp"
#include "../stream/aio_socket_stream.hpp"
#if !defined(_WIN32) && !defined(_WIN64)
#include <netinet/in.h> // just for "struct sockaddr_storage"
#endif
struct HTTP_HDR;
struct HTTP_HDR_RES;
struct HTTP_RES;
struct HTTP_HDR_REQ;
struct HTTP_REQ;
struct ACL_ASTREAM_CTX;
namespace acl {
class string;
class aio_handle;
class aio_socket_stream;
class socket_stream;
class zlib_stream;
class websocket;
class polarssl_conf;
class polarssl_io;
class http_header;
/**
* HTTP 客户端异步通信类,不仅支持标准 HTTP 通信协议,而且支持 Websocket 通信,
* 对于 HTTP 协议及 Websocket 通信均支持 SSL 加密传输;
* 另外,对于 HTTP 协议,根据用户设置,可以自动解压 GZIP 响应数据,这样在回调
* 方法 on_http_res_body() 中收到的便是解压后的明文数据。
*/
class ACL_CPP_API http_aclient : public aio_open_callback
{
public:
/**
* 构造函数
* @param handle {aio_handle&} 异步通信事件引擎句柄
* @param ssl_conf {polarssl_conf*} 非 NULL 时自动采用 SSL 通信方式
*/
http_aclient(aio_handle& handle, polarssl_conf* ssl_conf = NULL);
virtual ~http_aclient(void);
/**
* 当对象销毁时的回调方法,子类必须实现
*/
virtual void destroy(void) = 0;
/**
* 获得 HTTP 请求头,以便于应用添加 HTTP 请求头中的字段内容
* @return {http_header&}
*/
http_header& request_header(void);
/**
* 针对 HTTP 协议的响应数据是否自动进行解压
* @param on {bool}
* @return {http_aclient&}
*/
http_aclient& unzip_body(bool on);
/**
* 是否针对 GZIP 压缩数据自动进行解压
* @return {bool}
*/
bool is_unzip_body(void) const
{
return unzip_;
}
/**
* 开始异步连接远程 WEB 服务器
* @param addr {const char*} 远程 WEB 服务器地址,格式为:
* domain:port 或 ip:port, 当地址为域名时,内部自动进行异步域名解析
* 过程,但要求在程序开始时必须通过 aio_handle::set_dns() 设置过域名
* 服务器地址,如果地址为 IP 则不需要先设置域名服务器地址
* @param conn_timeout {int} 连接超时时间(秒)
* @param rw_timeout {int} 网络 IO 读写超时时间(秒)
* @return {bool} 返回 false 表示连接失败,返回 true 表示进入异步连接中
*/
bool open(const char* addr, int conn_timeout, int rw_timeout);
/**
* 异步关闭连接
*/
void close(void);
/**
* 获得本次连接(无论成功或失败)所使用的 DNS 服务地址
* @param out {string&} 存储结果
* @return {bool} 是否成功获得
*/
bool get_ns_addr(string& out);
/**
* 连接成功后可调用本方法获得异步连接对象
* @return {aio_socket_stream*}
*/
aio_socket_stream* get_conn(void) const
{
return conn_;
}
protected:
/**
* 当连接成功后的回调方法,子类必须实现,子类应在该方法里构造 HTTP 请求
* 并调用 send_request 方法向 WEB 服务器发送 HTTP 请求
* @return {bool} 该方法如果返回 false 则内部会自动关闭连接
*/
virtual bool on_connect(void) = 0;
/**
* 当连接超时后的回调方法
*/
virtual void on_connect_timeout(void) {}
/**
* 当连接失败后的回调方法
*/
virtual void on_connect_failed(void) {}
/**
* 当网络读超时时的回调方法
* @return {bool} 当读超时回调方法返回 true则内部会继续读数据如果
* 返回 false则该连接将会被关闭接着回调 on_disconnect() 虚方法
*/
virtual bool on_read_timeout(void) { return false; }
/**
* 对于连接成功后连接关闭后的回调方法,内部调用此方法后便立即回调
* destroy() 方法
*/
virtual void on_disconnect(void) {};
/**
* 当接收到 WEB 服务端的响应头时的回调方法
* @param header {const http_header&}
* @return {bool} 返回 false 则将会关闭连接,否则继续读
*/
virtual bool on_http_res_hdr(const http_header& header)
{
(void) header;
return true;
}
/**
* 当接收到 WEB 服务端的响应体时的回调方法,该方法可能会被多次回调
* 直到响应数据读完或出错
* @param data {char*} 读到的部分数据体内容
* @param dlen {size_t} 本次读到的 data 数据的长度
* @return {bool} 返回 false 则将会关闭连接,否则继续读
*/
virtual bool on_http_res_body(char* data, size_t dlen)
{
(void) data;
(void) dlen;
return true;
}
/**
* 当读完 HTTP 响应体或出错后的回调方法
* @param success {bool} 是否成功读完 HTTP 响应体数据
* @return {bool} 如果成功读完数据体后返回 false 则会关闭连接
*/
virtual bool on_http_res_finish(bool success)
{
(void) success;
return true;
}
/**
* 当 websocket 握手成功后的回调方法
* @return {bool} 返回 false 表示需要关闭连接,否则继续
*/
virtual bool on_ws_handshake(void)
{
// 开始异步读 websocket 数据
this->ws_read_wait(0);
return true;
}
/**
* 当 websocket 握手失败后的回调方法
* @param status {int} 服务器返回的 HTTP 响应状态码
*/
virtual void on_ws_handshake_failed(int status) { (void) status; }
/**
* 当读到一个 text 类型的帧时的回调方法
* @return {bool} 返回 true 表示继续读,否则则要求关闭连接
*/
virtual bool on_ws_frame_text(void) { return true; }
/**
* 当读到一个 binary 类型的帧时的回调方法
* @return {bool} 返回 true 表示继续读,否则则要求关闭连接
*/
virtual bool on_ws_frame_binary(void) { return true; }
/**
* 当读到一个关闭帧数据时的回调方法
*/
virtual void on_ws_frame_closed(void) {}
/**
* 在 websocket 通信方式,当读到数据体时的回调方法
* @param data {char*} 读到的数据地址
* @param dlen {size_t} 读到的数据长度
* @return {bool} 返回 true 表示继续读,否则则要求关闭连接
*/
virtual bool on_ws_frame_data(char* data, size_t dlen)
{
(void) data;
(void) dlen;
return true;
}
/**
* 当读完一帧数据时的回调方法
* @return {bool} 返回 true 表示继续读,否则则要求关闭连接
*/
virtual bool on_ws_frame_finish(void) { return true; }
/**
* 当收到 ping 数据包时的回调方法,当该回调方法返回后,如果用户没有将
* data 数据清理,则内部会自动给对端写入 pong 信息,如果用户将 data 缓
* 冲区清理掉,则该方法返回后不会给对接发 pong 信息
* @param data {string&} 读到的数据
*/
virtual void on_ws_frame_ping(string& data)
{
(void) data;
}
/**
* 收到 pong 数据时的回调方法
* @param data {string&} 读到的数据
*/
virtual void on_ws_frame_pong(string& data)
{
(void) data;
}
public:
/**
* 向 WEB 服务器发送 HTTP 请求,内部在发送后会自动开始读 HTTP 响应过程
* @param body {const void*} HTTP 请求的数据体,当为 NULL 时,内部会自
* 动采用 HTTP GET 方法
* @param len {size_t} body 非 NULL 时表示数据体的长度
*/
void send_request(const void* body, size_t len);
/**
* 与服务器进行 WEBSOCKET 握手
*/
void ws_handshake(const void* key, size_t len);
void ws_handshake(const char* key = "123456789xxx");
/**
* 当调用 ws_handshake() 时,内部会填充与 websocket 相关的请求头信息,
* 同时通过此回调告之调用者最终发给 websocket 服务器完整的请求头信息
*/
virtual void ws_handshake_before(http_header& reqhdr)
{
(void) reqhdr;
}
/**
* 开始异步读 websocket 数据
* @param timeout {int} 读超时时间,如果该值 <= 0则不设读超时时间
* 否则当读超时时,超时回调方法便会被调用;
* 注意:
* 该值与 open() 中的 rw_timeout 有所不同open() 方法中的读超时仅限
* 定标准 HTTP IO 过程及 SSL 握手过程的读超时,而此处的读超时则用来
* 限制与 websocket 相关的读超时,这主要是考虑到 websocket 应用很多
* 都是长连接场景
*/
void ws_read_wait(int timeout = 0);
/**
* 异步发送一个 FRAME_TEXT 类型的数据帧
* @param data {char*} 内部可能因添加掩码原因被改变内容
* @param len {size_t} data 数据长度
* @return {bool}
*/
bool ws_send_text(char* data, size_t len);
/**
* 异步发送一个 FRAME_BINARY 类型的数据帧
* @param data {void*} 内部可能因添加掩码原因被改变内容
* @param len {size_t} data 数据长度
* @return {bool}
*/
bool ws_send_binary(void* data, size_t len);
/**
* 异步发送一个 FRAME_PING 类型的数据帧
* @param data {void*} 内部可能因添加掩码原因被改变内容
* @param len {size_t} data 数据长度
* @return {bool}
*/
bool ws_send_ping(void* data, size_t len);
/**
* 异步发送一个 FRAME_PONG 类型的数据帧
* @param data {void*} 内部可能因添加掩码原因被改变内容
* @param len {size_t} data 数据长度
* @return {bool}
*/
bool ws_send_pong(void* data, size_t len);
protected:
// @override dummy
bool open_callback(void) { return true; }
// @override
bool timeout_callback(void);
// @override
void close_callback(void);
// @override
bool read_wakeup(void);
// @override
bool read_callback(char* data, int len);
protected:
unsigned status_;
aio_handle& handle_;
polarssl_conf* ssl_conf_;
int rw_timeout_;
aio_socket_stream* conn_;
socket_stream* stream_;
http_header* header_;
HTTP_HDR_RES* hdr_res_;
HTTP_RES* http_res_;
bool keep_alive_;
websocket* ws_in_;
websocket* ws_out_;
string* buff_;
bool unzip_; // 是否自动解压响应体数据
zlib_stream* zstream_; // 解压对象
int gzip_header_left_; // gzip 传输时压缩头部长度
struct sockaddr_storage ns_addr_;
bool handle_connect(const ACL_ASTREAM_CTX* ctx);
bool handle_ssl_handshake(void);
bool handle_res_hdr(int status);
bool handle_res_body(char* data, int dlen);
bool res_plain(char* data, int dlen);
bool res_unzip(zlib_stream& zstream, char* data, int dlen);
bool handle_res_body_finish(char* data, int dlen);
bool res_plain_finish(char* data, int dlen);
bool res_unzip_finish(zlib_stream& zstream, char* data, int dlen);
bool handle_websocket(void);
bool handle_ws_data(void);
bool handle_ws_ping(void);
bool handle_ws_pong(void);
bool handle_ws_other(void);
private:
static int connect_callback(const ACL_ASTREAM_CTX* ctx);
static int http_res_hdr_cllback(int status, void* ctx);
static int http_res_callback(int status, char* data, int dlen, void* ctx);
};
} // namespace acl