]>
Commit | Line | Data |
---|---|---|
c2e86add | 1 | #include "builtin.h" |
2a9c3fe8 | 2 | #include "commit.h" |
584c6cc9 | 3 | #include "refs.h" |
f3a3214e | 4 | #include "pkt-line.h" |
de1a2fdd | 5 | #include "sideband.h" |
38b1c662 | 6 | #include "run-command.h" |
6b62816c | 7 | #include "remote.h" |
96249c04 | 8 | #include "send-pack.h" |
de1a2fdd | 9 | #include "quote.h" |
f1863d0d | 10 | #include "transport.h" |
ff5effdf | 11 | #include "version.h" |
61221472 | 12 | |
2a245013 | 13 | static const char send_pack_usage[] = |
1b1dd23f | 14 | "git send-pack [--all | --mirror] [--dry-run] [--force] [--receive-pack=<git-receive-pack>] [--verbose] [--thin] [<host>:]<directory> [<ref>...]\n" |
18bd8821 | 15 | " --all and explicit <ref> specification are mutually exclusive."; |
96249c04 | 16 | |
6828f72f | 17 | static struct send_pack_args args; |
61221472 | 18 | |
de1a2fdd SP |
19 | static void print_helper_status(struct ref *ref) |
20 | { | |
21 | struct strbuf buf = STRBUF_INIT; | |
22 | ||
23 | for (; ref; ref = ref->next) { | |
24 | const char *msg = NULL; | |
25 | const char *res; | |
26 | ||
27 | switch(ref->status) { | |
28 | case REF_STATUS_NONE: | |
29 | res = "error"; | |
30 | msg = "no match"; | |
31 | break; | |
32 | ||
33 | case REF_STATUS_OK: | |
34 | res = "ok"; | |
35 | break; | |
36 | ||
37 | case REF_STATUS_UPTODATE: | |
38 | res = "ok"; | |
39 | msg = "up to date"; | |
40 | break; | |
41 | ||
42 | case REF_STATUS_REJECT_NONFASTFORWARD: | |
43 | res = "error"; | |
44 | msg = "non-fast forward"; | |
45 | break; | |
46 | ||
47 | case REF_STATUS_REJECT_NODELETE: | |
48 | case REF_STATUS_REMOTE_REJECT: | |
49 | res = "error"; | |
50 | break; | |
51 | ||
52 | case REF_STATUS_EXPECTING_REPORT: | |
53 | default: | |
54 | continue; | |
55 | } | |
56 | ||
57 | strbuf_reset(&buf); | |
58 | strbuf_addf(&buf, "%s %s", res, ref->name); | |
59 | if (ref->remote_status) | |
60 | msg = ref->remote_status; | |
61 | if (msg) { | |
62 | strbuf_addch(&buf, ' '); | |
63 | quote_two_c_style(&buf, "", msg, 0); | |
64 | } | |
65 | strbuf_addch(&buf, '\n'); | |
66 | ||
67 | safe_write(1, buf.buf, buf.len); | |
68 | } | |
69 | strbuf_release(&buf); | |
70 | } | |
71 | ||
96249c04 | 72 | int cmd_send_pack(int argc, const char **argv, const char *prefix) |
61221472 | 73 | { |
64fcef2d DB |
74 | int i, nr_refspecs = 0; |
75 | const char **refspecs = NULL; | |
96249c04 | 76 | const char *remote_name = NULL; |
b516968f | 77 | struct remote *remote = NULL; |
96249c04 | 78 | const char *dest = NULL; |
64fcef2d DB |
79 | int fd[2]; |
80 | struct child_process *conn; | |
81 | struct extra_have_objects extra_have; | |
6d2bf96e | 82 | struct ref *remote_refs, *local_refs; |
64fcef2d | 83 | int ret; |
de1a2fdd | 84 | int helper_status = 0; |
64fcef2d DB |
85 | int send_all = 0; |
86 | const char *receivepack = "git-receive-pack"; | |
87 | int flags; | |
f1863d0d | 88 | int nonfastforward = 0; |
391b1f20 | 89 | int progress = -1; |
84a9b58c | 90 | |
61221472 | 91 | argv++; |
d089391c | 92 | for (i = 1; i < argc; i++, argv++) { |
96249c04 | 93 | const char *arg = *argv; |
61221472 LT |
94 | |
95 | if (*arg == '-') { | |
cc44c765 | 96 | if (!prefixcmp(arg, "--receive-pack=")) { |
64fcef2d | 97 | receivepack = arg + 15; |
d23842fd UKK |
98 | continue; |
99 | } | |
cc44c765 | 100 | if (!prefixcmp(arg, "--exec=")) { |
64fcef2d | 101 | receivepack = arg + 7; |
61221472 LT |
102 | continue; |
103 | } | |
b516968f DB |
104 | if (!prefixcmp(arg, "--remote=")) { |
105 | remote_name = arg + 9; | |
106 | continue; | |
107 | } | |
d089391c | 108 | if (!strcmp(arg, "--all")) { |
64fcef2d | 109 | send_all = 1; |
d089391c LT |
110 | continue; |
111 | } | |
a63103ae | 112 | if (!strcmp(arg, "--dry-run")) { |
96249c04 | 113 | args.dry_run = 1; |
a63103ae BE |
114 | continue; |
115 | } | |
28b9d6e5 AW |
116 | if (!strcmp(arg, "--mirror")) { |
117 | args.send_mirror = 1; | |
118 | continue; | |
119 | } | |
2a9c3fe8 | 120 | if (!strcmp(arg, "--force")) { |
96249c04 | 121 | args.force_update = 1; |
2a9c3fe8 LT |
122 | continue; |
123 | } | |
c207e34f CB |
124 | if (!strcmp(arg, "--quiet")) { |
125 | args.quiet = 1; | |
126 | continue; | |
127 | } | |
41f93a2c | 128 | if (!strcmp(arg, "--verbose")) { |
96249c04 | 129 | args.verbose = 1; |
41f93a2c LT |
130 | continue; |
131 | } | |
391b1f20 JK |
132 | if (!strcmp(arg, "--progress")) { |
133 | progress = 1; | |
134 | continue; | |
135 | } | |
136 | if (!strcmp(arg, "--no-progress")) { | |
137 | progress = 0; | |
138 | continue; | |
139 | } | |
2245be3e | 140 | if (!strcmp(arg, "--thin")) { |
96249c04 | 141 | args.use_thin_pack = 1; |
2245be3e JH |
142 | continue; |
143 | } | |
de1a2fdd SP |
144 | if (!strcmp(arg, "--stateless-rpc")) { |
145 | args.stateless_rpc = 1; | |
146 | continue; | |
147 | } | |
148 | if (!strcmp(arg, "--helper-status")) { | |
149 | helper_status = 1; | |
150 | continue; | |
151 | } | |
61221472 LT |
152 | usage(send_pack_usage); |
153 | } | |
d089391c LT |
154 | if (!dest) { |
155 | dest = arg; | |
156 | continue; | |
157 | } | |
64fcef2d DB |
158 | refspecs = (const char **) argv; |
159 | nr_refspecs = argc - i; | |
61221472 LT |
160 | break; |
161 | } | |
162 | if (!dest) | |
163 | usage(send_pack_usage); | |
28b9d6e5 AW |
164 | /* |
165 | * --all and --mirror are incompatible; neither makes sense | |
166 | * with any refspecs. | |
167 | */ | |
64fcef2d DB |
168 | if ((refspecs && (send_all || args.send_mirror)) || |
169 | (send_all && args.send_mirror)) | |
0bc3cdfc | 170 | usage(send_pack_usage); |
37adac76 | 171 | |
b516968f DB |
172 | if (remote_name) { |
173 | remote = remote_get(remote_name); | |
28b91f8a | 174 | if (!remote_has_url(remote, dest)) { |
b516968f DB |
175 | die("Destination %s is not a uri for %s", |
176 | dest, remote_name); | |
177 | } | |
178 | } | |
179 | ||
391b1f20 JK |
180 | if (progress == -1) |
181 | progress = !args.quiet && isatty(2); | |
182 | args.progress = progress; | |
8d32e60d | 183 | |
de1a2fdd SP |
184 | if (args.stateless_rpc) { |
185 | conn = NULL; | |
186 | fd[0] = 0; | |
187 | fd[1] = 1; | |
188 | } else { | |
5a277f3f | 189 | conn = git_connect(fd, dest, receivepack, |
de1a2fdd SP |
190 | args.verbose ? CONNECT_VERBOSE : 0); |
191 | } | |
96249c04 | 192 | |
64fcef2d DB |
193 | memset(&extra_have, 0, sizeof(extra_have)); |
194 | ||
afe7c5ff | 195 | get_remote_heads(fd[0], &remote_refs, REF_NORMAL, &extra_have); |
64fcef2d | 196 | |
f1863d0d | 197 | transport_verify_remote_names(nr_refspecs, refspecs); |
64fcef2d DB |
198 | |
199 | local_refs = get_local_heads(); | |
96249c04 | 200 | |
64fcef2d DB |
201 | flags = MATCH_REFS_NONE; |
202 | ||
203 | if (send_all) | |
204 | flags |= MATCH_REFS_ALL; | |
205 | if (args.send_mirror) | |
206 | flags |= MATCH_REFS_MIRROR; | |
207 | ||
208 | /* match them up */ | |
29753cdd | 209 | if (match_push_refs(local_refs, &remote_refs, nr_refspecs, refspecs, flags)) |
64fcef2d | 210 | return -1; |
96249c04 | 211 | |
20e8b465 TRC |
212 | set_ref_status_for_push(remote_refs, args.send_mirror, |
213 | args.force_update); | |
214 | ||
64fcef2d | 215 | ret = send_pack(&args, fd, conn, remote_refs, &extra_have); |
96249c04 | 216 | |
de1a2fdd SP |
217 | if (helper_status) |
218 | print_helper_status(remote_refs); | |
219 | ||
64fcef2d | 220 | close(fd[1]); |
7f8e9828 | 221 | close(fd[0]); |
64fcef2d | 222 | |
98158e9c | 223 | ret |= finish_connect(conn); |
64fcef2d | 224 | |
de1a2fdd | 225 | if (!helper_status) |
f1863d0d | 226 | transport_print_push_status(dest, remote_refs, args.verbose, 0, &nonfastforward); |
64fcef2d DB |
227 | |
228 | if (!args.dry_run && remote) { | |
229 | struct ref *ref; | |
230 | for (ref = remote_refs; ref; ref = ref->next) | |
f1863d0d | 231 | transport_update_tracking_ref(remote, ref, args.verbose); |
64fcef2d DB |
232 | } |
233 | ||
f1863d0d | 234 | if (!ret && !transport_refs_pushed(remote_refs)) |
64fcef2d DB |
235 | fprintf(stderr, "Everything up-to-date\n"); |
236 | ||
237 | return ret; | |
61221472 | 238 | } |