1 #include "clar_libgit2.h"
6 static git_remote
*_remote
;
7 static git_repository
*_repo
;
8 static const git_refspec
*_refspec
;
10 void test_network_remote_remotes__initialize(void)
12 _repo
= cl_git_sandbox_init("testrepo.git");
14 cl_git_pass(git_remote_load(&_remote
, _repo
, "test"));
16 _refspec
= git_remote_get_refspec(_remote
, 0);
17 cl_assert(_refspec
!= NULL
);
20 void test_network_remote_remotes__cleanup(void)
22 git_remote_free(_remote
);
25 cl_git_sandbox_cleanup();
28 void test_network_remote_remotes__parsing(void)
30 git_remote
*_remote2
= NULL
;
32 cl_assert_equal_s(git_remote_name(_remote
), "test");
33 cl_assert_equal_s(git_remote_url(_remote
), "git://github.com/libgit2/libgit2");
34 cl_assert(git_remote_pushurl(_remote
) == NULL
);
36 cl_assert_equal_s(git_remote__urlfordirection(_remote
, GIT_DIRECTION_FETCH
),
37 "git://github.com/libgit2/libgit2");
38 cl_assert_equal_s(git_remote__urlfordirection(_remote
, GIT_DIRECTION_PUSH
),
39 "git://github.com/libgit2/libgit2");
41 cl_git_pass(git_remote_load(&_remote2
, _repo
, "test_with_pushurl"));
42 cl_assert_equal_s(git_remote_name(_remote2
), "test_with_pushurl");
43 cl_assert_equal_s(git_remote_url(_remote2
), "git://github.com/libgit2/fetchlibgit2");
44 cl_assert_equal_s(git_remote_pushurl(_remote2
), "git://github.com/libgit2/pushlibgit2");
46 cl_assert_equal_s(git_remote__urlfordirection(_remote2
, GIT_DIRECTION_FETCH
),
47 "git://github.com/libgit2/fetchlibgit2");
48 cl_assert_equal_s(git_remote__urlfordirection(_remote2
, GIT_DIRECTION_PUSH
),
49 "git://github.com/libgit2/pushlibgit2");
51 git_remote_free(_remote2
);
54 void test_network_remote_remotes__pushurl(void)
56 cl_git_pass(git_remote_set_pushurl(_remote
, "git://github.com/libgit2/notlibgit2"));
57 cl_assert_equal_s(git_remote_pushurl(_remote
), "git://github.com/libgit2/notlibgit2");
59 cl_git_pass(git_remote_set_pushurl(_remote
, NULL
));
60 cl_assert(git_remote_pushurl(_remote
) == NULL
);
63 void test_network_remote_remotes__error_when_not_found(void)
66 cl_git_fail_with(git_remote_load(&r
, _repo
, "does-not-exist"), GIT_ENOTFOUND
);
68 cl_assert(giterr_last() != NULL
);
69 cl_assert(giterr_last()->klass
== GITERR_CONFIG
);
72 void test_network_remote_remotes__error_when_no_push_available(void)
77 cl_git_pass(git_remote_create_anonymous(&r
, _repo
, cl_fixture("testrepo.git"), NULL
));
79 cl_git_pass(git_remote_set_transport(r
, git_transport_local
, NULL
));
81 cl_git_pass(git_remote_connect(r
, GIT_DIRECTION_PUSH
));
83 /* Make sure that push is really not available */
84 r
->transport
->push
= NULL
;
86 cl_git_pass(git_push_new(&p
, r
));
87 cl_git_pass(git_push_add_refspec(p
, "refs/heads/master"));
88 cl_git_fail_with(git_push_finish(p
), GIT_ERROR
);
94 void test_network_remote_remotes__refspec_parsing(void)
96 cl_assert_equal_s(git_refspec_src(_refspec
), "refs/heads/*");
97 cl_assert_equal_s(git_refspec_dst(_refspec
), "refs/remotes/test/*");
100 void test_network_remote_remotes__add_fetchspec(void)
104 size
= git_remote_refspec_count(_remote
);
106 cl_git_pass(git_remote_add_fetch(_remote
, "refs/*:refs/*"));
109 cl_assert_equal_i((int)size
, (int)git_remote_refspec_count(_remote
));
111 _refspec
= git_remote_get_refspec(_remote
, size
- 1);
112 cl_assert_equal_s(git_refspec_src(_refspec
), "refs/*");
113 cl_assert_equal_s(git_refspec_dst(_refspec
), "refs/*");
114 cl_assert_equal_s(git_refspec_string(_refspec
), "refs/*:refs/*");
115 cl_assert_equal_b(_refspec
->push
, false);
118 void test_network_remote_remotes__dup(void)
123 cl_git_pass(git_remote_dup(&dup
, _remote
));
125 cl_assert_equal_s(git_remote_name(dup
), git_remote_name(_remote
));
126 cl_assert_equal_s(git_remote_url(dup
), git_remote_url(_remote
));
127 cl_assert_equal_s(git_remote_pushurl(dup
), git_remote_pushurl(_remote
));
129 cl_git_pass(git_remote_get_fetch_refspecs(&array
, _remote
));
130 cl_assert_equal_i(1, (int)array
.count
);
131 cl_assert_equal_s("+refs/heads/*:refs/remotes/test/*", array
.strings
[0]);
132 git_strarray_free(&array
);
134 cl_git_pass(git_remote_get_push_refspecs(&array
, _remote
));
135 cl_assert_equal_i(0, (int)array
.count
);
136 git_strarray_free(&array
);
138 git_remote_free(dup
);
141 void test_network_remote_remotes__add_pushspec(void)
145 size
= git_remote_refspec_count(_remote
);
147 cl_git_pass(git_remote_add_push(_remote
, "refs/*:refs/*"));
149 cl_assert_equal_i((int)size
, (int)git_remote_refspec_count(_remote
));
151 _refspec
= git_remote_get_refspec(_remote
, size
- 1);
152 cl_assert_equal_s(git_refspec_src(_refspec
), "refs/*");
153 cl_assert_equal_s(git_refspec_dst(_refspec
), "refs/*");
154 cl_assert_equal_s(git_refspec_string(_refspec
), "refs/*:refs/*");
156 cl_assert_equal_b(_refspec
->push
, true);
159 void test_network_remote_remotes__save(void)
162 const char *fetch_refspec1
= "refs/heads/ns1/*:refs/remotes/upstream/ns1/*";
163 const char *fetch_refspec2
= "refs/heads/ns2/*:refs/remotes/upstream/ns2/*";
164 const char *push_refspec1
= "refs/heads/ns1/*:refs/heads/ns1/*";
165 const char *push_refspec2
= "refs/heads/ns2/*:refs/heads/ns2/*";
167 git_remote_free(_remote
);
170 /* Set up the remote and save it to config */
171 cl_git_pass(git_remote_create(&_remote
, _repo
, "upstream", "git://github.com/libgit2/libgit2"));
172 git_remote_clear_refspecs(_remote
);
174 cl_git_pass(git_remote_add_fetch(_remote
, fetch_refspec1
));
175 cl_git_pass(git_remote_add_fetch(_remote
, fetch_refspec2
));
176 cl_git_pass(git_remote_add_push(_remote
, push_refspec1
));
177 cl_git_pass(git_remote_add_push(_remote
, push_refspec2
));
178 cl_git_pass(git_remote_set_pushurl(_remote
, "git://github.com/libgit2/libgit2_push"));
179 cl_git_pass(git_remote_save(_remote
));
180 git_remote_free(_remote
);
183 /* Load it from config and make sure everything matches */
184 cl_git_pass(git_remote_load(&_remote
, _repo
, "upstream"));
186 cl_git_pass(git_remote_get_fetch_refspecs(&array
, _remote
));
187 cl_assert_equal_i(2, (int)array
.count
);
188 cl_assert_equal_s(fetch_refspec1
, array
.strings
[0]);
189 cl_assert_equal_s(fetch_refspec2
, array
.strings
[1]);
190 git_strarray_free(&array
);
192 cl_git_pass(git_remote_get_push_refspecs(&array
, _remote
));
193 cl_assert_equal_i(2, (int)array
.count
);
194 cl_assert_equal_s(push_refspec1
, array
.strings
[0]);
195 cl_assert_equal_s(push_refspec2
, array
.strings
[1]);
196 git_strarray_free(&array
);
198 cl_assert_equal_s(git_remote_url(_remote
), "git://github.com/libgit2/libgit2");
199 cl_assert_equal_s(git_remote_pushurl(_remote
), "git://github.com/libgit2/libgit2_push");
201 /* remove the pushurl again and see if we can save that too */
202 cl_git_pass(git_remote_set_pushurl(_remote
, NULL
));
203 cl_git_pass(git_remote_save(_remote
));
204 git_remote_free(_remote
);
207 cl_git_pass(git_remote_load(&_remote
, _repo
, "upstream"));
208 cl_assert(git_remote_pushurl(_remote
) == NULL
);
211 void test_network_remote_remotes__fnmatch(void)
213 cl_assert(git_refspec_src_matches(_refspec
, "refs/heads/master"));
214 cl_assert(git_refspec_src_matches(_refspec
, "refs/heads/multi/level/branch"));
217 void test_network_remote_remotes__transform(void)
219 git_buf ref
= GIT_BUF_INIT
;
221 cl_git_pass(git_refspec_transform(&ref
, _refspec
, "refs/heads/master"));
222 cl_assert_equal_s(ref
.ptr
, "refs/remotes/test/master");
226 void test_network_remote_remotes__transform_destination_to_source(void)
228 git_buf ref
= GIT_BUF_INIT
;
230 cl_git_pass(git_refspec_rtransform(&ref
, _refspec
, "refs/remotes/test/master"));
231 cl_assert_equal_s(ref
.ptr
, "refs/heads/master");
235 void test_network_remote_remotes__missing_refspecs(void)
239 git_remote_free(_remote
);
242 cl_git_pass(git_repository_config(&cfg
, _repo
));
243 cl_git_pass(git_config_set_string(cfg
, "remote.specless.url", "http://example.com"));
244 cl_git_pass(git_remote_load(&_remote
, _repo
, "specless"));
246 git_config_free(cfg
);
249 void test_network_remote_remotes__nonmatch_upstream_refspec(void)
254 "refs/tags/*:refs/tags/*",
256 git_strarray specs
= {
261 cl_git_pass(git_remote_create(&remote
, _repo
, "taggy", git_repository_path(_repo
)));
264 * Set the current branch's upstream remote to a dummy ref so we call into the code
265 * which tries to check for the current branch's upstream in the refspecs
267 cl_git_pass(git_repository_config(&config
, _repo
));
268 cl_git_pass(git_config_set_string(config
, "branch.master.remote", "taggy"));
269 cl_git_pass(git_config_set_string(config
, "branch.master.merge", "refs/heads/foo"));
271 cl_git_pass(git_remote_fetch(remote
, &specs
, NULL
, NULL
));
273 git_remote_free(remote
);
276 void test_network_remote_remotes__list(void)
281 cl_git_pass(git_remote_list(&list
, _repo
));
282 cl_assert(list
.count
== 5);
283 git_strarray_free(&list
);
285 cl_git_pass(git_repository_config(&cfg
, _repo
));
287 /* Create a new remote */
288 cl_git_pass(git_config_set_string(cfg
, "remote.specless.url", "http://example.com"));
290 /* Update a remote (previously without any url/pushurl entry) */
291 cl_git_pass(git_config_set_string(cfg
, "remote.no-remote-url.pushurl", "http://example.com"));
293 cl_git_pass(git_remote_list(&list
, _repo
));
294 cl_assert(list
.count
== 7);
295 git_strarray_free(&list
);
297 git_config_free(cfg
);
300 void test_network_remote_remotes__loading_a_missing_remote_returns_ENOTFOUND(void)
302 git_remote_free(_remote
);
305 cl_assert_equal_i(GIT_ENOTFOUND
, git_remote_load(&_remote
, _repo
, "just-left-few-minutes-ago"));
308 void test_network_remote_remotes__loading_with_an_invalid_name_returns_EINVALIDSPEC(void)
310 git_remote_free(_remote
);
313 cl_assert_equal_i(GIT_EINVALIDSPEC
, git_remote_load(&_remote
, _repo
, "Inv@{id"));
317 * $ git remote add addtest http://github.com/libgit2/libgit2
322 * url = http://github.com/libgit2/libgit2
323 * fetch = +refs/heads/\*:refs/remotes/addtest/\*
325 void test_network_remote_remotes__add(void)
327 git_remote_free(_remote
);
330 cl_git_pass(git_remote_create(&_remote
, _repo
, "addtest", "http://github.com/libgit2/libgit2"));
331 cl_assert_equal_i(GIT_REMOTE_DOWNLOAD_TAGS_AUTO
, git_remote_autotag(_remote
));
333 git_remote_free(_remote
);
336 cl_git_pass(git_remote_load(&_remote
, _repo
, "addtest"));
337 cl_assert_equal_i(GIT_REMOTE_DOWNLOAD_TAGS_AUTO
, git_remote_autotag(_remote
));
339 _refspec
= git_vector_get(&_remote
->refspecs
, 0);
340 cl_assert_equal_s("refs/heads/*", git_refspec_src(_refspec
));
341 cl_assert(git_refspec_force(_refspec
) == 1);
342 cl_assert_equal_s("refs/remotes/addtest/*", git_refspec_dst(_refspec
));
343 cl_assert_equal_s(git_remote_url(_remote
), "http://github.com/libgit2/libgit2");
346 void test_network_remote_remotes__cannot_add_a_nameless_remote(void)
352 git_remote_create(&remote
, _repo
, NULL
, "git://github.com/libgit2/libgit2"));
355 void test_network_remote_remotes__cannot_save_an_inmemory_remote(void)
359 cl_git_pass(git_remote_create_anonymous(&remote
, _repo
, "git://github.com/libgit2/libgit2", NULL
));
361 cl_assert_equal_p(NULL
, git_remote_name(remote
));
363 cl_git_fail(git_remote_save(remote
));
364 git_remote_free(remote
);
367 void test_network_remote_remotes__cannot_add_a_remote_with_an_invalid_name(void)
369 git_remote
*remote
= NULL
;
373 git_remote_create(&remote
, _repo
, "Inv@{id", "git://github.com/libgit2/libgit2"));
374 cl_assert_equal_p(remote
, NULL
);
378 git_remote_create(&remote
, _repo
, "", "git://github.com/libgit2/libgit2"));
379 cl_assert_equal_p(remote
, NULL
);
382 void test_network_remote_remotes__tagopt(void)
387 cl_git_pass(git_repository_config(&cfg
, _repo
));
389 git_remote_set_autotag(_remote
, GIT_REMOTE_DOWNLOAD_TAGS_ALL
);
390 cl_git_pass(git_remote_save(_remote
));
391 cl_git_pass(git_config_get_string(&opt
, cfg
, "remote.test.tagopt"));
392 cl_assert_equal_s("--tags", opt
);
394 git_remote_set_autotag(_remote
, GIT_REMOTE_DOWNLOAD_TAGS_NONE
);
395 cl_git_pass(git_remote_save(_remote
));
396 cl_git_pass(git_config_get_string(&opt
, cfg
, "remote.test.tagopt"));
397 cl_assert_equal_s("--no-tags", opt
);
399 git_remote_set_autotag(_remote
, GIT_REMOTE_DOWNLOAD_TAGS_AUTO
);
400 cl_git_pass(git_remote_save(_remote
));
401 cl_assert(git_config_get_string(&opt
, cfg
, "remote.test.tagopt") == GIT_ENOTFOUND
);
403 git_config_free(cfg
);
406 void test_network_remote_remotes__can_load_with_an_empty_url(void)
408 git_remote
*remote
= NULL
;
410 cl_git_pass(git_remote_load(&remote
, _repo
, "empty-remote-url"));
412 cl_assert(remote
->url
== NULL
);
413 cl_assert(remote
->pushurl
== NULL
);
415 cl_git_fail(git_remote_connect(remote
, GIT_DIRECTION_FETCH
));
417 cl_assert(giterr_last() != NULL
);
418 cl_assert(giterr_last()->klass
== GITERR_INVALID
);
420 git_remote_free(remote
);
423 void test_network_remote_remotes__can_load_with_only_an_empty_pushurl(void)
425 git_remote
*remote
= NULL
;
427 cl_git_pass(git_remote_load(&remote
, _repo
, "empty-remote-pushurl"));
429 cl_assert(remote
->url
== NULL
);
430 cl_assert(remote
->pushurl
== NULL
);
432 cl_git_fail(git_remote_connect(remote
, GIT_DIRECTION_FETCH
));
434 git_remote_free(remote
);
437 void test_network_remote_remotes__returns_ENOTFOUND_when_neither_url_nor_pushurl(void)
439 git_remote
*remote
= NULL
;
442 git_remote_load(&remote
, _repo
, "no-remote-url"), GIT_ENOTFOUND
);
445 void assert_cannot_create_remote(const char *name
, int expected_error
)
447 git_remote
*remote
= NULL
;
450 git_remote_create(&remote
, _repo
, name
, "git://github.com/libgit2/libgit2"),
453 cl_assert_equal_p(remote
, NULL
);
456 void test_network_remote_remotes__cannot_create_a_remote_which_name_conflicts_with_an_existing_remote(void)
458 assert_cannot_create_remote("test", GIT_EEXISTS
);
461 void test_network_remote_remotes__cannot_create_a_remote_which_name_is_invalid(void)
463 assert_cannot_create_remote("/", GIT_EINVALIDSPEC
);
464 assert_cannot_create_remote("//", GIT_EINVALIDSPEC
);
465 assert_cannot_create_remote(".lock", GIT_EINVALIDSPEC
);
466 assert_cannot_create_remote("a.lock", GIT_EINVALIDSPEC
);
469 void test_network_remote_remote__git_remote_create_with_fetchspec(void)
474 cl_git_pass(git_remote_create_with_fetchspec(&remote
, _repo
, "test-new", "git://github.com/libgit2/libgit2", "+refs/*:refs/*"));
475 git_remote_get_fetch_refspecs(&array
, remote
);
476 cl_assert_equal_s("+refs/*:refs/*", array
.strings
[0]);
477 git_remote_free(remote
);
480 static const char *fetch_refspecs
[] = {
481 "+refs/heads/*:refs/remotes/origin/*",
482 "refs/tags/*:refs/tags/*",
483 "+refs/pull/*:refs/pull/*",
486 static const char *push_refspecs
[] = {
487 "refs/heads/*:refs/heads/*",
488 "refs/tags/*:refs/tags/*",
489 "refs/notes/*:refs/notes/*",
492 void test_network_remote_remotes__query_refspecs(void)
498 cl_git_pass(git_remote_create_anonymous(&remote
, _repo
, "git://github.com/libgit2/libgit2", NULL
));
500 for (i
= 0; i
< 3; i
++) {
501 cl_git_pass(git_remote_add_fetch(remote
, fetch_refspecs
[i
]));
502 cl_git_pass(git_remote_add_push(remote
, push_refspecs
[i
]));
505 cl_git_pass(git_remote_get_fetch_refspecs(&array
, remote
));
506 for (i
= 0; i
< 3; i
++) {
507 cl_assert_equal_s(fetch_refspecs
[i
], array
.strings
[i
]);
509 git_strarray_free(&array
);
511 cl_git_pass(git_remote_get_push_refspecs(&array
, remote
));
512 for (i
= 0; i
< 3; i
++) {
513 cl_assert_equal_s(push_refspecs
[i
], array
.strings
[i
]);
515 git_strarray_free(&array
);
517 git_remote_free(remote
);
520 static int remote_single_branch(git_remote
**out
, git_repository
*repo
, const char *name
, const char *url
, void *payload
)
522 char *fetch_refspecs
[] = {
523 "refs/heads/first-merge:refs/remotes/origin/first-merge",
525 git_strarray fetch_refspecs_strarray
= {
532 cl_git_pass(git_remote_create(out
, repo
, name
, url
));
533 cl_git_pass(git_remote_set_fetch_refspecs(*out
, &fetch_refspecs_strarray
));
538 void test_network_remote_remotes__fetch_from_anonymous(void)
542 cl_git_pass(git_remote_create_anonymous(&remote
, _repo
, cl_fixture("testrepo.git"),
543 "refs/heads/*:refs/other/*"));
544 cl_git_pass(git_remote_fetch(remote
, NULL
, NULL
, NULL
));
545 git_remote_free(remote
);
548 void test_network_remote_remotes__single_branch(void)
550 git_clone_options opts
= GIT_CLONE_OPTIONS_INIT
;
551 git_repository
*repo
;
555 opts
.remote_cb
= remote_single_branch
;
556 opts
.checkout_branch
= "first-merge";
558 cl_git_pass(git_clone(&repo
, "git://github.com/libgit2/TestGitRepository", "./single-branch", &opts
));
559 cl_git_pass(git_reference_list(&refs
, repo
));
561 for (i
= 0; i
< refs
.count
; i
++) {
562 if (!git__prefixcmp(refs
.strings
[i
], "refs/heads/"))
565 cl_assert_equal_i(1, count
);
567 git_strarray_free(&refs
);
568 git_repository_free(repo
);
571 void test_network_remote_remotes__restricted_refspecs(void)
573 git_clone_options opts
= GIT_CLONE_OPTIONS_INIT
;
574 git_repository
*repo
;
576 opts
.remote_cb
= remote_single_branch
;
578 cl_git_fail_with(GIT_EINVALIDSPEC
, git_clone(&repo
, "git://github.com/libgit2/TestGitRepository", "./restrict-refspec", &opts
));