6d1d59c9b1af8c1ca7d6e78c8a589395ff971338
[git/git.git] / t / t5545-push-options.sh
1 #!/bin/sh
2
3 test_description='pushing to a repository using push options'
4
5 . ./test-lib.sh
6
7 mk_repo_pair () {
8 rm -rf workbench upstream &&
9 test_create_repo upstream &&
10 test_create_repo workbench &&
11 (
12 cd upstream &&
13 git config receive.denyCurrentBranch warn &&
14 mkdir -p .git/hooks &&
15 cat >.git/hooks/pre-receive <<-'EOF' &&
16 #!/bin/sh
17 if test -n "$GIT_PUSH_OPTION_COUNT"; then
18 i=0
19 >hooks/pre-receive.push_options
20 while test "$i" -lt "$GIT_PUSH_OPTION_COUNT"; do
21 eval "value=\$GIT_PUSH_OPTION_$i"
22 echo $value >>hooks/pre-receive.push_options
23 i=$((i + 1))
24 done
25 fi
26 EOF
27 chmod u+x .git/hooks/pre-receive
28
29 cat >.git/hooks/post-receive <<-'EOF' &&
30 #!/bin/sh
31 if test -n "$GIT_PUSH_OPTION_COUNT"; then
32 i=0
33 >hooks/post-receive.push_options
34 while test "$i" -lt "$GIT_PUSH_OPTION_COUNT"; do
35 eval "value=\$GIT_PUSH_OPTION_$i"
36 echo $value >>hooks/post-receive.push_options
37 i=$((i + 1))
38 done
39 fi
40 EOF
41 chmod u+x .git/hooks/post-receive
42 ) &&
43 (
44 cd workbench &&
45 git remote add up ../upstream
46 )
47 }
48
49 # Compare the ref ($1) in upstream with a ref value from workbench ($2)
50 # i.e. test_refs second HEAD@{2}
51 test_refs () {
52 test $# = 2 &&
53 git -C upstream rev-parse --verify "$1" >expect &&
54 git -C workbench rev-parse --verify "$2" >actual &&
55 test_cmp expect actual
56 }
57
58 test_expect_success 'one push option works for a single branch' '
59 mk_repo_pair &&
60 git -C upstream config receive.advertisePushOptions true &&
61 (
62 cd workbench &&
63 test_commit one &&
64 git push --mirror up &&
65 test_commit two &&
66 git push --push-option=asdf up master
67 ) &&
68 test_refs master master &&
69 echo "asdf" >expect &&
70 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
71 test_cmp expect upstream/.git/hooks/post-receive.push_options
72 '
73
74 test_expect_success 'push option denied by remote' '
75 mk_repo_pair &&
76 git -C upstream config receive.advertisePushOptions false &&
77 (
78 cd workbench &&
79 test_commit one &&
80 git push --mirror up &&
81 test_commit two &&
82 test_must_fail git push --push-option=asdf up master
83 ) &&
84 test_refs master HEAD@{1}
85 '
86
87 test_expect_success 'two push options work' '
88 mk_repo_pair &&
89 git -C upstream config receive.advertisePushOptions true &&
90 (
91 cd workbench &&
92 test_commit one &&
93 git push --mirror up &&
94 test_commit two &&
95 git push --push-option=asdf --push-option="more structured text" up master
96 ) &&
97 test_refs master master &&
98 printf "asdf\nmore structured text\n" >expect &&
99 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
100 test_cmp expect upstream/.git/hooks/post-receive.push_options
101 '
102
103 test_expect_success 'push options and submodules' '
104 test_when_finished "rm -rf parent" &&
105 test_when_finished "rm -rf parent_upstream" &&
106 mk_repo_pair &&
107 git -C upstream config receive.advertisePushOptions true &&
108 cp -r upstream parent_upstream &&
109 test_commit -C upstream one &&
110
111 test_create_repo parent &&
112 git -C parent remote add up ../parent_upstream &&
113 test_commit -C parent one &&
114 git -C parent push --mirror up &&
115
116 git -C parent submodule add ../upstream workbench &&
117 git -C parent/workbench remote add up ../../upstream &&
118 git -C parent commit -m "add submoule" &&
119
120 test_commit -C parent/workbench two &&
121 git -C parent add workbench &&
122 git -C parent commit -m "update workbench" &&
123
124 git -C parent push \
125 --push-option=asdf --push-option="more structured text" \
126 --recurse-submodules=on-demand up master &&
127
128 git -C upstream rev-parse --verify master >expect &&
129 git -C parent/workbench rev-parse --verify master >actual &&
130 test_cmp expect actual &&
131
132 git -C parent_upstream rev-parse --verify master >expect &&
133 git -C parent rev-parse --verify master >actual &&
134 test_cmp expect actual &&
135
136 printf "asdf\nmore structured text\n" >expect &&
137 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
138 test_cmp expect upstream/.git/hooks/post-receive.push_options &&
139 test_cmp expect parent_upstream/.git/hooks/pre-receive.push_options &&
140 test_cmp expect parent_upstream/.git/hooks/post-receive.push_options
141 '
142
143 test_expect_success 'default push option' '
144 mk_repo_pair &&
145 git -C upstream config receive.advertisePushOptions true &&
146 (
147 cd workbench &&
148 test_commit one &&
149 git push --mirror up &&
150 test_commit two &&
151 git -c push.pushOption=default push up master
152 ) &&
153 test_refs master master &&
154 echo "default" >expect &&
155 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
156 test_cmp expect upstream/.git/hooks/post-receive.push_options
157 '
158
159 test_expect_success 'two default push options' '
160 mk_repo_pair &&
161 git -C upstream config receive.advertisePushOptions true &&
162 (
163 cd workbench &&
164 test_commit one &&
165 git push --mirror up &&
166 test_commit two &&
167 git -c push.pushOption=default1 -c push.pushOption=default2 push up master
168 ) &&
169 test_refs master master &&
170 printf "default1\ndefault2\n" >expect &&
171 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
172 test_cmp expect upstream/.git/hooks/post-receive.push_options
173 '
174
175 test_expect_success 'push option from command line overrides from-config push option' '
176 mk_repo_pair &&
177 git -C upstream config receive.advertisePushOptions true &&
178 (
179 cd workbench &&
180 test_commit one &&
181 git push --mirror up &&
182 test_commit two &&
183 git -c push.pushOption=default push --push-option=manual up master
184 ) &&
185 test_refs master master &&
186 echo "manual" >expect &&
187 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
188 test_cmp expect upstream/.git/hooks/post-receive.push_options
189 '
190
191 test_expect_success 'empty value of push.pushOption in config clears the list' '
192 mk_repo_pair &&
193 git -C upstream config receive.advertisePushOptions true &&
194 (
195 cd workbench &&
196 test_commit one &&
197 git push --mirror up &&
198 test_commit two &&
199 git -c push.pushOption=default1 -c push.pushOption= -c push.pushOption=default2 push up master
200 ) &&
201 test_refs master master &&
202 echo "default2" >expect &&
203 test_cmp expect upstream/.git/hooks/pre-receive.push_options &&
204 test_cmp expect upstream/.git/hooks/post-receive.push_options
205 '
206
207 test_expect_success 'invalid push option in config' '
208 mk_repo_pair &&
209 git -C upstream config receive.advertisePushOptions true &&
210 (
211 cd workbench &&
212 test_commit one &&
213 git push --mirror up &&
214 test_commit two &&
215 test_must_fail git -c push.pushOption push up master
216 ) &&
217 test_refs master HEAD@{1}
218 '
219
220 test_expect_success 'push options keep quoted characters intact (direct)' '
221 mk_repo_pair &&
222 git -C upstream config receive.advertisePushOptions true &&
223 test_commit -C workbench one &&
224 git -C workbench push --push-option="\"embedded quotes\"" up master &&
225 echo "\"embedded quotes\"" >expect &&
226 test_cmp expect upstream/.git/hooks/pre-receive.push_options
227 '
228
229 . "$TEST_DIRECTORY"/lib-httpd.sh
230 start_httpd
231
232 # set up http repository for fetching/pushing, with push options config
233 # bool set to $1
234 mk_http_pair () {
235 test_when_finished "rm -rf test_http_clone" &&
236 test_when_finished 'rm -rf "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git' &&
237 mk_repo_pair &&
238 git -C upstream config receive.advertisePushOptions "$1" &&
239 git -C upstream config http.receivepack true &&
240 cp -R upstream/.git "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git &&
241 git clone "$HTTPD_URL"/smart/upstream test_http_clone
242 }
243
244 test_expect_success 'push option denied properly by http server' '
245 mk_http_pair false &&
246 test_commit -C test_http_clone one &&
247 test_must_fail git -C test_http_clone push --push-option=asdf origin master 2>actual &&
248 test_i18ngrep "the receiving end does not support push options" actual &&
249 git -C test_http_clone push origin master
250 '
251
252 test_expect_success 'push options work properly across http' '
253 mk_http_pair true &&
254
255 test_commit -C test_http_clone one &&
256 git -C test_http_clone push origin master &&
257 git -C "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git rev-parse --verify master >expect &&
258 git -C test_http_clone rev-parse --verify master >actual &&
259 test_cmp expect actual &&
260
261 test_commit -C test_http_clone two &&
262 git -C test_http_clone push --push-option=asdf --push-option="more structured text" origin master &&
263 printf "asdf\nmore structured text\n" >expect &&
264 test_cmp expect "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git/hooks/pre-receive.push_options &&
265 test_cmp expect "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git/hooks/post-receive.push_options &&
266
267 git -C "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git rev-parse --verify master >expect &&
268 git -C test_http_clone rev-parse --verify master >actual &&
269 test_cmp expect actual
270 '
271
272 test_expect_success 'push options keep quoted characters intact (http)' '
273 mk_http_pair true &&
274
275 test_commit -C test_http_clone one &&
276 git -C test_http_clone push --push-option="\"embedded quotes\"" origin master &&
277 echo "\"embedded quotes\"" >expect &&
278 test_cmp expect "$HTTPD_DOCUMENT_ROOT_PATH"/upstream.git/hooks/pre-receive.push_options
279 '
280
281 test_done