Blame src/transport.c

Packit Service 20376f
/*
Packit Service 20376f
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit Service 20376f
 *
Packit Service 20376f
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit Service 20376f
 * a Linking Exception. For full terms see the included COPYING file.
Packit Service 20376f
 */
Packit Service 20376f
#include "common.h"
Packit Service 20376f
#include "git2/types.h"
Packit Service 20376f
#include "git2/remote.h"
Packit Service 20376f
#include "git2/net.h"
Packit Service 20376f
#include "git2/transport.h"
Packit Service 20376f
#include "git2/sys/transport.h"
Packit Service 20376f
#include "path.h"
Packit Service 20376f
Packit Service 20376f
typedef struct transport_definition {
Packit Service 20376f
	char *prefix;
Packit Service 20376f
	git_transport_cb fn;
Packit Service 20376f
	void *param;
Packit Service 20376f
} transport_definition;
Packit Service 20376f
Packit Service 20376f
static git_smart_subtransport_definition http_subtransport_definition = { git_smart_subtransport_http, 1, NULL };
Packit Service 20376f
static git_smart_subtransport_definition git_subtransport_definition = { git_smart_subtransport_git, 0, NULL };
Packit Service 20376f
#ifdef GIT_SSH
Packit Service 20376f
static git_smart_subtransport_definition ssh_subtransport_definition = { git_smart_subtransport_ssh, 0, NULL };
Packit Service 20376f
#endif
Packit Service 20376f
Packit Service 20376f
static transport_definition local_transport_definition = { "file://", git_transport_local, NULL };
Packit Service 20376f
Packit Service 20376f
static transport_definition transports[] = {
Packit Service 20376f
	{ "git://",   git_transport_smart, &git_subtransport_definition },
Packit Service 20376f
	{ "http://",  git_transport_smart, &http_subtransport_definition },
Packit Service 20376f
	{ "https://", git_transport_smart, &http_subtransport_definition },
Packit Service 20376f
	{ "file://",  git_transport_local, NULL },
Packit Service 20376f
#ifdef GIT_SSH
Packit Service 20376f
	{ "ssh://",   git_transport_smart, &ssh_subtransport_definition },
Packit Service 20376f
	{ "ssh+git://",   git_transport_smart, &ssh_subtransport_definition },
Packit Service 20376f
	{ "git+ssh://",   git_transport_smart, &ssh_subtransport_definition },
Packit Service 20376f
#endif
Packit Service 20376f
	{ NULL, 0, 0 }
Packit Service 20376f
};
Packit Service 20376f
Packit Service 20376f
static git_vector custom_transports = GIT_VECTOR_INIT;
Packit Service 20376f
Packit Service 20376f
#define GIT_TRANSPORT_COUNT (sizeof(transports)/sizeof(transports[0])) - 1
Packit Service 20376f
Packit Service 20376f
static transport_definition * transport_find_by_url(const char *url)
Packit Service 20376f
{
Packit Service 20376f
	size_t i = 0;
Packit Service 20376f
	transport_definition *d;
Packit Service 20376f
Packit Service 20376f
	/* Find a user transport who wants to deal with this URI */
Packit Service 20376f
	git_vector_foreach(&custom_transports, i, d) {
Packit Service 20376f
		if (strncasecmp(url, d->prefix, strlen(d->prefix)) == 0) {
Packit Service 20376f
			return d;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* Find a system transport for this URI */
Packit Service 20376f
	for (i = 0; i < GIT_TRANSPORT_COUNT; ++i) {
Packit Service 20376f
		d = &transports[i];
Packit Service 20376f
Packit Service 20376f
		if (strncasecmp(url, d->prefix, strlen(d->prefix)) == 0) {
Packit Service 20376f
			return d;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return NULL;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static int transport_find_fn(
Packit Service 20376f
	git_transport_cb *out,
Packit Service 20376f
	const char *url,
Packit Service 20376f
	void **param)
Packit Service 20376f
{
Packit Service 20376f
	transport_definition *definition = transport_find_by_url(url);
Packit Service 20376f
Packit Service 20376f
#ifdef GIT_WIN32
Packit Service 20376f
	/* On Windows, it might not be possible to discern between absolute local
Packit Service 20376f
	 * and ssh paths - first check if this is a valid local path that points
Packit Service 20376f
	 * to a directory and if so assume local path, else assume SSH */
Packit Service 20376f
Packit Service 20376f
	/* Check to see if the path points to a file on the local file system */
Packit Service 20376f
	if (!definition && git_path_exists(url) && git_path_isdir(url))
Packit Service 20376f
		definition = &local_transport_definition;
Packit Service 20376f
#endif
Packit Service 20376f
Packit Service 20376f
	/* For other systems, perform the SSH check first, to avoid going to the
Packit Service 20376f
	 * filesystem if it is not necessary */
Packit Service 20376f
Packit Service 20376f
	/* It could be a SSH remote path. Check to see if there's a : */
Packit Service 20376f
	if (!definition && strrchr(url, ':')) {
Packit Service 20376f
		/* re-search transports again with ssh:// as url
Packit Service 20376f
		 * so that we can find a third party ssh transport */
Packit Service 20376f
		definition = transport_find_by_url("ssh://");
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
#ifndef GIT_WIN32
Packit Service 20376f
	/* Check to see if the path points to a file on the local file system */
Packit Service 20376f
	if (!definition && git_path_exists(url) && git_path_isdir(url))
Packit Service 20376f
		definition = &local_transport_definition;
Packit Service 20376f
#endif
Packit Service 20376f
Packit Service 20376f
	if (!definition)
Packit Service 20376f
		return GIT_ENOTFOUND;
Packit Service 20376f
Packit Service 20376f
	*out = definition->fn;
Packit Service 20376f
	*param = definition->param;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/**************
Packit Service 20376f
 * Public API *
Packit Service 20376f
 **************/
Packit Service 20376f
Packit Service 20376f
int git_transport_new(git_transport **out, git_remote *owner, const char *url)
Packit Service 20376f
{
Packit Service 20376f
	git_transport_cb fn;
Packit Service 20376f
	git_transport *transport;
Packit Service 20376f
	void *param;
Packit Service 20376f
	int error;
Packit Service 20376f
Packit Service 20376f
	if ((error = transport_find_fn(&fn, url, &param)) == GIT_ENOTFOUND) {
Packit Service 20376f
		giterr_set(GITERR_NET, "unsupported URL protocol");
Packit Service 20376f
		return -1;
Packit Service 20376f
	} else if (error < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	if ((error = fn(&transport, owner, param)) < 0)
Packit Service 20376f
		return error;
Packit Service 20376f
Packit Service 20376f
	GITERR_CHECK_VERSION(transport, GIT_TRANSPORT_VERSION, "git_transport");
Packit Service 20376f
Packit Service 20376f
	*out = transport;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_transport_register(
Packit Service 20376f
	const char *scheme,
Packit Service 20376f
	git_transport_cb cb,
Packit Service 20376f
	void *param)
Packit Service 20376f
{
Packit Service 20376f
	git_buf prefix = GIT_BUF_INIT;
Packit Service 20376f
	transport_definition *d, *definition = NULL;
Packit Service 20376f
	size_t i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	assert(scheme);
Packit Service 20376f
	assert(cb);
Packit Service 20376f
Packit Service 20376f
	if ((error = git_buf_printf(&prefix, "%s://", scheme)) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	git_vector_foreach(&custom_transports, i, d) {
Packit Service 20376f
		if (strcasecmp(d->prefix, prefix.ptr) == 0) {
Packit Service 20376f
			error = GIT_EEXISTS;
Packit Service 20376f
			goto on_error;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	definition = git__calloc(1, sizeof(transport_definition));
Packit Service 20376f
	GITERR_CHECK_ALLOC(definition);
Packit Service 20376f
Packit Service 20376f
	definition->prefix = git_buf_detach(&prefix);
Packit Service 20376f
	definition->fn = cb;
Packit Service 20376f
	definition->param = param;
Packit Service 20376f
Packit Service 20376f
	if (git_vector_insert(&custom_transports, definition) < 0)
Packit Service 20376f
		goto on_error;
Packit Service 20376f
Packit Service 20376f
	return 0;
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	git_buf_free(&prefix);
Packit Service 20376f
	git__free(definition);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_transport_unregister(const char *scheme)
Packit Service 20376f
{
Packit Service 20376f
	git_buf prefix = GIT_BUF_INIT;
Packit Service 20376f
	transport_definition *d;
Packit Service 20376f
	size_t i;
Packit Service 20376f
	int error = 0;
Packit Service 20376f
Packit Service 20376f
	assert(scheme);
Packit Service 20376f
Packit Service 20376f
	if ((error = git_buf_printf(&prefix, "%s://", scheme)) < 0)
Packit Service 20376f
		goto done;
Packit Service 20376f
Packit Service 20376f
	git_vector_foreach(&custom_transports, i, d) {
Packit Service 20376f
		if (strcasecmp(d->prefix, prefix.ptr) == 0) {
Packit Service 20376f
			if ((error = git_vector_remove(&custom_transports, i)) < 0)
Packit Service 20376f
				goto done;
Packit Service 20376f
Packit Service 20376f
			git__free(d->prefix);
Packit Service 20376f
			git__free(d);
Packit Service 20376f
Packit Service 20376f
			if (!custom_transports.length)
Packit Service 20376f
				git_vector_free(&custom_transports);
Packit Service 20376f
Packit Service 20376f
			error = 0;
Packit Service 20376f
			goto done;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	error = GIT_ENOTFOUND;
Packit Service 20376f
Packit Service 20376f
done:
Packit Service 20376f
	git_buf_free(&prefix);
Packit Service 20376f
	return error;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_transport_init(git_transport *opts, unsigned int version)
Packit Service 20376f
{
Packit Service 20376f
	GIT_INIT_STRUCTURE_FROM_TEMPLATE(
Packit Service 20376f
		opts, version, git_transport, GIT_TRANSPORT_INIT);
Packit Service 20376f
	return 0;
Packit Service 20376f
}