Blame src/netops.c

Packit ae9e2a
/*
Packit ae9e2a
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit ae9e2a
 *
Packit ae9e2a
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit ae9e2a
 * a Linking Exception. For full terms see the included COPYING file.
Packit ae9e2a
 */
Packit ae9e2a
Packit ae9e2a
#include <ctype.h>
Packit ae9e2a
#include "git2/errors.h"
Packit ae9e2a
Packit ae9e2a
#include "common.h"
Packit ae9e2a
#include "netops.h"
Packit ae9e2a
#include "posix.h"
Packit ae9e2a
#include "buffer.h"
Packit ae9e2a
#include "http_parser.h"
Packit ae9e2a
#include "global.h"
Packit ae9e2a
Packit ae9e2a
int gitno_recv(gitno_buffer *buf)
Packit ae9e2a
{
Packit ae9e2a
	return buf->recv(buf);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void gitno_buffer_setup_callback(
Packit ae9e2a
	gitno_buffer *buf,
Packit ae9e2a
	char *data,
Packit ae9e2a
	size_t len,
Packit ae9e2a
	int (*recv)(gitno_buffer *buf), void *cb_data)
Packit ae9e2a
{
Packit ae9e2a
	memset(data, 0x0, len);
Packit ae9e2a
	buf->data = data;
Packit ae9e2a
	buf->len = len;
Packit ae9e2a
	buf->offset = 0;
Packit ae9e2a
	buf->recv = recv;
Packit ae9e2a
	buf->cb_data = cb_data;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int recv_stream(gitno_buffer *buf)
Packit ae9e2a
{
Packit ae9e2a
	git_stream *io = (git_stream *) buf->cb_data;
Packit ae9e2a
	int ret;
Packit ae9e2a
Packit ae9e2a
	ret = git_stream_read(io, buf->data + buf->offset, buf->len - buf->offset);
Packit ae9e2a
	if (ret < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	buf->offset += ret;
Packit ae9e2a
	return ret;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void gitno_buffer_setup_fromstream(git_stream *st, gitno_buffer *buf, char *data, size_t len)
Packit ae9e2a
{
Packit ae9e2a
	memset(data, 0x0, len);
Packit ae9e2a
	buf->data = data;
Packit ae9e2a
	buf->len = len;
Packit ae9e2a
	buf->offset = 0;
Packit ae9e2a
	buf->recv = recv_stream;
Packit ae9e2a
	buf->cb_data = st;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
/* Consume up to ptr and move the rest of the buffer to the beginning */
Packit ae9e2a
void gitno_consume(gitno_buffer *buf, const char *ptr)
Packit ae9e2a
{
Packit ae9e2a
	size_t consumed;
Packit ae9e2a
Packit ae9e2a
	assert(ptr - buf->data >= 0);
Packit ae9e2a
	assert(ptr - buf->data <= (int) buf->len);
Packit ae9e2a
Packit ae9e2a
	consumed = ptr - buf->data;
Packit ae9e2a
Packit ae9e2a
	memmove(buf->data, ptr, buf->offset - consumed);
Packit ae9e2a
	memset(buf->data + buf->offset, 0x0, buf->len - buf->offset);
Packit ae9e2a
	buf->offset -= consumed;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
/* Consume const bytes and move the rest of the buffer to the beginning */
Packit ae9e2a
void gitno_consume_n(gitno_buffer *buf, size_t cons)
Packit ae9e2a
{
Packit ae9e2a
	memmove(buf->data, buf->data + cons, buf->len - buf->offset);
Packit ae9e2a
	memset(buf->data + cons, 0x0, buf->len - buf->offset);
Packit ae9e2a
	buf->offset -= cons;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
/* Match host names according to RFC 2818 rules */
Packit ae9e2a
int gitno__match_host(const char *pattern, const char *host)
Packit ae9e2a
{
Packit ae9e2a
	for (;;) {
Packit ae9e2a
		char c = git__tolower(*pattern++);
Packit ae9e2a
Packit ae9e2a
		if (c == '\0')
Packit ae9e2a
			return *host ? -1 : 0;
Packit ae9e2a
Packit ae9e2a
		if (c == '*') {
Packit ae9e2a
			c = *pattern;
Packit ae9e2a
			/* '*' at the end matches everything left */
Packit ae9e2a
			if (c == '\0')
Packit ae9e2a
				return 0;
Packit ae9e2a
Packit ae9e2a
	/*
Packit ae9e2a
	 * We've found a pattern, so move towards the next matching
Packit ae9e2a
	 * char. The '.' is handled specially because wildcards aren't
Packit ae9e2a
	 * allowed to cross subdomains.
Packit ae9e2a
	 */
Packit ae9e2a
Packit ae9e2a
			while(*host) {
Packit ae9e2a
				char h = git__tolower(*host);
Packit ae9e2a
				if (c == h)
Packit ae9e2a
					return gitno__match_host(pattern, host++);
Packit ae9e2a
				if (h == '.')
Packit ae9e2a
					return gitno__match_host(pattern, host);
Packit ae9e2a
				host++;
Packit ae9e2a
			}
Packit ae9e2a
			return -1;
Packit ae9e2a
		}
Packit ae9e2a
Packit ae9e2a
		if (c != git__tolower(*host++))
Packit ae9e2a
			return -1;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return -1;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static const char *prefix_http = "http://";
Packit ae9e2a
static const char *prefix_https = "https://";
Packit ae9e2a
Packit ae9e2a
int gitno_connection_data_from_url(
Packit ae9e2a
		gitno_connection_data *data,
Packit ae9e2a
		const char *url,
Packit ae9e2a
		const char *service_suffix)
Packit ae9e2a
{
Packit ae9e2a
	int error = -1;
Packit ae9e2a
	const char *default_port = NULL, *path_search_start = NULL;
Packit ae9e2a
	char *original_host = NULL;
Packit ae9e2a
Packit ae9e2a
	/* service_suffix is optional */
Packit ae9e2a
	assert(data && url);
Packit ae9e2a
Packit ae9e2a
	/* Save these for comparison later */
Packit ae9e2a
	original_host = data->host;
Packit ae9e2a
	data->host = NULL;
Packit ae9e2a
	gitno_connection_data_free_ptrs(data);
Packit ae9e2a
Packit ae9e2a
	if (!git__prefixcmp(url, prefix_http)) {
Packit ae9e2a
		path_search_start = url + strlen(prefix_http);
Packit ae9e2a
		default_port = "80";
Packit ae9e2a
Packit ae9e2a
		if (data->use_ssl) {
Packit ae9e2a
			giterr_set(GITERR_NET, "redirect from HTTPS to HTTP is not allowed");
Packit ae9e2a
			goto cleanup;
Packit ae9e2a
		}
Packit ae9e2a
	} else if (!git__prefixcmp(url, prefix_https)) {
Packit ae9e2a
		path_search_start = url + strlen(prefix_https);
Packit ae9e2a
		default_port = "443";
Packit ae9e2a
		data->use_ssl = true;
Packit ae9e2a
	} else if (url[0] == '/')
Packit ae9e2a
		default_port = data->use_ssl ? "443" : "80";
Packit ae9e2a
Packit ae9e2a
	if (!default_port) {
Packit ae9e2a
		giterr_set(GITERR_NET, "unrecognized URL prefix");
Packit ae9e2a
		goto cleanup;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	error = gitno_extract_url_parts(
Packit ae9e2a
		&data->host, &data->port, &data->path, &data->user, &data->pass,
Packit ae9e2a
		url, default_port);
Packit ae9e2a
Packit ae9e2a
	if (url[0] == '/') {
Packit ae9e2a
		/* Relative redirect; reuse original host name and port */
Packit ae9e2a
		path_search_start = url;
Packit ae9e2a
		git__free(data->host);
Packit ae9e2a
		data->host = original_host;
Packit ae9e2a
		original_host = NULL;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (!error) {
Packit ae9e2a
		const char *path = strchr(path_search_start, '/');
Packit ae9e2a
		size_t pathlen = strlen(path);
Packit ae9e2a
		size_t suffixlen = service_suffix ? strlen(service_suffix) : 0;
Packit ae9e2a
Packit ae9e2a
		if (suffixlen &&
Packit ae9e2a
		    !memcmp(path + pathlen - suffixlen, service_suffix, suffixlen)) {
Packit ae9e2a
			git__free(data->path);
Packit ae9e2a
			data->path = git__strndup(path, pathlen - suffixlen);
Packit ae9e2a
		} else {
Packit ae9e2a
			git__free(data->path);
Packit ae9e2a
			data->path = git__strdup(path);
Packit ae9e2a
		}
Packit ae9e2a
Packit ae9e2a
		/* Check for errors in the resulting data */
Packit ae9e2a
		if (original_host && url[0] != '/' && strcmp(original_host, data->host)) {
Packit ae9e2a
			giterr_set(GITERR_NET, "cross host redirect not allowed");
Packit ae9e2a
			error = -1;
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
cleanup:
Packit ae9e2a
	if (original_host) git__free(original_host);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void gitno_connection_data_free_ptrs(gitno_connection_data *d)
Packit ae9e2a
{
Packit ae9e2a
	git__free(d->host); d->host = NULL;
Packit ae9e2a
	git__free(d->port); d->port = NULL;
Packit ae9e2a
	git__free(d->path); d->path = NULL;
Packit ae9e2a
	git__free(d->user); d->user = NULL;
Packit ae9e2a
	git__free(d->pass); d->pass = NULL;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
#define hex2c(c) ((c | 32) % 39 - 9)
Packit ae9e2a
static char* unescape(char *str)
Packit ae9e2a
{
Packit ae9e2a
	int x, y;
Packit ae9e2a
	int len = (int)strlen(str);
Packit ae9e2a
Packit ae9e2a
	for (x=y=0; str[y]; ++x, ++y) {
Packit ae9e2a
		if ((str[x] = str[y]) == '%') {
Packit ae9e2a
			if (y < len-2 && isxdigit(str[y+1]) && isxdigit(str[y+2])) {
Packit ae9e2a
				str[x] = (hex2c(str[y+1]) << 4) + hex2c(str[y+2]);
Packit ae9e2a
				y += 2;
Packit ae9e2a
			}
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
	str[x] = '\0';
Packit ae9e2a
	return str;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int gitno_extract_url_parts(
Packit ae9e2a
		char **host,
Packit ae9e2a
		char **port,
Packit ae9e2a
		char **path,
Packit ae9e2a
		char **username,
Packit ae9e2a
		char **password,
Packit ae9e2a
		const char *url,
Packit ae9e2a
		const char *default_port)
Packit ae9e2a
{
Packit ae9e2a
	struct http_parser_url u = {0};
Packit ae9e2a
	const char *_host, *_port, *_path, *_userinfo;
Packit ae9e2a
Packit ae9e2a
	if (http_parser_parse_url(url, strlen(url), false, &u)) {
Packit ae9e2a
		giterr_set(GITERR_NET, "malformed URL '%s'", url);
Packit ae9e2a
		return GIT_EINVALIDSPEC;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	_host = url+u.field_data[UF_HOST].off;
Packit ae9e2a
	_port = url+u.field_data[UF_PORT].off;
Packit ae9e2a
	_path = url+u.field_data[UF_PATH].off;
Packit ae9e2a
	_userinfo = url+u.field_data[UF_USERINFO].off;
Packit ae9e2a
Packit ae9e2a
	if (u.field_set & (1 << UF_HOST)) {
Packit ae9e2a
		*host = git__substrdup(_host, u.field_data[UF_HOST].len);
Packit ae9e2a
		GITERR_CHECK_ALLOC(*host);
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (u.field_set & (1 << UF_PORT))
Packit ae9e2a
		*port = git__substrdup(_port, u.field_data[UF_PORT].len);
Packit ae9e2a
	else
Packit ae9e2a
		*port = git__strdup(default_port);
Packit ae9e2a
	GITERR_CHECK_ALLOC(*port);
Packit ae9e2a
Packit ae9e2a
	if (path) {
Packit ae9e2a
		if (u.field_set & (1 << UF_PATH)) {
Packit ae9e2a
			*path = git__substrdup(_path, u.field_data[UF_PATH].len);
Packit ae9e2a
			GITERR_CHECK_ALLOC(*path);
Packit ae9e2a
		} else {
Packit ae9e2a
			git__free(*port);
Packit ae9e2a
			*port = NULL;
Packit ae9e2a
			git__free(*host);
Packit ae9e2a
			*host = NULL;
Packit ae9e2a
			giterr_set(GITERR_NET, "invalid url, missing path");
Packit ae9e2a
			return GIT_EINVALIDSPEC;
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (u.field_set & (1 << UF_USERINFO)) {
Packit ae9e2a
		const char *colon = memchr(_userinfo, ':', u.field_data[UF_USERINFO].len);
Packit ae9e2a
		if (colon) {
Packit ae9e2a
			*username = unescape(git__substrdup(_userinfo, colon - _userinfo));
Packit ae9e2a
			*password = unescape(git__substrdup(colon+1, u.field_data[UF_USERINFO].len - (colon+1-_userinfo)));
Packit ae9e2a
			GITERR_CHECK_ALLOC(*password);
Packit ae9e2a
		} else {
Packit ae9e2a
			*username = git__substrdup(_userinfo, u.field_data[UF_USERINFO].len);
Packit ae9e2a
		}
Packit ae9e2a
		GITERR_CHECK_ALLOC(*username);
Packit ae9e2a
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}