diff -c/--cc: do not mistake "resolved as deletion" as "use working tree"
[git] / t / t4020-diff-external.sh
1 #!/bin/sh
2
3 test_description='external diff interface test'
4
5 . ./test-lib.sh
6
7 _z40=0000000000000000000000000000000000000000
8
9 test_expect_success setup '
10
11         test_tick &&
12         echo initial >file &&
13         git add file &&
14         git commit -m initial &&
15
16         test_tick &&
17         echo second >file &&
18         git add file &&
19         git commit -m second &&
20
21         test_tick &&
22         echo third >file
23 '
24
25 test_expect_success 'GIT_EXTERNAL_DIFF environment' '
26
27         GIT_EXTERNAL_DIFF=echo git diff | {
28                 read path oldfile oldhex oldmode newfile newhex newmode &&
29                 test "z$path" = zfile &&
30                 test "z$oldmode" = z100644 &&
31                 test "z$newhex" = "z$_z40" &&
32                 test "z$newmode" = z100644 &&
33                 oh=$(git rev-parse --verify HEAD:file) &&
34                 test "z$oh" = "z$oldhex"
35         }
36
37 '
38
39 test_expect_success 'GIT_EXTERNAL_DIFF environment should apply only to diff' '
40
41         GIT_EXTERNAL_DIFF=echo git log -p -1 HEAD |
42         grep "^diff --git a/file b/file"
43
44 '
45
46 test_expect_success 'GIT_EXTERNAL_DIFF environment and --no-ext-diff' '
47
48         GIT_EXTERNAL_DIFF=echo git diff --no-ext-diff |
49         grep "^diff --git a/file b/file"
50
51 '
52
53 test_expect_success 'diff attribute' '
54
55         git config diff.parrot.command echo &&
56
57         echo >.gitattributes "file diff=parrot" &&
58
59         git diff | {
60                 read path oldfile oldhex oldmode newfile newhex newmode &&
61                 test "z$path" = zfile &&
62                 test "z$oldmode" = z100644 &&
63                 test "z$newhex" = "z$_z40" &&
64                 test "z$newmode" = z100644 &&
65                 oh=$(git rev-parse --verify HEAD:file) &&
66                 test "z$oh" = "z$oldhex"
67         }
68
69 '
70
71 test_expect_success 'diff attribute should apply only to diff' '
72
73         git log -p -1 HEAD |
74         grep "^diff --git a/file b/file"
75
76 '
77
78 test_expect_success 'diff attribute and --no-ext-diff' '
79
80         git diff --no-ext-diff |
81         grep "^diff --git a/file b/file"
82
83 '
84
85 test_expect_success 'diff attribute' '
86
87         git config --unset diff.parrot.command &&
88         git config diff.color.command echo &&
89
90         echo >.gitattributes "file diff=color" &&
91
92         git diff | {
93                 read path oldfile oldhex oldmode newfile newhex newmode &&
94                 test "z$path" = zfile &&
95                 test "z$oldmode" = z100644 &&
96                 test "z$newhex" = "z$_z40" &&
97                 test "z$newmode" = z100644 &&
98                 oh=$(git rev-parse --verify HEAD:file) &&
99                 test "z$oh" = "z$oldhex"
100         }
101
102 '
103
104 test_expect_success 'diff attribute should apply only to diff' '
105
106         git log -p -1 HEAD |
107         grep "^diff --git a/file b/file"
108
109 '
110
111 test_expect_success 'diff attribute and --no-ext-diff' '
112
113         git diff --no-ext-diff |
114         grep "^diff --git a/file b/file"
115
116 '
117
118 test_expect_success 'no diff with -diff' '
119         echo >.gitattributes "file -diff" &&
120         git diff | grep Binary
121 '
122
123 echo NULZbetweenZwords | perl -pe 'y/Z/\000/' > file
124
125 test_expect_success 'force diff with "diff"' '
126         echo >.gitattributes "file diff" &&
127         git diff >actual &&
128         test_cmp "$TEST_DIRECTORY"/t4020/diff.NUL actual
129 '
130
131 test_expect_success 'GIT_EXTERNAL_DIFF with more than one changed files' '
132         echo anotherfile > file2 &&
133         git add file2 &&
134         git commit -m "added 2nd file" &&
135         echo modified >file2 &&
136         GIT_EXTERNAL_DIFF=echo git diff
137 '
138
139 test_expect_success 'GIT_EXTERNAL_DIFF generates pretty paths' '
140         touch file.ext &&
141         git add file.ext &&
142         echo with extension > file.ext &&
143         GIT_EXTERNAL_DIFF=echo git diff file.ext | grep ......_file\.ext &&
144         git update-index --force-remove file.ext &&
145         rm file.ext
146 '
147
148 echo "#!$SHELL_PATH" >fake-diff.sh
149 cat >> fake-diff.sh <<\EOF
150 cat $2 >> crlfed.txt
151 EOF
152 chmod a+x fake-diff.sh
153
154 keep_only_cr () {
155         tr -dc '\015'
156 }
157
158 test_expect_success 'external diff with autocrlf = true' '
159         git config core.autocrlf true &&
160         GIT_EXTERNAL_DIFF=./fake-diff.sh git diff &&
161         test $(wc -l < crlfed.txt) = $(cat crlfed.txt | keep_only_cr | wc -c)
162 '
163
164 test_expect_success 'diff --cached' '
165         git add file &&
166         git update-index --assume-unchanged file &&
167         echo second >file &&
168         git diff --cached >actual &&
169         test_cmp "$TEST_DIRECTORY"/t4020/diff.NUL actual
170 '
171
172 test_done