clar_libgit2.h 7.53 KB
Newer Older
1 2
#ifndef __CLAR_LIBGIT2__
#define __CLAR_LIBGIT2__
Vicent Marti committed
3

4
#include "clar.h"
Vicent Marti committed
5 6
#include <git2.h>
#include "common.h"
7
#include "posix.h"
Vicent Marti committed
8 9

/**
10 11
 * Replace for `clar_must_pass` that passes the last library error as the
 * test failure message.
Vicent Marti committed
12
 *
13
 * Use this wrapper around all `git_` library calls that return error codes!
Vicent Marti committed
14
 */
15
#define cl_git_pass(expr) cl_git_expect((expr), 0, __FILE__, __func__, __LINE__)
16

17
#define cl_git_fail_with(error, expr) cl_git_expect((expr), error, __FILE__, __func__, __LINE__)
18

19
#define cl_git_expect(expr, expected, file, func, line) do { \
20
	int _lg2_error; \
21
	git_error_clear(); \
22
	if ((_lg2_error = (expr)) != expected) \
23
		cl_git_report_failure(_lg2_error, expected, file, func, line, "Function call failed: " #expr); \
24
	} while (0)
Vicent Marti committed
25 26

/**
27
 * Wrapper for `clar_must_fail` -- this one is
Vicent Marti committed
28 29 30
 * just for consistency. Use with `git_` library
 * calls that are supposed to fail!
 */
31 32
#define cl_git_fail(expr) do { \
	if ((expr) == 0) \
33
		git_error_clear(), \
34
		cl_git_report_failure(0, 0, __FILE__, __func__, __LINE__, "Function call succeeded: " #expr); \
35
	} while (0)
36

37 38 39 40 41 42
/**
 * Like cl_git_pass, only for Win32 error code conventions
 */
#define cl_win32_pass(expr) do { \
	int _win32_res; \
	if ((_win32_res = (expr)) == 0) { \
43
		git_error_set(GIT_ERROR_OS, "Returned: %d, system error code: %lu", _win32_res, GetLastError()); \
44
		cl_git_report_failure(_win32_res, 0, __FILE__, __func__, __LINE__, "System call failed: " #expr); \
45 46 47
	} \
	} while(0)

48 49 50 51 52 53 54 55 56 57 58 59 60
/**
 * Thread safe assertions; you cannot use `cl_git_report_failure` from a
 * child thread since it will try to `longjmp` to abort and "the effect of
 * a call to longjmp() where initialization of the jmp_buf structure was
 * not performed in the calling thread is undefined."
 *
 * Instead, callers can provide a clar thread error context to a thread,
 * which will populate and return it on failure.  Callers can check the
 * status with `cl_git_thread_check`.
 */
typedef struct {
	int error;
	const char *file;
61
	const char *func;
62 63 64 65 66
	int line;
	const char *expr;
	char error_msg[4096];
} cl_git_thread_err;

67
#ifdef GIT_THREADS
68
# define cl_git_thread_pass(threaderr, expr) cl_git_thread_pass_(threaderr, (expr), __FILE__, __func__, __LINE__)
69 70 71
#else
# define cl_git_thread_pass(threaderr, expr) cl_git_pass(expr)
#endif
72

73
#define cl_git_thread_pass_(__threaderr, __expr, __file, __func, __line) do { \
74
	git_error_clear(); \
75
	if ((((cl_git_thread_err *)__threaderr)->error = (__expr)) != 0) { \
76
		const git_error *_last = git_error_last(); \
77
		((cl_git_thread_err *)__threaderr)->file = __file; \
78
		((cl_git_thread_err *)__threaderr)->func = __func; \
79 80 81 82 83 84 85 86 87
		((cl_git_thread_err *)__threaderr)->line = __line; \
		((cl_git_thread_err *)__threaderr)->expr = "Function call failed: " #__expr; \
		p_snprintf(((cl_git_thread_err *)__threaderr)->error_msg, 4096, "thread 0x%" PRIxZ " - error %d - %s", \
			git_thread_currentid(), ((cl_git_thread_err *)__threaderr)->error, \
			_last ? _last->message : "<no message>"); \
		git_thread_exit(__threaderr); \
	} \
	} while (0)

88
GIT_INLINE(void) cl_git_thread_check(void *data)
89 90 91
{
	cl_git_thread_err *threaderr = (cl_git_thread_err *)data;
	if (threaderr->error != 0)
92
		clar__assert(0, threaderr->file, threaderr->func, threaderr->line, threaderr->expr, threaderr->error_msg, 1);
93 94
}

95
void cl_git_report_failure(int, int, const char *, const char *, int, const char *);
96

97 98
#define cl_assert_at_line(expr,file,func,line) \
	clar__assert((expr) != 0, file, func, line, "Expression is not true: " #expr, NULL, 1)
99

100 101
GIT_INLINE(void) clar__assert_in_range(
	int lo, int val, int hi,
102 103
	const char *file, const char *func, int line,
	const char *err, int should_abort)
104 105 106
{
	if (lo > val || hi < val) {
		char buf[128];
107
		p_snprintf(buf, sizeof(buf), "%d not in [%d,%d]", val, lo, hi);
108
		clar__fail(file, func, line, err, buf, should_abort);
109 110 111
	}
}

112
#define cl_assert_equal_sz(sz1,sz2) do { \
nulltoken committed
113
	size_t __sz1 = (size_t)(sz1), __sz2 = (size_t)(sz2); \
114
	clar__assert_equal(__FILE__,__func__,__LINE__,#sz1 " != " #sz2, 1, "%"PRIuZ, __sz1, __sz2); \
115 116
} while (0)

117
#define cl_assert_in_range(L,V,H) \
118
	clar__assert_in_range((L),(V),(H),__FILE__,__func__,__LINE__,"Range check: " #V " in [" #L "," #H "]", 1)
119

120
#define cl_assert_equal_file(DATA,SIZE,PATH) \
121
	clar__assert_equal_file(DATA,SIZE,0,PATH,__FILE__,__func__,(int)__LINE__)
122 123

#define cl_assert_equal_file_ignore_cr(DATA,SIZE,PATH) \
124
	clar__assert_equal_file(DATA,SIZE,1,PATH,__FILE__,__func__,(int)__LINE__)
125 126 127 128 129 130 131

void clar__assert_equal_file(
	const char *expected_data,
	size_t expected_size,
	int ignore_cr,
	const char *path,
	const char *file,
132
	const char *func,
nulltoken committed
133
	int line);
134

135
GIT_INLINE(void) clar__assert_equal_oid(
136
	const char *file, const char *func, int line, const char *desc,
137 138 139 140 141 142 143 144
	const git_oid *one, const git_oid *two)
{
	if (git_oid_cmp(one, two)) {
		char err[] = "\"........................................\" != \"........................................\"";

		git_oid_fmt(&err[1], one);
		git_oid_fmt(&err[47], two);

145
		clar__fail(file, func, line, desc, err, 1);
146 147 148 149
	}
}

#define cl_assert_equal_oid(one, two) \
150
	clar__assert_equal_oid(__FILE__, __func__, __LINE__, \
151 152
		"OID mismatch: " #one " != " #two, (one), (two))

153 154 155 156 157 158 159 160 161
/*
 * Some utility macros for building long strings
 */
#define REP4(STR)	 STR STR STR STR
#define REP15(STR)	 REP4(STR) REP4(STR) REP4(STR) STR STR STR
#define REP16(STR)	 REP4(REP4(STR))
#define REP256(STR)  REP16(REP16(STR))
#define REP1024(STR) REP4(REP256(STR))

162 163
/* Write the contents of a buffer to disk */
void cl_git_mkfile(const char *filename, const char *content);
164
void cl_git_append2file(const char *filename, const char *new_content);
165
void cl_git_rewritefile(const char *filename, const char *new_content);
166 167
void cl_git_write2file(const char *path, const char *data,
	size_t datalen, int flags, unsigned int mode);
168
void cl_git_rmfile(const char *filename);
169

170 171 172
bool cl_toggle_filemode(const char *filename);
bool cl_is_chmod_supported(void);

173 174
/* Environment wrappers */
char *cl_getenv(const char *name);
175
bool cl_is_env_set(const char *name);
176 177
int cl_setenv(const char *name, const char *value);

178 179 180
/* Reliable rename */
int cl_rename(const char *source, const char *dest);

181 182 183
/* Git sandbox setup helpers */

git_repository *cl_git_sandbox_init(const char *sandbox);
184
git_repository *cl_git_sandbox_init_new(const char *name);
185
void cl_git_sandbox_cleanup(void);
186
git_repository *cl_git_sandbox_reopen(void);
187

188 189 190 191 192 193 194
/*
 * build a sandbox-relative from path segments
 * is_dir will add a trailing slash
 * vararg must be a NULL-terminated char * list
 */
const char *cl_git_sandbox_path(int is_dir, ...);

195 196 197 198
/* Local-repo url helpers */
const char* cl_git_fixture_url(const char *fixturename);
const char* cl_git_path_url(const char *path);

199 200 201
/* Test repository cleaner */
int cl_git_remove_placeholders(const char *directory_path, const char *filename);

202 203 204 205 206 207 208 209
/* commit creation helpers */
void cl_repo_commit_from_index(
	git_oid *out,
	git_repository *repo,
	git_signature *sig,
	git_time_t time,
	const char *msg);

210 211
/* config setting helpers */
void cl_repo_set_bool(git_repository *repo, const char *cfg, int value);
212
int cl_repo_get_bool(git_repository *repo, const char *cfg);
213

214 215
void cl_repo_set_string(git_repository *repo, const char *cfg, const char *value);

216 217 218 219 220 221 222
/* set up a fake "home" directory and set libgit2 GLOBAL search path.
 *
 * automatically configures cleanup function to restore the regular search
 * path, although you can call it explicitly if you wish (with NULL).
 */
void cl_fake_home(void);
void cl_fake_home_cleanup(void *);
223

224 225
void cl_sandbox_set_search_path_defaults(void);

226
#ifdef GIT_WIN32
227 228 229 230 231 232
# define cl_msleep(x) Sleep(x)
#else
# define cl_msleep(x) usleep(1000 * (x))
#endif

#ifdef GIT_WIN32
233 234 235
bool cl_sandbox_supports_8dot3(void);
#endif

Vicent Marti committed
236
#endif