Fix sparse warnings
Fix warnings from 'make check'.
- These files don't include 'builtin.h' causing sparse to complain that
cmd_* isn't declared:
builtin/clone.c:364, builtin/fetch-pack.c:797,
builtin/fmt-merge-msg.c:34, builtin/hash-object.c:78,
builtin/merge-index.c:69, builtin/merge-recursive.c:22
builtin/merge-tree.c:341, builtin/mktag.c:156, builtin/notes.c:426
builtin/notes.c:822, builtin/pack-redundant.c:596,
builtin/pack-refs.c:10, builtin/patch-id.c:60, builtin/patch-id.c:149,
builtin/remote.c:1512, builtin/remote-ext.c:240,
builtin/remote-fd.c:53, builtin/reset.c:236, builtin/send-pack.c:384,
builtin/unpack-file.c:25, builtin/var.c:75
- These files have symbols which should be marked static since they're
only file scope:
submodule.c:12, diff.c:631, replace_object.c:92, submodule.c:13,
submodule.c:14, trace.c:78, transport.c:195, transport-helper.c:79,
unpack-trees.c:19, url.c:3, url.c:18, url.c:104, url.c:117, url.c:123,
url.c:129, url.c:136, thread-utils.c:21, thread-utils.c:48
- These files redeclare symbols to be different types:
builtin/index-pack.c:210, parse-options.c:564, parse-options.c:571,
usage.c:49, usage.c:58, usage.c:63, usage.c:72
- These files use a literal integer 0 when they really should use a NULL
pointer:
daemon.c:663, fast-import.c:2942, imap-send.c:1072, notes-merge.c:362
While we're in the area, clean up some unused #includes in builtin files
(mostly exec_cmd.h).
Signed-off-by: Stephen Boyd <bebarino@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
14 years ago
|
|
|
#include "builtin.h"
|
|
|
|
#include "config.h"
|
|
|
|
#include "commit.h"
|
|
|
|
#include "refs.h"
|
|
|
|
#include "pkt-line.h"
|
|
|
|
#include "sideband.h"
|
|
|
|
#include "run-command.h"
|
|
|
|
#include "remote.h"
|
|
|
|
#include "connect.h"
|
|
|
|
#include "send-pack.h"
|
|
|
|
#include "quote.h"
|
|
|
|
#include "transport.h"
|
|
|
|
#include "version.h"
|
|
|
|
#include "sha1-array.h"
|
|
|
|
#include "gpg-interface.h"
|
|
|
|
#include "gettext.h"
|
|
|
|
#include "protocol.h"
|
|
|
|
|
|
|
|
static const char * const send_pack_usage[] = {
|
|
|
|
N_("git send-pack [--all | --mirror] [--dry-run] [--force] "
|
|
|
|
"[--receive-pack=<git-receive-pack>] [--verbose] [--thin] [--atomic] "
|
|
|
|
"[<host>:]<directory> [<ref>...]\n"
|
|
|
|
" --all and explicit <ref> specification are mutually exclusive."),
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct send_pack_args args;
|
|
|
|
|
|
|
|
static void print_helper_status(struct ref *ref)
|
|
|
|
{
|
|
|
|
struct strbuf buf = STRBUF_INIT;
|
|
|
|
|
|
|
|
for (; ref; ref = ref->next) {
|
|
|
|
const char *msg = NULL;
|
|
|
|
const char *res;
|
|
|
|
|
|
|
|
switch(ref->status) {
|
|
|
|
case REF_STATUS_NONE:
|
|
|
|
res = "error";
|
|
|
|
msg = "no match";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_OK:
|
|
|
|
res = "ok";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_UPTODATE:
|
|
|
|
res = "ok";
|
|
|
|
msg = "up to date";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_REJECT_NONFASTFORWARD:
|
|
|
|
res = "error";
|
|
|
|
msg = "non-fast forward";
|
|
|
|
break;
|
|
|
|
|
push: introduce REJECT_FETCH_FIRST and REJECT_NEEDS_FORCE
When we push to update an existing ref, if:
* the object at the tip of the remote is not a commit; or
* the object we are pushing is not a commit,
it won't be correct to suggest to fetch, integrate and push again,
as the old and new objects will not "merge". We should explain that
the push must be forced when there is a non-committish object is
involved in such a case.
If we do not have the current object at the tip of the remote, we do
not even know that object, when fetched, is something that can be
merged. In such a case, suggesting to pull first just like
non-fast-forward case may not be technically correct, but in
practice, most such failures are seen when you try to push your work
to a branch without knowing that somebody else already pushed to
update the same branch since you forked, so "pull first" would work
as a suggestion most of the time. And if the object at the tip is
not a commit, "pull first" will fail, without making any permanent
damage. As a side effect, it also makes the error message the user
will get during the next "push" attempt easier to understand, now
the user is aware that a non-commit object is involved.
In these cases, the current code already rejects such a push on the
client end, but we used the same error and advice messages as the
ones used when rejecting a non-fast-forward push, i.e. pull from
there and integrate before pushing again.
Introduce new rejection reasons and reword the messages
appropriately.
[jc: with help by Peff on message details]
Signed-off-by: Junio C Hamano <gitster@pobox.com>
12 years ago
|
|
|
case REF_STATUS_REJECT_FETCH_FIRST:
|
|
|
|
res = "error";
|
|
|
|
msg = "fetch first";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_REJECT_NEEDS_FORCE:
|
|
|
|
res = "error";
|
|
|
|
msg = "needs force";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_REJECT_STALE:
|
|
|
|
res = "error";
|
|
|
|
msg = "stale info";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_REJECT_ALREADY_EXISTS:
|
|
|
|
res = "error";
|
|
|
|
msg = "already exists";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_REJECT_NODELETE:
|
|
|
|
case REF_STATUS_REMOTE_REJECT:
|
|
|
|
res = "error";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case REF_STATUS_EXPECTING_REPORT:
|
|
|
|
default:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
strbuf_reset(&buf);
|
|
|
|
strbuf_addf(&buf, "%s %s", res, ref->name);
|
|
|
|
if (ref->remote_status)
|
|
|
|
msg = ref->remote_status;
|
|
|
|
if (msg) {
|
|
|
|
strbuf_addch(&buf, ' ');
|
|
|
|
quote_two_c_style(&buf, "", msg, 0);
|
|
|
|
}
|
|
|
|
strbuf_addch(&buf, '\n');
|
|
|
|
|
|
|
|
write_or_die(1, buf.buf, buf.len);
|
|
|
|
}
|
|
|
|
strbuf_release(&buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int send_pack_config(const char *k, const char *v, void *cb)
|
|
|
|
{
|
|
|
|
git_gpg_config(k, v, NULL);
|
|
|
|
|
|
|
|
if (!strcmp(k, "push.gpgsign")) {
|
|
|
|
const char *value;
|
|
|
|
if (!git_config_get_value("push.gpgsign", &value)) {
|
|
|
|
switch (git_parse_maybe_bool(value)) {
|
|
|
|
case 0:
|
|
|
|
args.push_cert = SEND_PACK_PUSH_CERT_NEVER;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
args.push_cert = SEND_PACK_PUSH_CERT_ALWAYS;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (value && !strcasecmp(value, "if-asked"))
|
|
|
|
args.push_cert = SEND_PACK_PUSH_CERT_IF_ASKED;
|
|
|
|
else
|
|
|
|
return error("Invalid value for '%s'", k);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return git_default_config(k, v, cb);
|
|
|
|
}
|
|
|
|
|
|
|
|
int cmd_send_pack(int argc, const char **argv, const char *prefix)
|
|
|
|
{
|
|
|
|
struct refspec rs = REFSPEC_INIT_PUSH;
|
|
|
|
const char *remote_name = NULL;
|
|
|
|
struct remote *remote = NULL;
|
|
|
|
const char *dest = NULL;
|
|
|
|
int fd[2];
|
|
|
|
struct child_process *conn;
|
|
|
|
struct oid_array extra_have = OID_ARRAY_INIT;
|
|
|
|
struct oid_array shallow = OID_ARRAY_INIT;
|
|
|
|
struct ref *remote_refs, *local_refs;
|
|
|
|
int ret;
|
|
|
|
int helper_status = 0;
|
|
|
|
int send_all = 0;
|
|
|
|
int verbose = 0;
|
|
|
|
const char *receivepack = "git-receive-pack";
|
|
|
|
unsigned dry_run = 0;
|
|
|
|
unsigned send_mirror = 0;
|
|
|
|
unsigned force_update = 0;
|
|
|
|
unsigned quiet = 0;
|
|
|
|
int push_cert = 0;
|
|
|
|
struct string_list push_options = STRING_LIST_INIT_NODUP;
|
|
|
|
unsigned use_thin_pack = 0;
|
|
|
|
unsigned atomic = 0;
|
|
|
|
unsigned stateless_rpc = 0;
|
|
|
|
int flags;
|
|
|
|
unsigned int reject_reasons;
|
|
|
|
int progress = -1;
|
|
|
|
int from_stdin = 0;
|
remote.c: add command line option parser for "--force-with-lease"
Update "git push" and "git send-pack" to parse this commnd line
option.
The intended sematics is:
* "--force-with-lease" alone, without specifying the details, will
protect _all_ remote refs that are going to be updated by
requiring their current value to be the same as some reasonable
default, unless otherwise specified;
* "--force-with-lease=refname", without specifying the expected
value, will protect that refname, if it is going to be updated,
by requiring its current value to be the same as some reasonable
default.
* "--force-with-lease=refname:value" will protect that refname, if
it is going to be updated, by requiring its current value to be
the same as the specified value; and
* "--no-force-with-lease" will cancel all the previous --force-with-lease on the
command line.
For now, "some reasonable default" is tentatively defined as "the
value of the remote-tracking branch we have for the ref of the
remote being updated", and it is an error if we do not have such a
remote-tracking branch. But this is known to be fragile, its use is
not yet recommended, and hopefully we will find more reasonable
default as we gain experience with this feature. The manual marks
the feature as experimental unless the expected value is specified
explicitly for this reason.
Because the command line options are parsed _before_ we know which
remote we are pushing to, there needs further processing to the
parsed data after we instantiate the transport object to:
* expand "refname" given by the user to a full refname to be
matched with the list of "struct ref" used in match_push_refs()
and set_ref_status_for_push(); and
* learning the actual local ref that is the remote-tracking branch
for the specified remote ref.
Further, some processing need to be deferred until we find the set
of remote refs and match_push_refs() returns in order to find the
ones that need to be checked after explicit ones have been processed
for "--force-with-lease" (no specific details).
These post-processing will be the topic of the next patch.
This option was originally called "cas" (for "compare and swap"),
the name which nobody liked because it was too technical. The
second attempt called it "lockref" (because it is conceptually like
pushing after taking a lock) but the word "lock" was hated because
it implied that it may reject push by others, which is not the way
this option works. This round calls it "force-with-lease". You
assume you took the lease on the ref when you fetched to decide what
the rebased history should be, and you can push back only if the
lease has not been broken.
Signed-off-by: Junio C Hamano <gitster@pobox.com>
12 years ago
|
|
|
struct push_cas_option cas = {0};
|
|
|
|
struct packet_reader reader;
|
|
|
|
|
|
|
|
struct option options[] = {
|
|
|
|
OPT__VERBOSITY(&verbose),
|
|
|
|
OPT_STRING(0, "receive-pack", &receivepack, "receive-pack", N_("receive pack program")),
|
|
|
|
OPT_STRING(0, "exec", &receivepack, "receive-pack", N_("receive pack program")),
|
|
|
|
OPT_STRING(0, "remote", &remote_name, "remote", N_("remote name")),
|
|
|
|
OPT_BOOL(0, "all", &send_all, N_("push all refs")),
|
|
|
|
OPT_BOOL('n' , "dry-run", &dry_run, N_("dry run")),
|
|
|
|
OPT_BOOL(0, "mirror", &send_mirror, N_("mirror all refs")),
|
|
|
|
OPT_BOOL('f', "force", &force_update, N_("force updates")),
|
|
|
|
{ OPTION_CALLBACK,
|
|
|
|
0, "signed", &push_cert, "(yes|no|if-asked)", N_("GPG sign the push"),
|
|
|
|
PARSE_OPT_OPTARG, option_parse_push_signed },
|
|
|
|
OPT_STRING_LIST(0, "push-option", &push_options,
|
|
|
|
N_("server-specific"),
|
|
|
|
N_("option to transmit")),
|
|
|
|
OPT_BOOL(0, "progress", &progress, N_("force progress reporting")),
|
|
|
|
OPT_BOOL(0, "thin", &use_thin_pack, N_("use thin pack")),
|
|
|
|
OPT_BOOL(0, "atomic", &atomic, N_("request atomic transaction on remote side")),
|
|
|
|
OPT_BOOL(0, "stateless-rpc", &stateless_rpc, N_("use stateless RPC protocol")),
|
|
|
|
OPT_BOOL(0, "stdin", &from_stdin, N_("read refs from stdin")),
|
|
|
|
OPT_BOOL(0, "helper-status", &helper_status, N_("print status from remote helper")),
|
|
|
|
{ OPTION_CALLBACK,
|
|
|
|
0, CAS_OPT_NAME, &cas, N_("<refname>:<expect>"),
|
|
|
|
N_("require old value of ref to be at this value"),
|
|
|
|
PARSE_OPT_OPTARG, parseopt_push_cas_option },
|
|
|
|
OPT_END()
|
|
|
|
};
|
|
|
|
|
|
|
|
git_config(send_pack_config, NULL);
|
|
|
|
argc = parse_options(argc, argv, prefix, options, send_pack_usage, 0);
|
|
|
|
if (argc > 0) {
|
|
|
|
dest = argv[0];
|
|
|
|
refspec_appendn(&rs, argv + 1, argc - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!dest)
|
|
|
|
usage_with_options(send_pack_usage, options);
|
|
|
|
|
|
|
|
args.verbose = verbose;
|
|
|
|
args.dry_run = dry_run;
|
|
|
|
args.send_mirror = send_mirror;
|
|
|
|
args.force_update = force_update;
|
|
|
|
args.quiet = quiet;
|
|
|
|
args.push_cert = push_cert;
|
|
|
|
args.progress = progress;
|
|
|
|
args.use_thin_pack = use_thin_pack;
|
|
|
|
args.atomic = atomic;
|
|
|
|
args.stateless_rpc = stateless_rpc;
|
|
|
|
args.push_options = push_options.nr ? &push_options : NULL;
|
|
|
|
|
|
|
|
if (from_stdin) {
|
|
|
|
if (args.stateless_rpc) {
|
|
|
|
const char *buf;
|
|
|
|
while ((buf = packet_read_line(0, NULL)))
|
|
|
|
refspec_append(&rs, buf);
|
|
|
|
} else {
|
|
|
|
struct strbuf line = STRBUF_INIT;
|
|
|
|
while (strbuf_getline(&line, stdin) != EOF)
|
|
|
|
refspec_append(&rs, line.buf);
|
|
|
|
strbuf_release(&line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* --all and --mirror are incompatible; neither makes sense
|
|
|
|
* with any refspecs.
|
|
|
|
*/
|
|
|
|
if ((rs.nr > 0 && (send_all || args.send_mirror)) ||
|
|
|
|
(send_all && args.send_mirror))
|
|
|
|
usage_with_options(send_pack_usage, options);
|
|
|
|
|
|
|
|
if (remote_name) {
|
|
|
|
remote = remote_get(remote_name);
|
|
|
|
if (!remote_has_url(remote, dest)) {
|
|
|
|
die("Destination %s is not a uri for %s",
|
|
|
|
dest, remote_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (progress == -1)
|
|
|
|
progress = !args.quiet && isatty(2);
|
|
|
|
args.progress = progress;
|
send-pack: show progress when isatty(2)
The send_pack_args struct has two verbosity flags: "quiet"
and "progress". Originally, if "quiet" was set, we would
tell pack-objects explicitly to be quiet, and if "progress"
was set, we would tell it to show progress. Otherwise, we
told it neither, and it relied on isatty(2) to make the
decision itself.
However, commit 01fdc21 changed the meaning of these
variables. Now both "quiet" and "!progress" instruct us to
tell pack-objects to be quiet (and a non-zero "progress"
means the same as before). This works well for transports
which call send_pack directly, as the transport code copies
transport->progress into send_pack_args->progress, and they
both have the same meaning.
However, the code path of calling "git send-pack" was left
behind. It always sets "progress" to 0, and thus always
tells pack-objects to be quiet. We can work around this by
checking isatty(2) ourselves in the cmd_send_pack code path,
restoring the original behavior of the send-pack command.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
13 years ago
|
|
|
|
|
|
|
if (args.stateless_rpc) {
|
|
|
|
conn = NULL;
|
|
|
|
fd[0] = 0;
|
|
|
|
fd[1] = 1;
|
|
|
|
} else {
|
|
|
|
conn = git_connect(fd, dest, receivepack,
|
|
|
|
args.verbose ? CONNECT_VERBOSE : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
packet_reader_init(&reader, fd[0], NULL, 0,
|
|
|
|
PACKET_READ_CHOMP_NEWLINE |
|
pack-protocol.txt: accept error packets in any context
In the Git pack protocol definition, an error packet may appear only in
a certain context. However, servers can face a runtime error (e.g. I/O
error) at an arbitrary timing. This patch changes the protocol to allow
an error packet to be sent instead of any packet.
Without this protocol spec change, when a server cannot process a
request, there's no way to tell that to a client. Since the server
cannot produce a valid response, it would be forced to cut a connection
without telling why. With this protocol spec change, the server can be
more gentle in this situation. An old client may see these error packets
as an unexpected packet, but this is not worse than having an unexpected
EOF.
Following this protocol spec change, the error packet handling code is
moved to pkt-line.c. Implementation wise, this implementation uses
pkt-line to communicate with a subprocess. Since this is not a part of
Git protocol, it's possible that a packet that is not supposed to be an
error packet is mistakenly parsed as an error packet. This error packet
handling is enabled only for the Git pack protocol parsing code
considering this.
Signed-off-by: Masaya Suzuki <masayasuzuki@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
6 years ago
|
|
|
PACKET_READ_GENTLE_ON_EOF |
|
|
|
|
PACKET_READ_DIE_ON_ERR_PACKET);
|
|
|
|
|
|
|
|
switch (discover_version(&reader)) {
|
|
|
|
case protocol_v2:
|
|
|
|
die("support for protocol v2 not implemented yet");
|
|
|
|
break;
|
|
|
|
case protocol_v1:
|
|
|
|
case protocol_v0:
|
|
|
|
get_remote_heads(&reader, &remote_refs, REF_NORMAL,
|
|
|
|
&extra_have, &shallow);
|
|
|
|
break;
|
|
|
|
case protocol_unknown_version:
|
|
|
|
BUG("unknown protocol version");
|
|
|
|
}
|
|
|
|
|
|
|
|
local_refs = get_local_heads();
|
|
|
|
|
|
|
|
flags = MATCH_REFS_NONE;
|
|
|
|
|
|
|
|
if (send_all)
|
|
|
|
flags |= MATCH_REFS_ALL;
|
|
|
|
if (args.send_mirror)
|
|
|
|
flags |= MATCH_REFS_MIRROR;
|
|
|
|
|
|
|
|
/* match them up */
|
|
|
|
if (match_push_refs(local_refs, &remote_refs, &rs, flags))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!is_empty_cas(&cas))
|
|
|
|
apply_push_cas(&cas, remote, remote_refs);
|
|
|
|
|
|
|
|
set_ref_status_for_push(remote_refs, args.send_mirror,
|
|
|
|
args.force_update);
|
|
|
|
|
|
|
|
ret = send_pack(&args, fd, conn, remote_refs, &extra_have);
|
|
|
|
|
|
|
|
if (helper_status)
|
|
|
|
print_helper_status(remote_refs);
|
|
|
|
|
|
|
|
close(fd[1]);
|
|
|
|
close(fd[0]);
|
|
|
|
|
|
|
|
ret |= finish_connect(conn);
|
|
|
|
|
|
|
|
if (!helper_status)
|
|
|
|
transport_print_push_status(dest, remote_refs, args.verbose, 0, &reject_reasons);
|
|
|
|
|
|
|
|
if (!args.dry_run && remote) {
|
|
|
|
struct ref *ref;
|
|
|
|
for (ref = remote_refs; ref; ref = ref->next)
|
|
|
|
transport_update_tracking_ref(remote, ref, args.verbose);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ret && !transport_refs_pushed(remote_refs))
|
|
|
|
fprintf(stderr, "Everything up-to-date\n");
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|