165 lines
3.6 KiB
C
165 lines
3.6 KiB
C
|
#ifndef HV_ASYNC_HTTP_CLIENT_H_
|
||
|
#define HV_ASYNC_HTTP_CLIENT_H_
|
||
|
|
||
|
#include <map>
|
||
|
#include <list>
|
||
|
|
||
|
#include "EventLoopThread.h"
|
||
|
#include "Channel.h"
|
||
|
|
||
|
#include "HttpMessage.h"
|
||
|
#include "HttpParser.h"
|
||
|
|
||
|
namespace hv {
|
||
|
|
||
|
template<typename Conn>
|
||
|
class ConnPool {
|
||
|
public:
|
||
|
int size() {
|
||
|
return conns_.size();
|
||
|
}
|
||
|
|
||
|
bool get(Conn& conn) {
|
||
|
if (conns_.empty()) return false;
|
||
|
conn = conns_.front();
|
||
|
conns_.pop_front();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool add(const Conn& conn) {
|
||
|
conns_.push_back(conn);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool remove(const Conn& conn) {
|
||
|
auto iter = conns_.begin();
|
||
|
while (iter != conns_.end()) {
|
||
|
if (*iter == conn) {
|
||
|
iter = conns_.erase(iter);
|
||
|
return true;
|
||
|
} else {
|
||
|
++iter;
|
||
|
}
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
std::list<Conn> conns_;
|
||
|
};
|
||
|
|
||
|
struct HttpClientTask {
|
||
|
HttpRequestPtr req;
|
||
|
HttpResponseCallback cb;
|
||
|
uint64_t start_time;
|
||
|
};
|
||
|
typedef std::shared_ptr<HttpClientTask> HttpClientTaskPtr;
|
||
|
|
||
|
struct HttpClientContext {
|
||
|
HttpClientTaskPtr task;
|
||
|
|
||
|
HttpResponsePtr resp;
|
||
|
HttpParserPtr parser;
|
||
|
TimerID timerID;
|
||
|
|
||
|
HttpClientContext() {
|
||
|
timerID = INVALID_TIMER_ID;
|
||
|
}
|
||
|
|
||
|
~HttpClientContext() {
|
||
|
cancelTimer();
|
||
|
}
|
||
|
|
||
|
void cancelTimer() {
|
||
|
if (timerID != INVALID_TIMER_ID) {
|
||
|
killTimer(timerID);
|
||
|
timerID = INVALID_TIMER_ID;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void cancelTask() {
|
||
|
cancelTimer();
|
||
|
task = NULL;
|
||
|
}
|
||
|
|
||
|
void callback() {
|
||
|
cancelTimer();
|
||
|
if (task && task->cb) {
|
||
|
task->cb(resp);
|
||
|
}
|
||
|
// NOTE: task done
|
||
|
task = NULL;
|
||
|
}
|
||
|
|
||
|
void successCallback() {
|
||
|
callback();
|
||
|
resp = NULL;
|
||
|
}
|
||
|
|
||
|
void errorCallback() {
|
||
|
resp = NULL;
|
||
|
callback();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class HV_EXPORT AsyncHttpClient : private EventLoopThread {
|
||
|
public:
|
||
|
AsyncHttpClient(EventLoopPtr loop = NULL) : EventLoopThread(loop) {
|
||
|
if (loop == NULL) {
|
||
|
EventLoopThread::start(true);
|
||
|
}
|
||
|
}
|
||
|
~AsyncHttpClient() {
|
||
|
EventLoopThread::stop(true);
|
||
|
}
|
||
|
|
||
|
// thread-safe
|
||
|
int send(const HttpRequestPtr& req, HttpResponseCallback resp_cb);
|
||
|
int send(const HttpClientTaskPtr& task) {
|
||
|
EventLoopThread::loop()->queueInLoop(std::bind(&AsyncHttpClient::sendInLoop, this, task));
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
void sendInLoop(HttpClientTaskPtr task) {
|
||
|
int err = doTask(task);
|
||
|
if (err != 0 && task->cb) {
|
||
|
task->cb(NULL);
|
||
|
}
|
||
|
}
|
||
|
int doTask(const HttpClientTaskPtr& task);
|
||
|
|
||
|
static int sendRequest(const SocketChannelPtr& channel);
|
||
|
|
||
|
// channel
|
||
|
const SocketChannelPtr& getChannel(int fd) {
|
||
|
return channels[fd];
|
||
|
// return fd < channels.capacity() ? channels[fd] : NULL;
|
||
|
}
|
||
|
|
||
|
const SocketChannelPtr& addChannel(hio_t* io) {
|
||
|
auto channel = std::make_shared<SocketChannel>(io);
|
||
|
channel->newContext<HttpClientContext>();
|
||
|
int fd = channel->fd();
|
||
|
channels[fd] = channel;
|
||
|
return channels[fd];
|
||
|
}
|
||
|
|
||
|
void removeChannel(const SocketChannelPtr& channel) {
|
||
|
channel->deleteContext<HttpClientContext>();
|
||
|
int fd = channel->fd();
|
||
|
channels.erase(fd);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
// NOTE: just one loop thread, no need mutex.
|
||
|
// fd => SocketChannelPtr
|
||
|
std::map<int, SocketChannelPtr> channels;
|
||
|
// peeraddr => ConnPool
|
||
|
std::map<std::string, ConnPool<int>> conn_pools;
|
||
|
};
|
||
|
|
||
|
}
|
||
|
|
||
|
#endif // HV_ASYNC_HTTP_CLIENT_H_
|