libevent实现的httpclient

还比较简单,主要是几个函数、结构体的使用

#include <stdlib.h>
#include <string.h>
#include "event2/event.h"
#include "event2/http.h"
#include "event2/buffer.h"
#include "event2/http_struct.h"
#include "event2/dns.h"


struct download_context{
struct evhttp_uri * uri;
struct event_base * base;
struct evdns_base * dnsbase;
struct evhttp_connection * conn;
struct evhttp_request *req;
struct evbuffer *buffer;
int ok;
};

static void download_callback(struct evhttp_request *req, void *arg);
static int download_renew_request(struct download_context *ctx);

static void download_callback(struct evhttp_request *req, void *arg)
{
struct download_context * ctx = (struct download_context*)arg;
struct evhttp_uri * new_uri = 0;
const char * new_location = 0;
if(!req){
printf("timeout\n");
return;
}

switch(req->response_code)
{
case HTTP_OK:
[size=large][color=darkred]event_base_loopexit(ctx->base, 0); //结束访问[/color][/size]
break;
case HTTP_MOVEPERM:
case HTTP_MOVETEMP:
new_location = evhttp_find_header(req->input_headers, "Location");
if(!new_location) return;
new_uri = evhttp_uri_parse(new_location);
if(!new_uri)return;
evhttp_uri_free(ctx->uri);
ctx->uri = new_uri;
download_renew_request(ctx);
return;
default:/* failed */
event_base_loopexit(ctx->base, 0);
return;
}

evbuffer_add_buffer(ctx->buffer, req->input_buffer);
ctx->ok = 1;
}

struct download_context * context_new(const char *url)
{
struct download_context * ctx = 0;
ctx = (struct download_context*)calloc(1, sizeof(struct download_context));
ctx->uri = evhttp_uri_parse(url);
if(!ctx->uri) return 0;

[size=large][color=darkred] ctx->base = event_base_new(); //如果原来有base,这句的影响?是否要开新的线程?[/color][/size]
ctx->buffer = evbuffer_new();
[size=large][color=darkred] ctx->dnsbase = evdns_base_new(ctx->base, 1); //这里也有内存泄露,注意有new的函数[/color][/size]

download_renew_request(ctx);
return ctx;
}

void context_free(struct download_context *ctx)
{
if(ctx->conn)
evhttp_connection_free(ctx->conn);

if(ctx->buffer)
evbuffer_free(ctx->buffer);

if(ctx->uri)
evhttp_uri_free(ctx->uri);

free(ctx);
}

static int download_renew_request(struct download_context *ctx)
{
int port = evhttp_uri_get_port(ctx->uri);
if(port == -1) port = 80;
if(ctx->conn) evhttp_connection_free(ctx->conn);

printf("host:%s, port:%d, path:%s\n", evhttp_uri_get_host(ctx->uri), port, evhttp_uri_get_path(ctx->uri));

ctx->conn = evhttp_connection_base_new(ctx->base, ctx->dnsbase, evhttp_uri_get_host(ctx->uri), port );
[size=large][color=darkred]ctx->req = evhttp_request_new(download_callback, ctx); //感觉有内存泄露,ctx作为回调函数的参数传入[/color][/size]
evhttp_make_request(ctx->conn, ctx->req, EVHTTP_REQ_GET, evhttp_uri_get_path(ctx->uri));
evhttp_add_header(ctx->req->output_headers, "Host", evhttp_uri_get_host(ctx->uri));

return 0;
}

struct evbuffer *download_url(const char *url)
{
struct download_context * ctx = context_new(url);
if(!ctx) return 0;

[size=large][color=darkred] event_base_dispatch(ctx->base); //在这里进入循环,等待访问结束[/color][/size]

struct evbuffer * retval = 0;
if(ctx->ok)
{
retval = ctx->buffer;
ctx->buffer = 0;
}

context_free(ctx);
return retval;
}

int main(int argc, char **argv)
{
struct evbuffer * data = 0;
if(argc < 2){
printf("usage: %s example.com/\n", argv[0]);
return 1;
}

#ifdef WIN32
WORD wVersionRequested;
WSADATA wsaData;

wVersionRequested = MAKEWORD(2, 2);

(void) WSAStartup(wVersionRequested, &wsaData);
#endif

data = download_url(argv[1]);

printf("got %d bytes\n", data ? evbuffer_get_length(data) : -1);

if(data)
{
const unsigned char * joined = evbuffer_pullup(data, -1);
printf("data itself:\n====================\n");
fwrite(joined, evbuffer_get_length(data), 1, stderr);
printf("\n====================\n");
evbuffer_free(data);
}

return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值