]> git.ipfire.org Git - thirdparty/git.git/blob - t/t2400-worktree-add.sh
Merge branch 'js/contributor-docs-updates'
[thirdparty/git.git] / t / t2400-worktree-add.sh
1 #!/bin/sh
2
3 test_description='test git worktree add'
4
5 GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main
6 export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME
7
8 TEST_CREATE_REPO_NO_TEMPLATE=1
9 . ./test-lib.sh
10
11 . "$TEST_DIRECTORY"/lib-rebase.sh
12
13 test_expect_success 'setup' '
14 test_commit init
15 '
16
17 test_expect_success '"add" an existing worktree' '
18 mkdir -p existing/subtree &&
19 test_must_fail git worktree add --detach existing main
20 '
21
22 test_expect_success '"add" an existing empty worktree' '
23 mkdir existing_empty &&
24 git worktree add --detach existing_empty main
25 '
26
27 test_expect_success '"add" using shorthand - fails when no previous branch' '
28 test_must_fail git worktree add existing_short -
29 '
30
31 test_expect_success '"add" using - shorthand' '
32 git checkout -b newbranch &&
33 echo hello >myworld &&
34 git add myworld &&
35 git commit -m myworld &&
36 git checkout main &&
37 git worktree add short-hand - &&
38 echo refs/heads/newbranch >expect &&
39 git -C short-hand rev-parse --symbolic-full-name HEAD >actual &&
40 test_cmp expect actual
41 '
42
43 test_expect_success '"add" refuses to checkout locked branch' '
44 test_must_fail git worktree add zere main &&
45 ! test -d zere &&
46 ! test -d .git/worktrees/zere
47 '
48
49 test_expect_success 'checking out paths not complaining about linked checkouts' '
50 (
51 cd existing_empty &&
52 echo dirty >>init.t &&
53 git checkout main -- init.t
54 )
55 '
56
57 test_expect_success '"add" worktree' '
58 git rev-parse HEAD >expect &&
59 git worktree add --detach here main &&
60 (
61 cd here &&
62 test_cmp ../init.t init.t &&
63 test_must_fail git symbolic-ref HEAD &&
64 git rev-parse HEAD >actual &&
65 test_cmp ../expect actual &&
66 git fsck
67 )
68 '
69
70 test_expect_success '"add" worktree with lock' '
71 git worktree add --detach --lock here-with-lock main &&
72 test_when_finished "git worktree unlock here-with-lock || :" &&
73 test -f .git/worktrees/here-with-lock/locked
74 '
75
76 test_expect_success '"add" worktree with lock and reason' '
77 lock_reason="why not" &&
78 git worktree add --detach --lock --reason "$lock_reason" here-with-lock-reason main &&
79 test_when_finished "git worktree unlock here-with-lock-reason || :" &&
80 test -f .git/worktrees/here-with-lock-reason/locked &&
81 echo "$lock_reason" >expect &&
82 test_cmp expect .git/worktrees/here-with-lock-reason/locked
83 '
84
85 test_expect_success '"add" worktree with reason but no lock' '
86 test_must_fail git worktree add --detach --reason "why not" here-with-reason-only main &&
87 test_path_is_missing .git/worktrees/here-with-reason-only/locked
88 '
89
90 test_expect_success '"add" worktree from a subdir' '
91 (
92 mkdir sub &&
93 cd sub &&
94 git worktree add --detach here main &&
95 cd here &&
96 test_cmp ../../init.t init.t
97 )
98 '
99
100 test_expect_success '"add" from a linked checkout' '
101 (
102 cd here &&
103 git worktree add --detach nested-here main &&
104 cd nested-here &&
105 git fsck
106 )
107 '
108
109 test_expect_success '"add" worktree creating new branch' '
110 git worktree add -b newmain there main &&
111 (
112 cd there &&
113 test_cmp ../init.t init.t &&
114 git symbolic-ref HEAD >actual &&
115 echo refs/heads/newmain >expect &&
116 test_cmp expect actual &&
117 git fsck
118 )
119 '
120
121 test_expect_success 'die the same branch is already checked out' '
122 (
123 cd here &&
124 test_must_fail git checkout newmain 2>actual &&
125 grep "already used by worktree at" actual
126 )
127 '
128
129 test_expect_success 'refuse to reset a branch in use elsewhere' '
130 (
131 cd here &&
132
133 # we know we are on detached HEAD but just in case ...
134 git checkout --detach HEAD &&
135 git rev-parse --verify HEAD >old.head &&
136
137 git rev-parse --verify refs/heads/newmain >old.branch &&
138 test_must_fail git checkout -B newmain 2>error &&
139 git rev-parse --verify refs/heads/newmain >new.branch &&
140 git rev-parse --verify HEAD >new.head &&
141
142 grep "already used by worktree at" error &&
143 test_cmp old.branch new.branch &&
144 test_cmp old.head new.head &&
145
146 # and we must be still on the same detached HEAD state
147 test_must_fail git symbolic-ref HEAD
148 )
149 '
150
151 test_expect_success SYMLINKS 'die the same branch is already checked out (symlink)' '
152 head=$(git -C there rev-parse --git-path HEAD) &&
153 ref=$(git -C there symbolic-ref HEAD) &&
154 rm "$head" &&
155 ln -s "$ref" "$head" &&
156 test_must_fail git -C here checkout newmain
157 '
158
159 test_expect_success 'not die the same branch is already checked out' '
160 (
161 cd here &&
162 git worktree add --force anothernewmain newmain
163 )
164 '
165
166 test_expect_success 'not die on re-checking out current branch' '
167 (
168 cd there &&
169 git checkout newmain
170 )
171 '
172
173 test_expect_success '"add" from a bare repo' '
174 (
175 git clone --bare . bare &&
176 cd bare &&
177 git worktree add -b bare-main ../there2 main
178 )
179 '
180
181 test_expect_success 'checkout from a bare repo without "add"' '
182 (
183 cd bare &&
184 test_must_fail git checkout main
185 )
186 '
187
188 test_expect_success '"add" default branch of a bare repo' '
189 (
190 git clone --bare . bare2 &&
191 cd bare2 &&
192 git worktree add ../there3 main &&
193 cd ../there3 &&
194 # Simple check that a Git command does not
195 # immediately fail with the current setup
196 git status
197 ) &&
198 cat >expect <<-EOF &&
199 init.t
200 EOF
201 ls there3 >actual &&
202 test_cmp expect actual
203 '
204
205 test_expect_success '"add" to bare repo with worktree config' '
206 (
207 git clone --bare . bare3 &&
208 cd bare3 &&
209 git config extensions.worktreeconfig true &&
210
211 # Add config values that are erroneous to have in
212 # a config.worktree file outside of the main
213 # working tree, to check that Git filters them out
214 # when copying config during "git worktree add".
215 git config --worktree core.bare true &&
216 git config --worktree core.worktree "$(pwd)" &&
217
218 # We want to check that bogus.key is copied
219 git config --worktree bogus.key value &&
220 git config --unset core.bare &&
221 git worktree add ../there4 main &&
222 cd ../there4 &&
223
224 # Simple check that a Git command does not
225 # immediately fail with the current setup
226 git status &&
227 git worktree add --detach ../there5 &&
228 cd ../there5 &&
229 git status
230 ) &&
231
232 # the worktree has the arbitrary value copied.
233 test_cmp_config -C there4 value bogus.key &&
234 test_cmp_config -C there5 value bogus.key &&
235
236 # however, core.bare and core.worktree were removed.
237 test_must_fail git -C there4 config core.bare &&
238 test_must_fail git -C there4 config core.worktree &&
239
240 cat >expect <<-EOF &&
241 init.t
242 EOF
243
244 ls there4 >actual &&
245 test_cmp expect actual &&
246 ls there5 >actual &&
247 test_cmp expect actual
248 '
249
250 test_expect_success 'checkout with grafts' '
251 test_when_finished rm .git/info/grafts &&
252 test_commit abc &&
253 SHA1=$(git rev-parse HEAD) &&
254 test_commit def &&
255 test_commit xyz &&
256 mkdir .git/info &&
257 echo "$(git rev-parse HEAD) $SHA1" >.git/info/grafts &&
258 cat >expected <<-\EOF &&
259 xyz
260 abc
261 EOF
262 git log --format=%s -2 >actual &&
263 test_cmp expected actual &&
264 git worktree add --detach grafted main &&
265 git --git-dir=grafted/.git log --format=%s -2 >actual &&
266 test_cmp expected actual
267 '
268
269 test_expect_success '"add" from relative HEAD' '
270 test_commit a &&
271 test_commit b &&
272 test_commit c &&
273 git rev-parse HEAD~1 >expected &&
274 git worktree add relhead HEAD~1 &&
275 git -C relhead rev-parse HEAD >actual &&
276 test_cmp expected actual
277 '
278
279 test_expect_success '"add -b" with <branch> omitted' '
280 git worktree add -b burble flornk &&
281 test_cmp_rev HEAD burble
282 '
283
284 test_expect_success '"add --detach" with <branch> omitted' '
285 git worktree add --detach fishhook &&
286 git rev-parse HEAD >expected &&
287 git -C fishhook rev-parse HEAD >actual &&
288 test_cmp expected actual &&
289 test_must_fail git -C fishhook symbolic-ref HEAD
290 '
291
292 test_expect_success '"add" with <branch> omitted' '
293 git worktree add wiffle/bat &&
294 test_cmp_rev HEAD bat
295 '
296
297 test_expect_success '"add" checks out existing branch of dwimd name' '
298 git branch dwim HEAD~1 &&
299 git worktree add dwim &&
300 test_cmp_rev HEAD~1 dwim &&
301 (
302 cd dwim &&
303 test_cmp_rev HEAD dwim
304 )
305 '
306
307 test_expect_success '"add <path>" dwim fails with checked out branch' '
308 git checkout -b test-branch &&
309 test_must_fail git worktree add test-branch &&
310 test_path_is_missing test-branch
311 '
312
313 test_expect_success '"add --force" with existing dwimd name doesnt die' '
314 git checkout test-branch &&
315 git worktree add --force test-branch
316 '
317
318 test_expect_success '"add" no auto-vivify with --detach and <branch> omitted' '
319 git worktree add --detach mish/mash &&
320 test_must_fail git rev-parse mash -- &&
321 test_must_fail git -C mish/mash symbolic-ref HEAD
322 '
323
324 # Helper function to test mutually exclusive options.
325 #
326 # Note: Quoted arguments containing spaces are not supported.
327 test_wt_add_excl () {
328 local opts="$*" &&
329 test_expect_success "'worktree add' with '$opts' has mutually exclusive options" '
330 test_must_fail git worktree add $opts 2>actual &&
331 grep -E "fatal:( options)? .* cannot be used together" actual
332 '
333 }
334
335 test_wt_add_excl -b poodle -B poodle bamboo main
336 test_wt_add_excl -b poodle --detach bamboo main
337 test_wt_add_excl -B poodle --detach bamboo main
338 test_wt_add_excl --orphan --detach bamboo
339 test_wt_add_excl --orphan --no-checkout bamboo
340 test_wt_add_excl --orphan bamboo main
341 test_wt_add_excl --orphan -b bamboo wtdir/ main
342
343 test_expect_success '"add -B" fails if the branch is checked out' '
344 git rev-parse newmain >before &&
345 test_must_fail git worktree add -B newmain bamboo main &&
346 git rev-parse newmain >after &&
347 test_cmp before after
348 '
349
350 test_expect_success 'add -B' '
351 git worktree add -B poodle bamboo2 main^ &&
352 git -C bamboo2 symbolic-ref HEAD >actual &&
353 echo refs/heads/poodle >expected &&
354 test_cmp expected actual &&
355 test_cmp_rev main^ poodle
356 '
357
358 test_expect_success 'add --quiet' '
359 test_when_finished "git worktree remove -f -f another-worktree" &&
360 git worktree add --quiet another-worktree main 2>actual &&
361 test_must_be_empty actual
362 '
363
364 test_expect_success 'add --quiet -b' '
365 test_when_finished "git branch -D quietnewbranch" &&
366 test_when_finished "git worktree remove -f -f another-worktree" &&
367 git worktree add --quiet -b quietnewbranch another-worktree 2>actual &&
368 test_must_be_empty actual
369 '
370
371 test_expect_success '"add --orphan"' '
372 test_when_finished "git worktree remove -f -f orphandir" &&
373 git worktree add --orphan -b neworphan orphandir &&
374 echo refs/heads/neworphan >expected &&
375 git -C orphandir symbolic-ref HEAD >actual &&
376 test_cmp expected actual
377 '
378
379 test_expect_success '"add --orphan (no -b)"' '
380 test_when_finished "git worktree remove -f -f neworphan" &&
381 git worktree add --orphan neworphan &&
382 echo refs/heads/neworphan >expected &&
383 git -C neworphan symbolic-ref HEAD >actual &&
384 test_cmp expected actual
385 '
386
387 test_expect_success '"add --orphan --quiet"' '
388 test_when_finished "git worktree remove -f -f orphandir" &&
389 git worktree add --quiet --orphan -b neworphan orphandir 2>log.actual &&
390 test_must_be_empty log.actual &&
391 echo refs/heads/neworphan >expected &&
392 git -C orphandir symbolic-ref HEAD >actual &&
393 test_cmp expected actual
394 '
395
396 test_expect_success '"add --orphan" fails if the branch already exists' '
397 test_when_finished "git branch -D existingbranch" &&
398 git worktree add -b existingbranch orphandir main &&
399 git worktree remove orphandir &&
400 test_must_fail git worktree add --orphan -b existingbranch orphandir
401 '
402
403 test_expect_success '"add --orphan" with empty repository' '
404 test_when_finished "rm -rf empty_repo" &&
405 echo refs/heads/newbranch >expected &&
406 GIT_DIR="empty_repo" git init --bare &&
407 git -C empty_repo worktree add --orphan -b newbranch worktreedir &&
408 git -C empty_repo/worktreedir symbolic-ref HEAD >actual &&
409 test_cmp expected actual
410 '
411
412 test_expect_success '"add" worktree with orphan branch and lock' '
413 git worktree add --lock --orphan -b orphanbr orphan-with-lock &&
414 test_when_finished "git worktree unlock orphan-with-lock || :" &&
415 test -f .git/worktrees/orphan-with-lock/locked
416 '
417
418 test_expect_success '"add" worktree with orphan branch, lock, and reason' '
419 lock_reason="why not" &&
420 git worktree add --detach --lock --reason "$lock_reason" orphan-with-lock-reason main &&
421 test_when_finished "git worktree unlock orphan-with-lock-reason || :" &&
422 test -f .git/worktrees/orphan-with-lock-reason/locked &&
423 echo "$lock_reason" >expect &&
424 test_cmp expect .git/worktrees/orphan-with-lock-reason/locked
425 '
426
427 # Note: Quoted arguments containing spaces are not supported.
428 test_wt_add_orphan_hint () {
429 local context="$1" &&
430 local use_branch=$2 &&
431 shift 2 &&
432 local opts="$*" &&
433 test_expect_success "'worktree add' show orphan hint in bad/orphan HEAD w/ $context" '
434 test_when_finished "rm -rf repo" &&
435 git init repo &&
436 (cd repo && test_commit commit) &&
437 git -C repo switch --orphan noref &&
438 test_must_fail git -C repo worktree add $opts foobar/ 2>actual &&
439 ! grep "error: unknown switch" actual &&
440 grep "hint: If you meant to create a worktree containing a new unborn branch" actual &&
441 if [ $use_branch -eq 1 ]
442 then
443 grep -E "^hint: +git worktree add --orphan -b [^ ]+ [^ ]+$" actual
444 else
445 grep -E "^hint: +git worktree add --orphan [^ ]+$" actual
446 fi
447
448 '
449 }
450
451 test_wt_add_orphan_hint 'no opts' 0
452 test_wt_add_orphan_hint '-b' 1 -b foobar_branch
453 test_wt_add_orphan_hint '-B' 1 -B foobar_branch
454
455 test_expect_success "'worktree add' doesn't show orphan hint in bad/orphan HEAD w/ --quiet" '
456 test_when_finished "rm -rf repo" &&
457 git init repo &&
458 (cd repo && test_commit commit) &&
459 test_must_fail git -C repo worktree add --quiet foobar_branch foobar/ 2>actual &&
460 ! grep "error: unknown switch" actual &&
461 ! grep "hint: If you meant to create a worktree containing a new unborn branch" actual
462 '
463
464 test_expect_success 'local clone from linked checkout' '
465 git clone --local here here-clone &&
466 ( cd here-clone && git fsck )
467 '
468
469 test_expect_success 'local clone --shared from linked checkout' '
470 git -C bare worktree add --detach ../baretree &&
471 git clone --local --shared baretree bare-clone &&
472 grep /bare/ bare-clone/.git/objects/info/alternates
473 '
474
475 test_expect_success '"add" worktree with --no-checkout' '
476 git worktree add --no-checkout -b swamp swamp &&
477 ! test -e swamp/init.t &&
478 git -C swamp reset --hard &&
479 test_cmp init.t swamp/init.t
480 '
481
482 test_expect_success '"add" worktree with --checkout' '
483 git worktree add --checkout -b swmap2 swamp2 &&
484 test_cmp init.t swamp2/init.t
485 '
486
487 test_expect_success 'put a worktree under rebase' '
488 git worktree add under-rebase &&
489 (
490 cd under-rebase &&
491 set_fake_editor &&
492 FAKE_LINES="edit 1" git rebase -i HEAD^ &&
493 git worktree list | grep "under-rebase.*detached HEAD"
494 )
495 '
496
497 test_expect_success 'add a worktree, checking out a rebased branch' '
498 test_must_fail git worktree add new-rebase under-rebase &&
499 ! test -d new-rebase
500 '
501
502 test_expect_success 'checking out a rebased branch from another worktree' '
503 git worktree add new-place &&
504 test_must_fail git -C new-place checkout under-rebase
505 '
506
507 test_expect_success 'not allow to delete a branch under rebase' '
508 (
509 cd under-rebase &&
510 test_must_fail git branch -D under-rebase
511 )
512 '
513
514 test_expect_success 'rename a branch under rebase not allowed' '
515 test_must_fail git branch -M under-rebase rebase-with-new-name
516 '
517
518 test_expect_success 'check out from current worktree branch ok' '
519 (
520 cd under-rebase &&
521 git checkout under-rebase &&
522 git checkout - &&
523 git rebase --abort
524 )
525 '
526
527 test_expect_success 'checkout a branch under bisect' '
528 git worktree add under-bisect &&
529 (
530 cd under-bisect &&
531 git bisect start &&
532 git bisect bad &&
533 git bisect good HEAD~2 &&
534 git worktree list | grep "under-bisect.*detached HEAD" &&
535 test_must_fail git worktree add new-bisect under-bisect &&
536 ! test -d new-bisect
537 )
538 '
539
540 test_expect_success 'rename a branch under bisect not allowed' '
541 test_must_fail git branch -M under-bisect bisect-with-new-name
542 '
543 # Is branch "refs/heads/$1" set to pull from "$2/$3"?
544 test_branch_upstream () {
545 printf "%s\n" "$2" "refs/heads/$3" >expect.upstream &&
546 {
547 git config "branch.$1.remote" &&
548 git config "branch.$1.merge"
549 } >actual.upstream &&
550 test_cmp expect.upstream actual.upstream
551 }
552
553 test_expect_success '--track sets up tracking' '
554 test_when_finished rm -rf track &&
555 git worktree add --track -b track track main &&
556 test_branch_upstream track . main
557 '
558
559 # setup remote repository $1 and repository $2 with $1 set up as
560 # remote. The remote has two branches, main and foo.
561 setup_remote_repo () {
562 git init $1 &&
563 (
564 cd $1 &&
565 test_commit $1_main &&
566 git checkout -b foo &&
567 test_commit upstream_foo
568 ) &&
569 git init $2 &&
570 (
571 cd $2 &&
572 test_commit $2_main &&
573 git remote add $1 ../$1 &&
574 git config remote.$1.fetch \
575 "refs/heads/*:refs/remotes/$1/*" &&
576 git fetch --all
577 )
578 }
579
580 test_expect_success '"add" <path> <remote/branch> w/ no HEAD' '
581 test_when_finished rm -rf repo_upstream repo_local foo &&
582 setup_remote_repo repo_upstream repo_local &&
583 git -C repo_local config --bool core.bare true &&
584 git -C repo_local branch -D main &&
585 git -C repo_local worktree add ./foo repo_upstream/foo
586 '
587
588 test_expect_success '--no-track avoids setting up tracking' '
589 test_when_finished rm -rf repo_upstream repo_local foo &&
590 setup_remote_repo repo_upstream repo_local &&
591 (
592 cd repo_local &&
593 git worktree add --no-track -b foo ../foo repo_upstream/foo
594 ) &&
595 (
596 cd foo &&
597 test_must_fail git config "branch.foo.remote" &&
598 test_must_fail git config "branch.foo.merge" &&
599 test_cmp_rev refs/remotes/repo_upstream/foo refs/heads/foo
600 )
601 '
602
603 test_expect_success '"add" <path> <non-existent-branch> fails' '
604 test_must_fail git worktree add foo non-existent
605 '
606
607 test_expect_success '"add" <path> <branch> dwims' '
608 test_when_finished rm -rf repo_upstream repo_dwim foo &&
609 setup_remote_repo repo_upstream repo_dwim &&
610 git init repo_dwim &&
611 (
612 cd repo_dwim &&
613 git worktree add ../foo foo
614 ) &&
615 (
616 cd foo &&
617 test_branch_upstream foo repo_upstream foo &&
618 test_cmp_rev refs/remotes/repo_upstream/foo refs/heads/foo
619 )
620 '
621
622 test_expect_success '"add" <path> <branch> dwims with checkout.defaultRemote' '
623 test_when_finished rm -rf repo_upstream repo_dwim foo &&
624 setup_remote_repo repo_upstream repo_dwim &&
625 git init repo_dwim &&
626 (
627 cd repo_dwim &&
628 git remote add repo_upstream2 ../repo_upstream &&
629 git fetch repo_upstream2 &&
630 test_must_fail git worktree add ../foo foo &&
631 git -c checkout.defaultRemote=repo_upstream worktree add ../foo foo &&
632 git status -uno --porcelain >status.actual &&
633 test_must_be_empty status.actual
634 ) &&
635 (
636 cd foo &&
637 test_branch_upstream foo repo_upstream foo &&
638 test_cmp_rev refs/remotes/repo_upstream/foo refs/heads/foo
639 )
640 '
641
642 test_expect_success 'git worktree add does not match remote' '
643 test_when_finished rm -rf repo_a repo_b foo &&
644 setup_remote_repo repo_a repo_b &&
645 (
646 cd repo_b &&
647 git worktree add ../foo
648 ) &&
649 (
650 cd foo &&
651 test_must_fail git config "branch.foo.remote" &&
652 test_must_fail git config "branch.foo.merge" &&
653 test_cmp_rev ! refs/remotes/repo_a/foo refs/heads/foo
654 )
655 '
656
657 test_expect_success 'git worktree add --guess-remote sets up tracking' '
658 test_when_finished rm -rf repo_a repo_b foo &&
659 setup_remote_repo repo_a repo_b &&
660 (
661 cd repo_b &&
662 git worktree add --guess-remote ../foo
663 ) &&
664 (
665 cd foo &&
666 test_branch_upstream foo repo_a foo &&
667 test_cmp_rev refs/remotes/repo_a/foo refs/heads/foo
668 )
669 '
670 test_expect_success 'git worktree add --guess-remote sets up tracking (quiet)' '
671 test_when_finished rm -rf repo_a repo_b foo &&
672 setup_remote_repo repo_a repo_b &&
673 (
674 cd repo_b &&
675 git worktree add --quiet --guess-remote ../foo 2>actual &&
676 test_must_be_empty actual
677 ) &&
678 (
679 cd foo &&
680 test_branch_upstream foo repo_a foo &&
681 test_cmp_rev refs/remotes/repo_a/foo refs/heads/foo
682 )
683 '
684
685 test_expect_success 'git worktree --no-guess-remote (quiet)' '
686 test_when_finished rm -rf repo_a repo_b foo &&
687 setup_remote_repo repo_a repo_b &&
688 (
689 cd repo_b &&
690 git worktree add --quiet --no-guess-remote ../foo
691 ) &&
692 (
693 cd foo &&
694 test_must_fail git config "branch.foo.remote" &&
695 test_must_fail git config "branch.foo.merge" &&
696 test_cmp_rev ! refs/remotes/repo_a/foo refs/heads/foo
697 )
698 '
699
700 test_expect_success 'git worktree add with worktree.guessRemote sets up tracking' '
701 test_when_finished rm -rf repo_a repo_b foo &&
702 setup_remote_repo repo_a repo_b &&
703 (
704 cd repo_b &&
705 git config worktree.guessRemote true &&
706 git worktree add ../foo
707 ) &&
708 (
709 cd foo &&
710 test_branch_upstream foo repo_a foo &&
711 test_cmp_rev refs/remotes/repo_a/foo refs/heads/foo
712 )
713 '
714
715 test_expect_success 'git worktree --no-guess-remote option overrides config' '
716 test_when_finished rm -rf repo_a repo_b foo &&
717 setup_remote_repo repo_a repo_b &&
718 (
719 cd repo_b &&
720 git config worktree.guessRemote true &&
721 git worktree add --no-guess-remote ../foo
722 ) &&
723 (
724 cd foo &&
725 test_must_fail git config "branch.foo.remote" &&
726 test_must_fail git config "branch.foo.merge" &&
727 test_cmp_rev ! refs/remotes/repo_a/foo refs/heads/foo
728 )
729 '
730
731 test_dwim_orphan () {
732 local info_text="No possible source branch, inferring '--orphan'" &&
733 local fetch_error_text="fatal: No local or remote refs exist despite at least one remote" &&
734 local orphan_hint="hint: If you meant to create a worktree containing a new unborn branch" &&
735 local invalid_ref_regex="^fatal: invalid reference: " &&
736 local bad_combo_regex="^fatal: options '[-a-z]*' and '[-a-z]*' cannot be used together" &&
737
738 local git_ns="repo" &&
739 local dashc_args="-C $git_ns" &&
740 local use_cd=0 &&
741
742 local bad_head=0 &&
743 local empty_repo=1 &&
744 local local_ref=0 &&
745 local use_quiet=0 &&
746 local remote=0 &&
747 local remote_ref=0 &&
748 local use_detach=0 &&
749 local use_new_branch=0 &&
750
751 local outcome="$1" &&
752 local outcome_text &&
753 local success &&
754 shift &&
755 local args="" &&
756 local context="" &&
757 case "$outcome" in
758 "infer")
759 success=1 &&
760 outcome_text='"add" DWIM infer --orphan'
761 ;;
762 "no_infer")
763 success=1 &&
764 outcome_text='"add" DWIM doesnt infer --orphan'
765 ;;
766 "fetch_error")
767 success=0 &&
768 outcome_text='"add" error need fetch'
769 ;;
770 "fatal_orphan_bad_combo")
771 success=0 &&
772 outcome_text='"add" error inferred "--orphan" gives illegal opts combo'
773 ;;
774 "warn_bad_head")
775 success=0 &&
776 outcome_text='"add" error, warn on bad HEAD, hint use orphan'
777 ;;
778 *)
779 echo "test_dwim_orphan(): invalid outcome: '$outcome'" >&2 &&
780 return 1
781 ;;
782 esac &&
783 while [ $# -gt 0 ]
784 do
785 case "$1" in
786 # How and from where to create the worktree
787 "-C_repo")
788 use_cd=0 &&
789 git_ns="repo" &&
790 dashc_args="-C $git_ns" &&
791 context="$context, 'git -C repo'"
792 ;;
793 "-C_wt")
794 use_cd=0 &&
795 git_ns="wt" &&
796 dashc_args="-C $git_ns" &&
797 context="$context, 'git -C wt'"
798 ;;
799 "cd_repo")
800 use_cd=1 &&
801 git_ns="repo" &&
802 dashc_args="" &&
803 context="$context, 'cd repo && git'"
804 ;;
805 "cd_wt")
806 use_cd=1 &&
807 git_ns="wt" &&
808 dashc_args="" &&
809 context="$context, 'cd wt && git'"
810 ;;
811
812 # Bypass the "pull first" warning
813 "force")
814 args="$args --force" &&
815 context="$context, --force"
816 ;;
817
818 # Try to use remote refs when DWIM
819 "guess_remote")
820 args="$args --guess-remote" &&
821 context="$context, --guess-remote"
822 ;;
823 "no_guess_remote")
824 args="$args --no-guess-remote" &&
825 context="$context, --no-guess-remote"
826 ;;
827
828 # Whether there is at least one local branch present
829 "local_ref")
830 empty_repo=0 &&
831 local_ref=1 &&
832 context="$context, >=1 local branches"
833 ;;
834 "no_local_ref")
835 empty_repo=0 &&
836 context="$context, 0 local branches"
837 ;;
838
839 # Whether the HEAD points at a valid ref (skip this opt when no refs)
840 "good_head")
841 # requires: local_ref
842 context="$context, valid HEAD"
843 ;;
844 "bad_head")
845 bad_head=1 &&
846 context="$context, invalid (or orphan) HEAD"
847 ;;
848
849 # Whether the code path is tested with the base add command, -b, or --detach
850 "no_-b")
851 use_new_branch=0 &&
852 context="$context, no --branch"
853 ;;
854 "-b")
855 use_new_branch=1 &&
856 context="$context, --branch"
857 ;;
858 "detach")
859 use_detach=1 &&
860 context="$context, --detach"
861 ;;
862
863 # Whether to check that all output is suppressed (except errors)
864 # or that the output is as expected
865 "quiet")
866 use_quiet=1 &&
867 args="$args --quiet" &&
868 context="$context, --quiet"
869 ;;
870 "no_quiet")
871 use_quiet=0 &&
872 context="$context, no --quiet (expect output)"
873 ;;
874
875 # Whether there is at least one remote attached to the repo
876 "remote")
877 empty_repo=0 &&
878 remote=1 &&
879 context="$context, >=1 remotes"
880 ;;
881 "no_remote")
882 empty_repo=0 &&
883 remote=0 &&
884 context="$context, 0 remotes"
885 ;;
886
887 # Whether there is at least one valid remote ref
888 "remote_ref")
889 # requires: remote
890 empty_repo=0 &&
891 remote_ref=1 &&
892 context="$context, >=1 fetched remote branches"
893 ;;
894 "no_remote_ref")
895 empty_repo=0 &&
896 remote_ref=0 &&
897 context="$context, 0 fetched remote branches"
898 ;;
899
900 # Options or flags that become illegal when --orphan is inferred
901 "no_checkout")
902 args="$args --no-checkout" &&
903 context="$context, --no-checkout"
904 ;;
905 "track")
906 args="$args --track" &&
907 context="$context, --track"
908 ;;
909
910 # All other options are illegal
911 *)
912 echo "test_dwim_orphan(): invalid arg: '$1'" >&2 &&
913 return 1
914 ;;
915 esac &&
916 shift
917 done &&
918 context="${context#', '}" &&
919 if [ $use_new_branch -eq 1 ]
920 then
921 args="$args -b foo"
922 elif [ $use_detach -eq 1 ]
923 then
924 args="$args --detach"
925 else
926 context="DWIM (no --branch), $context"
927 fi &&
928 if [ $empty_repo -eq 1 ]
929 then
930 context="empty repo, $context"
931 fi &&
932 args="$args ../foo" &&
933 context="${context%', '}" &&
934 test_expect_success "$outcome_text w/ $context" '
935 test_when_finished "rm -rf repo" &&
936 git init repo &&
937 if [ $local_ref -eq 1 ] && [ "$git_ns" = "repo" ]
938 then
939 (cd repo && test_commit commit) &&
940 if [ $bad_head -eq 1 ]
941 then
942 git -C repo symbolic-ref HEAD refs/heads/badbranch
943 fi
944 elif [ $local_ref -eq 1 ] && [ "$git_ns" = "wt" ]
945 then
946 test_when_finished "git -C repo worktree remove -f ../wt" &&
947 git -C repo worktree add --orphan -b main ../wt &&
948 (cd wt && test_commit commit) &&
949 if [ $bad_head -eq 1 ]
950 then
951 git -C wt symbolic-ref HEAD refs/heads/badbranch
952 fi
953 elif [ $local_ref -eq 0 ] && [ "$git_ns" = "wt" ]
954 then
955 test_when_finished "git -C repo worktree remove -f ../wt" &&
956 git -C repo worktree add --orphan -b orphanbranch ../wt
957 fi &&
958
959 if [ $remote -eq 1 ]
960 then
961 test_when_finished "rm -rf upstream" &&
962 git init upstream &&
963 (cd upstream && test_commit commit) &&
964 git -C upstream switch -c foo &&
965 git -C repo remote add upstream ../upstream
966 fi &&
967
968 if [ $remote_ref -eq 1 ]
969 then
970 git -C repo fetch
971 fi &&
972 if [ $success -eq 1 ]
973 then
974 test_when_finished git -C repo worktree remove ../foo
975 fi &&
976 (
977 if [ $use_cd -eq 1 ]
978 then
979 cd $git_ns
980 fi &&
981 if [ "$outcome" = "infer" ]
982 then
983 git $dashc_args worktree add $args 2>actual &&
984 if [ $use_quiet -eq 1 ]
985 then
986 test_must_be_empty actual
987 else
988 grep "$info_text" actual
989 fi
990 elif [ "$outcome" = "no_infer" ]
991 then
992 git $dashc_args worktree add $args 2>actual &&
993 if [ $use_quiet -eq 1 ]
994 then
995 test_must_be_empty actual
996 else
997 ! grep "$info_text" actual
998 fi
999 elif [ "$outcome" = "fetch_error" ]
1000 then
1001 test_must_fail git $dashc_args worktree add $args 2>actual &&
1002 grep "$fetch_error_text" actual
1003 elif [ "$outcome" = "fatal_orphan_bad_combo" ]
1004 then
1005 test_must_fail git $dashc_args worktree add $args 2>actual &&
1006 if [ $use_quiet -eq 1 ]
1007 then
1008 ! grep "$info_text" actual
1009 else
1010 grep "$info_text" actual
1011 fi &&
1012 grep "$bad_combo_regex" actual
1013 elif [ "$outcome" = "warn_bad_head" ]
1014 then
1015 test_must_fail git $dashc_args worktree add $args 2>actual &&
1016 if [ $use_quiet -eq 1 ]
1017 then
1018 grep "$invalid_ref_regex" actual &&
1019 ! grep "$orphan_hint" actual
1020 else
1021 headpath=$(git $dashc_args rev-parse --path-format=absolute --git-path HEAD) &&
1022 headcontents=$(cat "$headpath") &&
1023 grep "HEAD points to an invalid (or orphaned) reference" actual &&
1024 grep "HEAD path: .$headpath." actual &&
1025 grep "HEAD contents: .$headcontents." actual &&
1026 grep "$orphan_hint" actual &&
1027 ! grep "$info_text" actual
1028 fi &&
1029 grep "$invalid_ref_regex" actual
1030 else
1031 # Unreachable
1032 false
1033 fi
1034 ) &&
1035 if [ $success -ne 1 ]
1036 then
1037 test_path_is_missing foo
1038 fi
1039 '
1040 }
1041
1042 for quiet_mode in "no_quiet" "quiet"
1043 do
1044 for changedir_type in "cd_repo" "cd_wt" "-C_repo" "-C_wt"
1045 do
1046 dwim_test_args="$quiet_mode $changedir_type"
1047 test_dwim_orphan 'infer' $dwim_test_args no_-b
1048 test_dwim_orphan 'no_infer' $dwim_test_args no_-b local_ref good_head
1049 test_dwim_orphan 'infer' $dwim_test_args no_-b no_local_ref no_remote no_remote_ref no_guess_remote
1050 test_dwim_orphan 'infer' $dwim_test_args no_-b no_local_ref remote no_remote_ref no_guess_remote
1051 test_dwim_orphan 'fetch_error' $dwim_test_args no_-b no_local_ref remote no_remote_ref guess_remote
1052 test_dwim_orphan 'infer' $dwim_test_args no_-b no_local_ref remote no_remote_ref guess_remote force
1053 test_dwim_orphan 'no_infer' $dwim_test_args no_-b no_local_ref remote remote_ref guess_remote
1054
1055 test_dwim_orphan 'infer' $dwim_test_args -b
1056 test_dwim_orphan 'no_infer' $dwim_test_args -b local_ref good_head
1057 test_dwim_orphan 'infer' $dwim_test_args -b no_local_ref no_remote no_remote_ref no_guess_remote
1058 test_dwim_orphan 'infer' $dwim_test_args -b no_local_ref remote no_remote_ref no_guess_remote
1059 test_dwim_orphan 'infer' $dwim_test_args -b no_local_ref remote no_remote_ref guess_remote
1060 test_dwim_orphan 'infer' $dwim_test_args -b no_local_ref remote remote_ref guess_remote
1061
1062 test_dwim_orphan 'warn_bad_head' $dwim_test_args no_-b local_ref bad_head
1063 test_dwim_orphan 'warn_bad_head' $dwim_test_args -b local_ref bad_head
1064 test_dwim_orphan 'warn_bad_head' $dwim_test_args detach local_ref bad_head
1065 done
1066
1067 test_dwim_orphan 'fatal_orphan_bad_combo' $quiet_mode no_-b no_checkout
1068 test_dwim_orphan 'fatal_orphan_bad_combo' $quiet_mode no_-b track
1069 test_dwim_orphan 'fatal_orphan_bad_combo' $quiet_mode -b no_checkout
1070 test_dwim_orphan 'fatal_orphan_bad_combo' $quiet_mode -b track
1071 done
1072
1073 post_checkout_hook () {
1074 test_when_finished "rm -rf .git/hooks" &&
1075 mkdir .git/hooks &&
1076 test_hook -C "$1" post-checkout <<-\EOF
1077 {
1078 echo $*
1079 git rev-parse --git-dir --show-toplevel
1080 } >hook.actual
1081 EOF
1082 }
1083
1084 test_expect_success '"add" invokes post-checkout hook (branch)' '
1085 post_checkout_hook &&
1086 {
1087 echo $ZERO_OID $(git rev-parse HEAD) 1 &&
1088 echo $(pwd)/.git/worktrees/gumby &&
1089 echo $(pwd)/gumby
1090 } >hook.expect &&
1091 git worktree add gumby &&
1092 test_cmp hook.expect gumby/hook.actual
1093 '
1094
1095 test_expect_success '"add" invokes post-checkout hook (detached)' '
1096 post_checkout_hook &&
1097 {
1098 echo $ZERO_OID $(git rev-parse HEAD) 1 &&
1099 echo $(pwd)/.git/worktrees/grumpy &&
1100 echo $(pwd)/grumpy
1101 } >hook.expect &&
1102 git worktree add --detach grumpy &&
1103 test_cmp hook.expect grumpy/hook.actual
1104 '
1105
1106 test_expect_success '"add --no-checkout" suppresses post-checkout hook' '
1107 post_checkout_hook &&
1108 rm -f hook.actual &&
1109 git worktree add --no-checkout gloopy &&
1110 test_path_is_missing gloopy/hook.actual
1111 '
1112
1113 test_expect_success '"add" in other worktree invokes post-checkout hook' '
1114 post_checkout_hook &&
1115 {
1116 echo $ZERO_OID $(git rev-parse HEAD) 1 &&
1117 echo $(pwd)/.git/worktrees/guppy &&
1118 echo $(pwd)/guppy
1119 } >hook.expect &&
1120 git -C gloopy worktree add --detach ../guppy &&
1121 test_cmp hook.expect guppy/hook.actual
1122 '
1123
1124 test_expect_success '"add" in bare repo invokes post-checkout hook' '
1125 rm -rf bare &&
1126 git clone --bare . bare &&
1127 {
1128 echo $ZERO_OID $(git --git-dir=bare rev-parse HEAD) 1 &&
1129 echo $(pwd)/bare/worktrees/goozy &&
1130 echo $(pwd)/goozy
1131 } >hook.expect &&
1132 post_checkout_hook bare &&
1133 git -C bare worktree add --detach ../goozy &&
1134 test_cmp hook.expect goozy/hook.actual
1135 '
1136
1137 test_expect_success '"add" an existing but missing worktree' '
1138 git worktree add --detach pneu &&
1139 test_must_fail git worktree add --detach pneu &&
1140 rm -fr pneu &&
1141 test_must_fail git worktree add --detach pneu &&
1142 git worktree add --force --detach pneu
1143 '
1144
1145 test_expect_success '"add" an existing locked but missing worktree' '
1146 git worktree add --detach gnoo &&
1147 git worktree lock gnoo &&
1148 test_when_finished "git worktree unlock gnoo || :" &&
1149 rm -fr gnoo &&
1150 test_must_fail git worktree add --detach gnoo &&
1151 test_must_fail git worktree add --force --detach gnoo &&
1152 git worktree add --force --force --detach gnoo
1153 '
1154
1155 test_expect_success '"add" not tripped up by magic worktree matching"' '
1156 # if worktree "sub1/bar" exists, "git worktree add bar" in distinct
1157 # directory `sub2` should not mistakenly complain that `bar` is an
1158 # already-registered worktree
1159 mkdir sub1 sub2 &&
1160 git -C sub1 --git-dir=../.git worktree add --detach bozo &&
1161 git -C sub2 --git-dir=../.git worktree add --detach bozo
1162 '
1163
1164 test_expect_success FUNNYNAMES 'sanitize generated worktree name' '
1165 git worktree add --detach ". weird*..?.lock.lock" &&
1166 test -d .git/worktrees/---weird-.-
1167 '
1168
1169 test_expect_success '"add" should not fail because of another bad worktree' '
1170 git init add-fail &&
1171 (
1172 cd add-fail &&
1173 test_commit first &&
1174 mkdir sub &&
1175 git worktree add sub/to-be-deleted &&
1176 rm -rf sub &&
1177 git worktree add second
1178 )
1179 '
1180
1181 test_expect_success '"add" with uninitialized submodule, with submodule.recurse unset' '
1182 test_config_global protocol.file.allow always &&
1183 test_create_repo submodule &&
1184 test_commit -C submodule first &&
1185 test_create_repo project &&
1186 git -C project submodule add ../submodule &&
1187 git -C project add submodule &&
1188 test_tick &&
1189 git -C project commit -m add_sub &&
1190 git clone project project-clone &&
1191 git -C project-clone worktree add ../project-2
1192 '
1193 test_expect_success '"add" with uninitialized submodule, with submodule.recurse set' '
1194 git -C project-clone -c submodule.recurse worktree add ../project-3
1195 '
1196
1197 test_expect_success '"add" with initialized submodule, with submodule.recurse unset' '
1198 test_config_global protocol.file.allow always &&
1199 git -C project-clone submodule update --init &&
1200 git -C project-clone worktree add ../project-4
1201 '
1202
1203 test_expect_success '"add" with initialized submodule, with submodule.recurse set' '
1204 git -C project-clone -c submodule.recurse worktree add ../project-5
1205 '
1206
1207 test_done