http.c 15.5 KB
Newer Older
1 2 3
#include "http.h"

int data_received;
J
Junio C Hamano 已提交
4
int active_requests;
5 6

#ifdef USE_CURL_MULTI
M
Mike Hommey 已提交
7 8
static int max_requests = -1;
static CURLM *curlm;
9 10
#endif
#ifndef NO_CURL_EASY_DUPHANDLE
M
Mike Hommey 已提交
11
static CURL *curl_default;
12 13 14
#endif
char curl_errorstr[CURL_ERROR_SIZE];

M
Mike Hommey 已提交
15
static int curl_ssl_verify = -1;
J
Junio C Hamano 已提交
16
static const char *ssl_cert;
17
#if LIBCURL_VERSION_NUM >= 0x070902
J
Junio C Hamano 已提交
18
static const char *ssl_key;
19 20
#endif
#if LIBCURL_VERSION_NUM >= 0x070908
J
Junio C Hamano 已提交
21
static const char *ssl_capath;
22
#endif
J
Junio C Hamano 已提交
23
static const char *ssl_cainfo;
M
Mike Hommey 已提交
24 25
static long curl_low_speed_limit = -1;
static long curl_low_speed_time = -1;
J
Junio C Hamano 已提交
26 27
static int curl_ftp_no_epsv;
static const char *curl_http_proxy;
J
Junio C Hamano 已提交
28
static char *user_name, *user_pass;
29

M
Mike Hommey 已提交
30
static struct curl_slist *pragma_header;
31

J
Junio C Hamano 已提交
32
static struct active_request_slot *active_queue_head;
33

34
size_t fread_buffer(void *ptr, size_t eltsize, size_t nmemb, void *buffer_)
35 36
{
	size_t size = eltsize * nmemb;
37 38
	struct buffer *buffer = buffer_;

M
Mike Hommey 已提交
39 40 41
	if (size > buffer->buf.len - buffer->posn)
		size = buffer->buf.len - buffer->posn;
	memcpy(ptr, buffer->buf.buf + buffer->posn, size);
42
	buffer->posn += size;
M
Mike Hommey 已提交
43

44 45 46
	return size;
}

47
size_t fwrite_buffer(const void *ptr, size_t eltsize, size_t nmemb, void *buffer_)
48 49
{
	size_t size = eltsize * nmemb;
50 51
	struct strbuf *buffer = buffer_;

M
Mike Hommey 已提交
52
	strbuf_add(buffer, ptr, size);
53 54 55 56
	data_received++;
	return size;
}

57
size_t fwrite_null(const void *ptr, size_t eltsize, size_t nmemb, void *strbuf)
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
{
	data_received++;
	return eltsize * nmemb;
}

static void finish_active_slot(struct active_request_slot *slot);

#ifdef USE_CURL_MULTI
static void process_curl_messages(void)
{
	int num_messages;
	struct active_request_slot *slot;
	CURLMsg *curl_message = curl_multi_info_read(curlm, &num_messages);

	while (curl_message != NULL) {
		if (curl_message->msg == CURLMSG_DONE) {
			int curl_result = curl_message->data.result;
			slot = active_queue_head;
			while (slot != NULL &&
			       slot->curl != curl_message->easy_handle)
				slot = slot->next;
			if (slot != NULL) {
				curl_multi_remove_handle(curlm, slot->curl);
				slot->curl_result = curl_result;
				finish_active_slot(slot);
			} else {
				fprintf(stderr, "Received DONE message for unknown request!\n");
			}
		} else {
			fprintf(stderr, "Unknown CURL message received: %d\n",
				(int)curl_message->msg);
		}
		curl_message = curl_multi_info_read(curlm, &num_messages);
	}
}
#endif

95
static int http_options(const char *var, const char *value, void *cb)
96 97
{
	if (!strcmp("http.sslverify", var)) {
98
		curl_ssl_verify = git_config_bool(var, value);
99 100
		return 0;
	}
101 102
	if (!strcmp("http.sslcert", var))
		return git_config_string(&ssl_cert, var, value);
103
#if LIBCURL_VERSION_NUM >= 0x070902
104 105
	if (!strcmp("http.sslkey", var))
		return git_config_string(&ssl_key, var, value);
106 107
#endif
#if LIBCURL_VERSION_NUM >= 0x070908
108 109
	if (!strcmp("http.sslcapath", var))
		return git_config_string(&ssl_capath, var, value);
110
#endif
111 112
	if (!strcmp("http.sslcainfo", var))
		return git_config_string(&ssl_cainfo, var, value);
J
Junio C Hamano 已提交
113
#ifdef USE_CURL_MULTI
114
	if (!strcmp("http.maxrequests", var)) {
115
		max_requests = git_config_int(var, value);
116 117 118 119
		return 0;
	}
#endif
	if (!strcmp("http.lowspeedlimit", var)) {
120
		curl_low_speed_limit = (long)git_config_int(var, value);
121 122 123
		return 0;
	}
	if (!strcmp("http.lowspeedtime", var)) {
124
		curl_low_speed_time = (long)git_config_int(var, value);
125 126 127
		return 0;
	}

128 129 130 131
	if (!strcmp("http.noepsv", var)) {
		curl_ftp_no_epsv = git_config_bool(var, value);
		return 0;
	}
132 133
	if (!strcmp("http.proxy", var))
		return git_config_string(&curl_http_proxy, var, value);
134

135
	/* Fall back on the default ones */
136
	return git_default_config(var, value, cb);
137 138
}

J
Junio C Hamano 已提交
139 140 141 142 143 144 145 146 147 148 149 150 151 152
static void init_curl_http_auth(CURL *result)
{
	if (!user_name)
		curl_easy_setopt(result, CURLOPT_NETRC, CURL_NETRC_OPTIONAL);
	else {
		struct strbuf up = STRBUF_INIT;
		if (!user_pass)
			user_pass = xstrdup(getpass("Password: "));
		strbuf_addf(&up, "%s:%s", user_name, user_pass);
		curl_easy_setopt(result, CURLOPT_USERPWD,
				 strbuf_detach(&up, NULL));
	}
}

J
Junio C Hamano 已提交
153
static CURL *get_curl_handle(void)
154
{
J
Junio C Hamano 已提交
155
	CURL *result = curl_easy_init();
156

157 158 159 160 161 162 163 164 165 166
	if (!curl_ssl_verify) {
		curl_easy_setopt(result, CURLOPT_SSL_VERIFYPEER, 0);
		curl_easy_setopt(result, CURLOPT_SSL_VERIFYHOST, 0);
	} else {
		/* Verify authenticity of the peer's certificate */
		curl_easy_setopt(result, CURLOPT_SSL_VERIFYPEER, 1);
		/* The name in the cert must match whom we tried to connect */
		curl_easy_setopt(result, CURLOPT_SSL_VERIFYHOST, 2);
	}

167 168 169 170
#if LIBCURL_VERSION_NUM >= 0x070907
	curl_easy_setopt(result, CURLOPT_NETRC, CURL_NETRC_OPTIONAL);
#endif

J
Junio C Hamano 已提交
171 172
	init_curl_http_auth(result);

173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195
	if (ssl_cert != NULL)
		curl_easy_setopt(result, CURLOPT_SSLCERT, ssl_cert);
#if LIBCURL_VERSION_NUM >= 0x070902
	if (ssl_key != NULL)
		curl_easy_setopt(result, CURLOPT_SSLKEY, ssl_key);
#endif
#if LIBCURL_VERSION_NUM >= 0x070908
	if (ssl_capath != NULL)
		curl_easy_setopt(result, CURLOPT_CAPATH, ssl_capath);
#endif
	if (ssl_cainfo != NULL)
		curl_easy_setopt(result, CURLOPT_CAINFO, ssl_cainfo);
	curl_easy_setopt(result, CURLOPT_FAILONERROR, 1);

	if (curl_low_speed_limit > 0 && curl_low_speed_time > 0) {
		curl_easy_setopt(result, CURLOPT_LOW_SPEED_LIMIT,
				 curl_low_speed_limit);
		curl_easy_setopt(result, CURLOPT_LOW_SPEED_TIME,
				 curl_low_speed_time);
	}

	curl_easy_setopt(result, CURLOPT_FOLLOWLOCATION, 1);

196 197 198
	if (getenv("GIT_CURL_VERBOSE"))
		curl_easy_setopt(result, CURLOPT_VERBOSE, 1);

199 200
	curl_easy_setopt(result, CURLOPT_USERAGENT, GIT_USER_AGENT);

201 202 203
	if (curl_ftp_no_epsv)
		curl_easy_setopt(result, CURLOPT_FTP_USE_EPSV, 0);

204 205 206
	if (curl_http_proxy)
		curl_easy_setopt(result, CURLOPT_PROXY, curl_http_proxy);

207 208 209
	return result;
}

J
Junio C Hamano 已提交
210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249
static void http_auth_init(const char *url)
{
	char *at, *colon, *cp, *slash;
	int len;

	cp = strstr(url, "://");
	if (!cp)
		return;

	/*
	 * Ok, the URL looks like "proto://something".  Which one?
	 * "proto://<user>:<pass>@<host>/...",
	 * "proto://<user>@<host>/...", or just
	 * "proto://<host>/..."?
	 */
	cp += 3;
	at = strchr(cp, '@');
	colon = strchr(cp, ':');
	slash = strchrnul(cp, '/');
	if (!at || slash <= at)
		return; /* No credentials */
	if (!colon || at <= colon) {
		/* Only username */
		len = at - cp;
		user_name = xmalloc(len + 1);
		memcpy(user_name, cp, len);
		user_name[len] = '\0';
		user_pass = NULL;
	} else {
		len = colon - cp;
		user_name = xmalloc(len + 1);
		memcpy(user_name, cp, len);
		user_name[len] = '\0';
		len = at - (colon + 1);
		user_pass = xmalloc(len + 1);
		memcpy(user_pass, colon + 1, len);
		user_pass[len] = '\0';
	}
}

250 251 252 253 254 255 256
static void set_from_env(const char **var, const char *envname)
{
	const char *val = getenv(envname);
	if (val)
		*var = val;
}

257
void http_init(struct remote *remote)
258 259 260 261
{
	char *low_speed_limit;
	char *low_speed_time;

262 263
	git_config(http_options, NULL);

264 265
	curl_global_init(CURL_GLOBAL_ALL);

266 267 268
	if (remote && remote->http_proxy)
		curl_http_proxy = xstrdup(remote->http_proxy);

269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287
	pragma_header = curl_slist_append(pragma_header, "Pragma: no-cache");

#ifdef USE_CURL_MULTI
	{
		char *http_max_requests = getenv("GIT_HTTP_MAX_REQUESTS");
		if (http_max_requests != NULL)
			max_requests = atoi(http_max_requests);
	}

	curlm = curl_multi_init();
	if (curlm == NULL) {
		fprintf(stderr, "Error creating curl multi handle.\n");
		exit(1);
	}
#endif

	if (getenv("GIT_SSL_NO_VERIFY"))
		curl_ssl_verify = 0;

288
	set_from_env(&ssl_cert, "GIT_SSL_CERT");
289
#if LIBCURL_VERSION_NUM >= 0x070902
290
	set_from_env(&ssl_key, "GIT_SSL_KEY");
291 292
#endif
#if LIBCURL_VERSION_NUM >= 0x070908
293
	set_from_env(&ssl_capath, "GIT_SSL_CAPATH");
294
#endif
295
	set_from_env(&ssl_cainfo, "GIT_SSL_CAINFO");
296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311

	low_speed_limit = getenv("GIT_HTTP_LOW_SPEED_LIMIT");
	if (low_speed_limit != NULL)
		curl_low_speed_limit = strtol(low_speed_limit, NULL, 10);
	low_speed_time = getenv("GIT_HTTP_LOW_SPEED_TIME");
	if (low_speed_time != NULL)
		curl_low_speed_time = strtol(low_speed_time, NULL, 10);

	if (curl_ssl_verify == -1)
		curl_ssl_verify = 1;

#ifdef USE_CURL_MULTI
	if (max_requests < 1)
		max_requests = DEFAULT_MAX_REQUESTS;
#endif

312 313 314
	if (getenv("GIT_CURL_FTP_NO_EPSV"))
		curl_ftp_no_epsv = 1;

J
Junio C Hamano 已提交
315 316 317
	if (remote && remote->url && remote->url[0])
		http_auth_init(remote->url[0]);

318 319 320 321 322 323 324 325 326 327
#ifndef NO_CURL_EASY_DUPHANDLE
	curl_default = get_curl_handle();
#endif
}

void http_cleanup(void)
{
	struct active_request_slot *slot = active_queue_head;

	while (slot != NULL) {
328
		struct active_request_slot *next = slot->next;
329
		if (slot->curl != NULL) {
330
#ifdef USE_CURL_MULTI
331
			curl_multi_remove_handle(curlm, slot->curl);
332 333
#endif
			curl_easy_cleanup(slot->curl);
334
		}
335 336
		free(slot);
		slot = next;
337
	}
338
	active_queue_head = NULL;
339 340 341 342 343 344 345 346 347

#ifndef NO_CURL_EASY_DUPHANDLE
	curl_easy_cleanup(curl_default);
#endif

#ifdef USE_CURL_MULTI
	curl_multi_cleanup(curlm);
#endif
	curl_global_cleanup();
N
Nick Hengeveld 已提交
348 349

	curl_slist_free_all(pragma_header);
350
	pragma_header = NULL;
351 352

	if (curl_http_proxy) {
353
		free((void *)curl_http_proxy);
354 355
		curl_http_proxy = NULL;
	}
356 357 358 359 360 361 362 363 364 365 366 367 368
}

struct active_request_slot *get_active_slot(void)
{
	struct active_request_slot *slot = active_queue_head;
	struct active_request_slot *newslot;

#ifdef USE_CURL_MULTI
	int num_transfers;

	/* Wait for a slot to open up if the queue is full */
	while (active_requests >= max_requests) {
		curl_multi_perform(curlm, &num_transfers);
J
Junio C Hamano 已提交
369
		if (num_transfers < active_requests)
370 371 372 373
			process_curl_messages();
	}
#endif

J
Junio C Hamano 已提交
374
	while (slot != NULL && slot->in_use)
375
		slot = slot->next;
J
Junio C Hamano 已提交
376

377 378 379 380 381 382 383 384 385 386
	if (slot == NULL) {
		newslot = xmalloc(sizeof(*newslot));
		newslot->curl = NULL;
		newslot->in_use = 0;
		newslot->next = NULL;

		slot = active_queue_head;
		if (slot == NULL) {
			active_queue_head = newslot;
		} else {
J
Junio C Hamano 已提交
387
			while (slot->next != NULL)
388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404
				slot = slot->next;
			slot->next = newslot;
		}
		slot = newslot;
	}

	if (slot->curl == NULL) {
#ifdef NO_CURL_EASY_DUPHANDLE
		slot->curl = get_curl_handle();
#else
		slot->curl = curl_easy_duphandle(curl_default);
#endif
	}

	active_requests++;
	slot->in_use = 1;
	slot->local = NULL;
405
	slot->results = NULL;
N
Nick Hengeveld 已提交
406
	slot->finished = NULL;
407 408 409 410
	slot->callback_data = NULL;
	slot->callback_func = NULL;
	curl_easy_setopt(slot->curl, CURLOPT_HTTPHEADER, pragma_header);
	curl_easy_setopt(slot->curl, CURLOPT_ERRORBUFFER, curl_errorstr);
411 412 413 414 415
	curl_easy_setopt(slot->curl, CURLOPT_CUSTOMREQUEST, NULL);
	curl_easy_setopt(slot->curl, CURLOPT_READFUNCTION, NULL);
	curl_easy_setopt(slot->curl, CURLOPT_WRITEFUNCTION, NULL);
	curl_easy_setopt(slot->curl, CURLOPT_UPLOAD, 0);
	curl_easy_setopt(slot->curl, CURLOPT_HTTPGET, 1);
416 417 418 419 420 421 422 423

	return slot;
}

int start_active_slot(struct active_request_slot *slot)
{
#ifdef USE_CURL_MULTI
	CURLMcode curlm_result = curl_multi_add_handle(curlm, slot->curl);
424
	int num_transfers;
425 426 427 428 429 430 431

	if (curlm_result != CURLM_OK &&
	    curlm_result != CURLM_CALL_MULTI_PERFORM) {
		active_requests--;
		slot->in_use = 0;
		return 0;
	}
432 433 434 435 436 437

	/*
	 * We know there must be something to do, since we just added
	 * something.
	 */
	curl_multi_perform(curlm, &num_transfers);
438 439 440 441 442
#endif
	return 1;
}

#ifdef USE_CURL_MULTI
443 444 445 446 447 448
struct fill_chain {
	void *data;
	int (*fill)(void *);
	struct fill_chain *next;
};

J
Junio C Hamano 已提交
449
static struct fill_chain *fill_cfg;
450 451 452

void add_fill_function(void *data, int (*fill)(void *))
{
453
	struct fill_chain *new = xmalloc(sizeof(*new));
454 455 456 457 458 459 460 461 462
	struct fill_chain **linkp = &fill_cfg;
	new->data = data;
	new->fill = fill;
	new->next = NULL;
	while (*linkp)
		linkp = &(*linkp)->next;
	*linkp = new;
}

463 464 465 466
void fill_active_slots(void)
{
	struct active_request_slot *slot = active_queue_head;

467 468 469 470 471 472 473
	while (active_requests < max_requests) {
		struct fill_chain *fill;
		for (fill = fill_cfg; fill; fill = fill->next)
			if (fill->fill(fill->data))
				break;

		if (!fill)
474
			break;
475
	}
476 477 478 479 480 481 482 483 484 485

	while (slot != NULL) {
		if (!slot->in_use && slot->curl != NULL) {
			curl_easy_cleanup(slot->curl);
			slot->curl = NULL;
		}
		slot = slot->next;
	}
}

486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510
void step_active_slots(void)
{
	int num_transfers;
	CURLMcode curlm_result;

	do {
		curlm_result = curl_multi_perform(curlm, &num_transfers);
	} while (curlm_result == CURLM_CALL_MULTI_PERFORM);
	if (num_transfers < active_requests) {
		process_curl_messages();
		fill_active_slots();
	}
}
#endif

void run_active_slot(struct active_request_slot *slot)
{
#ifdef USE_CURL_MULTI
	long last_pos = 0;
	long current_pos;
	fd_set readfds;
	fd_set writefds;
	fd_set excfds;
	int max_fd;
	struct timeval select_timeout;
N
Nick Hengeveld 已提交
511
	int finished = 0;
512

N
Nick Hengeveld 已提交
513 514
	slot->finished = &finished;
	while (!finished) {
515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543
		data_received = 0;
		step_active_slots();

		if (!data_received && slot->local != NULL) {
			current_pos = ftell(slot->local);
			if (current_pos > last_pos)
				data_received++;
			last_pos = current_pos;
		}

		if (slot->in_use && !data_received) {
			max_fd = 0;
			FD_ZERO(&readfds);
			FD_ZERO(&writefds);
			FD_ZERO(&excfds);
			select_timeout.tv_sec = 0;
			select_timeout.tv_usec = 50000;
			select(max_fd, &readfds, &writefds,
			       &excfds, &select_timeout);
		}
	}
#else
	while (slot->in_use) {
		slot->curl_result = curl_easy_perform(slot->curl);
		finish_active_slot(slot);
	}
#endif
}

544
static void closedown_active_slot(struct active_request_slot *slot)
545
{
M
Mike Hommey 已提交
546 547
	active_requests--;
	slot->in_use = 0;
548 549 550 551 552 553
}

void release_active_slot(struct active_request_slot *slot)
{
	closedown_active_slot(slot);
	if (slot->curl) {
N
Nick Hengeveld 已提交
554
#ifdef USE_CURL_MULTI
555
		curl_multi_remove_handle(curlm, slot->curl);
N
Nick Hengeveld 已提交
556
#endif
557 558 559
		curl_easy_cleanup(slot->curl);
		slot->curl = NULL;
	}
N
Nick Hengeveld 已提交
560
#ifdef USE_CURL_MULTI
561
	fill_active_slots();
N
Nick Hengeveld 已提交
562
#endif
563 564 565 566 567
}

static void finish_active_slot(struct active_request_slot *slot)
{
	closedown_active_slot(slot);
M
Mike Hommey 已提交
568
	curl_easy_getinfo(slot->curl, CURLINFO_HTTP_CODE, &slot->http_code);
569

N
Nick Hengeveld 已提交
570 571 572
	if (slot->finished != NULL)
		(*slot->finished) = 1;

573 574 575 576 577 578
	/* Store slot results so they can be read after the slot is reused */
	if (slot->results != NULL) {
		slot->results->curl_result = slot->curl_result;
		slot->results->http_code = slot->http_code;
	}

M
Mike Hommey 已提交
579
	/* Run callback if appropriate */
J
Junio C Hamano 已提交
580
	if (slot->callback_func != NULL)
M
Mike Hommey 已提交
581
		slot->callback_func(slot->callback_data);
582 583 584 585 586 587 588 589 590 591 592 593 594 595
}

void finish_all_active_slots(void)
{
	struct active_request_slot *slot = active_queue_head;

	while (slot != NULL)
		if (slot->in_use) {
			run_active_slot(slot);
			slot = active_queue_head;
		} else {
			slot = slot->next;
		}
}
596 597 598 599 600 601 602 603 604 605 606 607 608 609 610

static inline int needs_quote(int ch)
{
	if (((ch >= 'A') && (ch <= 'Z'))
			|| ((ch >= 'a') && (ch <= 'z'))
			|| ((ch >= '0') && (ch <= '9'))
			|| (ch == '/')
			|| (ch == '-')
			|| (ch == '.'))
		return 0;
	return 1;
}

static inline int hex(int v)
{
J
Junio C Hamano 已提交
611 612 613 614
	if (v < 10)
		return '0' + v;
	else
		return 'A' + v - 10;
615 616 617 618
}

static char *quote_ref_url(const char *base, const char *ref)
{
619
	struct strbuf buf = STRBUF_INIT;
620
	const char *cp;
621
	int ch;
622

623 624 625 626 627
	strbuf_addstr(&buf, base);
	if (buf.len && buf.buf[buf.len - 1] != '/' && *ref != '/')
		strbuf_addstr(&buf, "/");

	for (cp = ref; (ch = *cp) != 0; cp++)
628
		if (needs_quote(ch))
629
			strbuf_addf(&buf, "%%%02x", ch);
630
		else
631
			strbuf_addch(&buf, *cp);
632

633
	return strbuf_detach(&buf, NULL);
634 635
}

636
int http_fetch_ref(const char *base, struct ref *ref)
637 638 639 640 641 642 643
{
	char *url;
	struct strbuf buffer = STRBUF_INIT;
	struct active_request_slot *slot;
	struct slot_results results;
	int ret;

644
	url = quote_ref_url(base, ref->name);
645 646 647 648 649 650 651 652 653 654 655
	slot = get_active_slot();
	slot->results = &results;
	curl_easy_setopt(slot->curl, CURLOPT_FILE, &buffer);
	curl_easy_setopt(slot->curl, CURLOPT_WRITEFUNCTION, fwrite_buffer);
	curl_easy_setopt(slot->curl, CURLOPT_HTTPHEADER, NULL);
	curl_easy_setopt(slot->curl, CURLOPT_URL, url);
	if (start_active_slot(slot)) {
		run_active_slot(slot);
		if (results.curl_result == CURLE_OK) {
			strbuf_rtrim(&buffer);
			if (buffer.len == 40)
656
				ret = get_sha1_hex(buffer.buf, ref->old_sha1);
657 658 659 660
			else if (!prefixcmp(buffer.buf, "ref: ")) {
				ref->symref = xstrdup(buffer.buf + 5);
				ret = 0;
			} else
661 662 663
				ret = 1;
		} else {
			ret = error("Couldn't get %s for %s\n%s",
664
				    url, ref->name, curl_errorstr);
665 666 667 668 669 670 671 672 673
		}
	} else {
		ret = error("Unable to start request");
	}

	strbuf_release(&buffer);
	free(url);
	return ret;
}