test_helpers.c 7.08 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
/*
 * This file is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License, version 2,
 * as published by the Free Software Foundation.
 *
 * In addition to the permissions in the GNU General Public License,
 * the authors give you unlimited permission to link the compiled
 * version of this file into combinations with other programs,
 * and to distribute those combinations without any restriction
 * coming from the use of this file.  (The General Public License
 * restrictions do apply in other respects; for example, they cover
 * modification of the file, and distribution when not linked into
 * a combined executable.)
 *
 * This file is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; see the file COPYING.  If not, write to
 * the Free Software Foundation, 51 Franklin Street, Fifth Floor,
 * Boston, MA 02110-1301, USA.
 */

26
#include "common.h"
27 28 29 30 31
#include "test_helpers.h"
#include "fileops.h"

int write_object_data(char *file, void *data, size_t len)
{
32 33
	git_file fd;
	int ret;
34

35 36 37 38
	if ((fd = gitfo_creat(file, S_IREAD | S_IWRITE)) < 0)
		return -1;
	ret = gitfo_write(fd, data, len);
	gitfo_close(fd);
39

40
	return ret;
41 42 43 44
}

int write_object_files(const char *odb_dir, object_data *d)
{
45 46 47 48 49 50 51 52
	if (gitfo_mkdir(odb_dir, 0755) < 0) {
		int err = errno;
		fprintf(stderr, "can't make directory \"%s\"", odb_dir);
		if (err == EEXIST)
			fprintf(stderr, " (already exists)");
		fprintf(stderr, "\n");
		return -1;
	}
53

54 55 56 57 58 59 60 61
	if ((gitfo_mkdir(d->dir, 0755) < 0) && (errno != EEXIST)) {
		fprintf(stderr, "can't make object directory \"%s\"\n", d->dir);
		return -1;
	}
	if (write_object_data(d->file, d->bytes, d->blen) < 0) {
		fprintf(stderr, "can't write object file \"%s\"\n", d->file);
		return -1;
	}
62

63
	return 0;
64 65 66 67
}

int remove_object_files(const char *odb_dir, object_data *d)
{
68 69 70 71 72 73 74 75
	if (gitfo_unlink(d->file) < 0) {
		fprintf(stderr, "can't delete object file \"%s\"\n", d->file);
		return -1;
	}
	if ((gitfo_rmdir(d->dir) < 0) && (errno != ENOTEMPTY)) {
		fprintf(stderr, "can't remove object directory \"%s\"\n", d->dir);
		return -1;
	}
76

77 78 79 80
	if (gitfo_rmdir(odb_dir) < 0) {
		fprintf(stderr, "can't remove directory \"%s\"\n", odb_dir);
		return -1;
	}
81

82
	return 0;
83 84
}

85
int remove_loose_object(const char *repository_folder, git_object *object)
86
{
87 88
	static const char *objects_folder = "objects/";

89
	char *ptr, *full_path, *top_folder;
90 91 92
	int path_length, objects_length;

	assert(repository_folder && object);
93

94 95 96
	objects_length = strlen(objects_folder);
	path_length = strlen(repository_folder);
	ptr = full_path = git__malloc(path_length + objects_length + GIT_OID_HEXSZ + 3);
97

98 99
	strcpy(ptr, repository_folder);
	strcpy(ptr + path_length, objects_folder);
100

101
	ptr = top_folder = ptr + path_length + objects_length;
102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
	*ptr++ = '/';
	git_oid_pathfmt(ptr, git_object_id(object));
	ptr += GIT_OID_HEXSZ + 1;
	*ptr = 0;

	if (gitfo_unlink(full_path) < 0) {
		fprintf(stderr, "can't delete object file \"%s\"\n", full_path);
		return -1;
	}

	*top_folder = 0;

	if ((gitfo_rmdir(full_path) < 0) && (errno != ENOTEMPTY)) {
		fprintf(stderr, "can't remove object directory \"%s\"\n", full_path);
		return -1;
	}

	free(full_path);

	return GIT_SUCCESS;
}

124
int cmp_objects(git_rawobj *o, object_data *d)
125
{
126
	if (o->type != git_object_string2type(d->type))
127 128 129 130 131 132
		return -1;
	if (o->len != d->dlen)
		return -1;
	if ((o->len > 0) && (memcmp(o->data, d->data, o->len) != 0))
		return -1;
	return 0;
133
}
134 135 136 137 138 139 140 141 142 143

int copy_file(const char *src, const char *dst)
{
	gitfo_buf source_buf;
	git_file dst_fd;
	int error = GIT_ERROR;

	if (gitfo_read_file(&source_buf, src) < GIT_SUCCESS)
		return GIT_ENOTFOUND;

144
	dst_fd = gitfo_creat_force(dst, 0644);
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
	if (dst_fd < 0)
		goto cleanup;

	error = gitfo_write(dst_fd, source_buf.data, source_buf.len);

cleanup:
	gitfo_free_buf(&source_buf);
	gitfo_close(dst_fd);

	return error;
}

int cmp_files(const char *a, const char *b)
{
	gitfo_buf buf_a, buf_b;
	int error = GIT_ERROR;

	if (gitfo_read_file(&buf_a, a) < GIT_SUCCESS)
		return GIT_ERROR;

	if (gitfo_read_file(&buf_b, b) < GIT_SUCCESS) {
		gitfo_free_buf(&buf_a);
		return GIT_ERROR;
	}

	if (buf_a.len == buf_b.len && !memcmp(buf_a.data, buf_b.data, buf_a.len))
		error = GIT_SUCCESS;

	gitfo_free_buf(&buf_a);
	gitfo_free_buf(&buf_b);

	return error;
}
178 179 180 181 182 183 184 185

static int remove_filesystem_element_recurs(void *GIT_UNUSED(nil), char *path)
{
	int error = GIT_SUCCESS;

	GIT_UNUSED_ARG(nil);

	error = gitfo_isdir(path);
186 187 188 189
	if (error == GIT_SUCCESS) {
		size_t root_size = strlen(path);

		error = gitfo_dirent(path, GIT_PATH_MAX, remove_filesystem_element_recurs, NULL);
190 191 192
		if (error < GIT_SUCCESS)
			return error;

193
		path[root_size] = 0;
194 195 196 197 198 199
		return rmdir(path);
	}

	return gitfo_unlink(path);
}

200
int rmdir_recurs(const char *directory_path)
201
{
202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218
	char buffer[GIT_PATH_MAX];
	strcpy(buffer, directory_path);
	return remove_filesystem_element_recurs(NULL, buffer);
}

typedef struct {
	size_t src_len, dst_len;
	char *dst;
} copydir_data;

static int copy_filesystem_element_recurs(void *_data, char *source)
{
	copydir_data *data = (copydir_data *)_data;

	data->dst[data->dst_len] = 0;
	git__joinpath(data->dst, data->dst, source + data->src_len);

219
	if (gitfo_isdir(source) == GIT_SUCCESS)
220 221 222 223 224
		return gitfo_dirent(source, GIT_PATH_MAX, copy_filesystem_element_recurs, _data);

	return copy_file(source, data->dst);
}

225
int copydir_recurs(const char *source_directory_path, const char *destination_directory_path)
226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242
{
	char source_buffer[GIT_PATH_MAX];
	char dest_buffer[GIT_PATH_MAX];
	copydir_data data;

	/* Source has to exist, Destination hast to _not_ exist */
	if (gitfo_isdir(source_directory_path)  || !gitfo_isdir(destination_directory_path))
		return GIT_EINVALIDPATH;

	git__joinpath(source_buffer, source_directory_path, "");
	data.src_len = strlen(source_buffer);

	git__joinpath(dest_buffer, destination_directory_path, "");
	data.dst = dest_buffer;
	data.dst_len = strlen(dest_buffer);

	return copy_filesystem_element_recurs(&data, source_buffer);
243
}
244 245 246 247 248 249 250 251 252 253 254 255 256 257 258

int open_temp_repo(git_repository **repo, const char *path)
{
	int error;
	if ((error = copydir_recurs(path, TEMP_REPO_FOLDER)) < GIT_SUCCESS)
		return error;

	return git_repository_open(repo, TEMP_REPO_FOLDER);
}

void close_temp_repo(git_repository *repo)
{
	git_repository_free(repo);
	rmdir_recurs(TEMP_REPO_FOLDER);
}
259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286

static int remove_placeholders_recurs(void *filename, char *path)
{
	char passed_filename[GIT_PATH_MAX];
	char *data = (char *)filename;

	if (!gitfo_isdir(path))
		return gitfo_dirent(path, GIT_PATH_MAX, remove_placeholders_recurs, data);

	 if (git__basename_r(passed_filename, sizeof(passed_filename), path) < GIT_SUCCESS)
		 return GIT_EINVALIDPATH;

	if (!strcmp(data, passed_filename))
		return gitfo_unlink(path);

	return GIT_SUCCESS;
}

int remove_placeholders(char *directory_path, char *filename)
{
	char buffer[GIT_PATH_MAX];

	if (gitfo_isdir(directory_path))
		return GIT_EINVALIDPATH;

	strcpy(buffer, directory_path);
	return remove_placeholders_recurs(filename, buffer);
}