Merge branch 'sp/smart-http'
* sp/smart-http: (37 commits) http-backend: Let gcc check the format of more printf-type functions. http-backend: Fix access beyond end of string. http-backend: Fix bad treatment of uintmax_t in Content-Length t5551-http-fetch: Work around broken Accept header in libcurl t5551-http-fetch: Work around some libcurl versions http-backend: Protect GIT_PROJECT_ROOT from /../ requests Git-aware CGI to provide dumb HTTP transport http-backend: Test configuration options http-backend: Use http.getanyfile to disable dumb HTTP serving test smart http fetch and push http tests: use /dumb/ URL prefix set httpd port before sourcing lib-httpd t5540-http-push: remove redundant fetches Smart HTTP fetch: gzip requests Smart fetch over HTTP: client side Smart push over HTTP: client side Discover refs via smart HTTP server when available http-backend: more explict LocationMatch http-backend: add example for gitweb on same URL http-backend: use mod_alias instead of mod_rewrite ... Conflicts: .gitignore remote-curl.cmaint
commit
905bf7742c
|
@ -55,6 +55,7 @@
|
|||
/git-grep
|
||||
/git-hash-object
|
||||
/git-help
|
||||
/git-http-backend
|
||||
/git-http-fetch
|
||||
/git-http-push
|
||||
/git-imap-send
|
||||
|
|
|
@ -1091,6 +1091,14 @@ http.maxRequests::
|
|||
How many HTTP requests to launch in parallel. Can be overridden
|
||||
by the 'GIT_HTTP_MAX_REQUESTS' environment variable. Default is 5.
|
||||
|
||||
http.postBuffer::
|
||||
Maximum size in bytes of the buffer used by smart HTTP
|
||||
transports when POSTing data to the remote system.
|
||||
For requests larger than this buffer size, HTTP/1.1 and
|
||||
Transfer-Encoding: chunked is used to avoid creating a
|
||||
massive pack file locally. Default is 1 MiB, which is
|
||||
sufficient for most requests.
|
||||
|
||||
http.lowSpeedLimit, http.lowSpeedTime::
|
||||
If the HTTP transfer speed is less than 'http.lowSpeedLimit'
|
||||
for longer than 'http.lowSpeedTime' seconds, the transfer is aborted.
|
||||
|
|
|
@ -0,0 +1,178 @@
|
|||
git-http-backend(1)
|
||||
===================
|
||||
|
||||
NAME
|
||||
----
|
||||
git-http-backend - Server side implementation of Git over HTTP
|
||||
|
||||
SYNOPSIS
|
||||
--------
|
||||
[verse]
|
||||
'git-http-backend'
|
||||
|
||||
DESCRIPTION
|
||||
-----------
|
||||
A simple CGI program to serve the contents of a Git repository to Git
|
||||
clients accessing the repository over http:// and https:// protocols.
|
||||
The program supports clients fetching using both the smart HTTP protcol
|
||||
and the backwards-compatible dumb HTTP protocol, as well as clients
|
||||
pushing using the smart HTTP protocol.
|
||||
|
||||
By default, only the `upload-pack` service is enabled, which serves
|
||||
'git-fetch-pack' and 'git-ls-remote' clients, which are invoked from
|
||||
'git-fetch', 'git-pull', and 'git-clone'. If the client is authenticated,
|
||||
the `receive-pack` service is enabled, which serves 'git-send-pack'
|
||||
clients, which is invoked from 'git-push'.
|
||||
|
||||
SERVICES
|
||||
--------
|
||||
These services can be enabled/disabled using the per-repository
|
||||
configuration file:
|
||||
|
||||
http.getanyfile::
|
||||
This serves older Git clients which are unable to use the
|
||||
upload pack service. When enabled, clients are able to read
|
||||
any file within the repository, including objects that are
|
||||
no longer reachable from a branch but are still present.
|
||||
It is enabled by default, but a repository can disable it
|
||||
by setting this configuration item to `false`.
|
||||
|
||||
http.uploadpack::
|
||||
This serves 'git-fetch-pack' and 'git-ls-remote' clients.
|
||||
It is enabled by default, but a repository can disable it
|
||||
by setting this configuration item to `false`.
|
||||
|
||||
http.receivepack::
|
||||
This serves 'git-send-pack' clients, allowing push. It is
|
||||
disabled by default for anonymous users, and enabled by
|
||||
default for users authenticated by the web server. It can be
|
||||
disabled by setting this item to `false`, or enabled for all
|
||||
users, including anonymous users, by setting it to `true`.
|
||||
|
||||
URL TRANSLATION
|
||||
---------------
|
||||
To determine the location of the repository on disk, 'git-http-backend'
|
||||
concatenates the environment variables PATH_INFO, which is set
|
||||
automatically by the web server, and GIT_PROJECT_ROOT, which must be set
|
||||
manually in the web server configuration. If GIT_PROJECT_ROOT is not
|
||||
set, 'git-http-backend' reads PATH_TRANSLATED, which is also set
|
||||
automatically by the web server.
|
||||
|
||||
EXAMPLES
|
||||
--------
|
||||
All of the following examples map 'http://$hostname/git/foo/bar.git'
|
||||
to '/var/www/git/foo/bar.git'.
|
||||
|
||||
Apache 2.x::
|
||||
Ensure mod_cgi, mod_alias, and mod_env are enabled, set
|
||||
GIT_PROJECT_ROOT (or DocumentRoot) appropriately, and
|
||||
create a ScriptAlias to the CGI:
|
||||
+
|
||||
----------------------------------------------------------------
|
||||
SetEnv GIT_PROJECT_ROOT /var/www/git
|
||||
ScriptAlias /git/ /usr/libexec/git-core/git-http-backend/
|
||||
----------------------------------------------------------------
|
||||
+
|
||||
To enable anonymous read access but authenticated write access,
|
||||
require authorization with a LocationMatch directive:
|
||||
+
|
||||
----------------------------------------------------------------
|
||||
<LocationMatch "^/git/.*/git-receive-pack$">
|
||||
AuthType Basic
|
||||
AuthName "Git Access"
|
||||
Require group committers
|
||||
...
|
||||
</LocationMatch>
|
||||
----------------------------------------------------------------
|
||||
+
|
||||
To require authentication for both reads and writes, use a Location
|
||||
directive around the repository, or one of its parent directories:
|
||||
+
|
||||
----------------------------------------------------------------
|
||||
<Location /git/private>
|
||||
AuthType Basic
|
||||
AuthName "Private Git Access"
|
||||
Require group committers
|
||||
...
|
||||
</Location>
|
||||
----------------------------------------------------------------
|
||||
+
|
||||
To serve gitweb at the same url, use a ScriptAliasMatch to only
|
||||
those URLs that 'git-http-backend' can handle, and forward the
|
||||
rest to gitweb:
|
||||
+
|
||||
----------------------------------------------------------------
|
||||
ScriptAliasMatch \
|
||||
"(?x)^/git/(.*/(HEAD | \
|
||||
info/refs | \
|
||||
objects/(info/[^/]+ | \
|
||||
[0-9a-f]{2}/[0-9a-f]{38} | \
|
||||
pack/pack-[0-9a-f]{40}\.(pack|idx)) | \
|
||||
git-(upload|receive)-pack))$" \
|
||||
/usr/libexec/git-core/git-http-backend/$1
|
||||
|
||||
ScriptAlias /git/ /var/www/cgi-bin/gitweb.cgi/
|
||||
----------------------------------------------------------------
|
||||
|
||||
Accelerated static Apache 2.x::
|
||||
Similar to the above, but Apache can be used to return static
|
||||
files that are stored on disk. On many systems this may
|
||||
be more efficient as Apache can ask the kernel to copy the
|
||||
file contents from the file system directly to the network:
|
||||
+
|
||||
----------------------------------------------------------------
|
||||
SetEnv GIT_PROJECT_ROOT /var/www/git
|
||||
|
||||
AliasMatch ^/git/(.*/objects/[0-9a-f]{2}/[0-9a-f]{38})$ /var/www/git/$1
|
||||
AliasMatch ^/git/(.*/objects/pack/pack-[0-9a-f]{40}.(pack|idx))$ /var/www/git/$1
|
||||
ScriptAlias /git/ /usr/libexec/git-core/git-http-backend/
|
||||
----------------------------------------------------------------
|
||||
+
|
||||
This can be combined with the gitweb configuration:
|
||||
+
|
||||
----------------------------------------------------------------
|
||||
SetEnv GIT_PROJECT_ROOT /var/www/git
|
||||
|
||||
AliasMatch ^/git/(.*/objects/[0-9a-f]{2}/[0-9a-f]{38})$ /var/www/git/$1
|
||||
AliasMatch ^/git/(.*/objects/pack/pack-[0-9a-f]{40}.(pack|idx))$ /var/www/git/$1
|
||||
ScriptAliasMatch \
|
||||
"(?x)^/git/(.*/(HEAD | \
|
||||
info/refs | \
|
||||
objects/info/[^/]+ | \
|
||||
git-(upload|receive)-pack))$" \
|
||||
/usr/libexec/git-core/git-http-backend/$1
|
||||
ScriptAlias /git/ /var/www/cgi-bin/gitweb.cgi/
|
||||
----------------------------------------------------------------
|
||||
|
||||
|
||||
ENVIRONMENT
|
||||
-----------
|
||||
'git-http-backend' relies upon the CGI environment variables set
|
||||
by the invoking web server, including:
|
||||
|
||||
* PATH_INFO (if GIT_PROJECT_ROOT is set, otherwise PATH_TRANSLATED)
|
||||
* REMOTE_USER
|
||||
* REMOTE_ADDR
|
||||
* CONTENT_TYPE
|
||||
* QUERY_STRING
|
||||
* REQUEST_METHOD
|
||||
|
||||
The backend process sets GIT_COMMITTER_NAME to '$REMOTE_USER' and
|
||||
GIT_COMMITTER_EMAIL to '$\{REMOTE_USER}@http.$\{REMOTE_ADDR\}',
|
||||
ensuring that any reflogs created by 'git-receive-pack' contain some
|
||||
identifying information of the remote user who performed the push.
|
||||
|
||||
All CGI environment variables are available to each of the hooks
|
||||
invoked by the 'git-receive-pack'.
|
||||
|
||||
Author
|
||||
------
|
||||
Written by Shawn O. Pearce <spearce@spearce.org>.
|
||||
|
||||
Documentation
|
||||
--------------
|
||||
Documentation by Shawn O. Pearce <spearce@spearce.org>.
|
||||
|
||||
GIT
|
||||
---
|
||||
Part of the linkgit:git[1] suite
|
|
@ -34,15 +34,51 @@ Commands are given by the caller on the helper's standard input, one per line.
|
|||
value of the ref. A space-separated list of attributes follows
|
||||
the name; unrecognized attributes are ignored. After the
|
||||
complete list, outputs a blank line.
|
||||
+
|
||||
If 'push' is supported this may be called as 'list for-push'
|
||||
to obtain the current refs prior to sending one or more 'push'
|
||||
commands to the helper.
|
||||
|
||||
'option' <name> <value>::
|
||||
Set the transport helper option <name> to <value>. Outputs a
|
||||
single line containing one of 'ok' (option successfully set),
|
||||
'unsupported' (option not recognized) or 'error <msg>'
|
||||
(option <name> is supported but <value> is not correct
|
||||
for it). Options should be set before other commands,
|
||||
and may how those commands behave.
|
||||
+
|
||||
Supported if the helper has the "option" capability.
|
||||
|
||||
'fetch' <sha1> <name>::
|
||||
Fetches the given object, writing the necessary objects to the
|
||||
database. Outputs a blank line when the fetch is
|
||||
complete. Only objects which were reported in the ref list
|
||||
with a sha1 may be fetched this way.
|
||||
Fetches the given object, writing the necessary objects
|
||||
to the database. Fetch commands are sent in a batch, one
|
||||
per line, and the batch is terminated with a blank line.
|
||||
Outputs a single blank line when all fetch commands in the
|
||||
same batch are complete. Only objects which were reported
|
||||
in the ref list with a sha1 may be fetched this way.
|
||||
+
|
||||
Optionally may output a 'lock <file>' line indicating a file under
|
||||
GIT_DIR/objects/pack which is keeping a pack until refs can be
|
||||
suitably updated.
|
||||
+
|
||||
Supported if the helper has the "fetch" capability.
|
||||
|
||||
'push' +<src>:<dst>::
|
||||
Pushes the given <src> commit or branch locally to the
|
||||
remote branch described by <dst>. A batch sequence of
|
||||
one or more push commands is terminated with a blank line.
|
||||
+
|
||||
Zero or more protocol options may be entered after the last 'push'
|
||||
command, before the batch's terminating blank line.
|
||||
+
|
||||
When the push is complete, outputs one or more 'ok <dst>' or
|
||||
'error <dst> <why>?' lines to indicate success or failure of
|
||||
each pushed ref. The status report output is terminated by
|
||||
a blank line. The option field <why> may be quoted in a C
|
||||
style string if it contains an LF.
|
||||
+
|
||||
Supported if the helper has the "push" capability.
|
||||
|
||||
If a fatal error occurs, the program writes the error message to
|
||||
stderr and exits. The caller should expect that a suitable error
|
||||
message has been printed if the child closes the connection without
|
||||
|
@ -57,10 +93,49 @@ CAPABILITIES
|
|||
'fetch'::
|
||||
This helper supports the 'fetch' command.
|
||||
|
||||
'option'::
|
||||
This helper supports the option command.
|
||||
|
||||
'push'::
|
||||
This helper supports the 'push' command.
|
||||
|
||||
REF LIST ATTRIBUTES
|
||||
-------------------
|
||||
|
||||
None are defined yet, but the caller must accept any which are supplied.
|
||||
'for-push'::
|
||||
The caller wants to use the ref list to prepare push
|
||||
commands. A helper might chose to acquire the ref list by
|
||||
opening a different type of connection to the destination.
|
||||
|
||||
OPTIONS
|
||||
-------
|
||||
'option verbosity' <N>::
|
||||
Change the level of messages displayed by the helper.
|
||||
When N is 0 the end-user has asked the process to be
|
||||
quiet, and the helper should produce only error output.
|
||||
N of 1 is the default level of verbosity, higher values
|
||||
of N correspond to the number of -v flags passed on the
|
||||
command line.
|
||||
|
||||
'option progress' \{'true'|'false'\}::
|
||||
Enable (or disable) progress messages displayed by the
|
||||
transport helper during a command.
|
||||
|
||||
'option depth' <depth>::
|
||||
Deepen the history of a shallow repository.
|
||||
|
||||
'option followtags' \{'true'|'false'\}::
|
||||
If enabled the helper should automatically fetch annotated
|
||||
tag objects if the object the tag points at was transferred
|
||||
during the fetch command. If the tag is not fetched by
|
||||
the helper a second fetch command will usually be sent to
|
||||
ask for the tag specifically. Some helpers may be able to
|
||||
use this option to avoid a second network connection.
|
||||
|
||||
'option dry-run' \{'true'|'false'\}:
|
||||
If true, pretend the operation completed successfully,
|
||||
but don't actually change any repository data. For most
|
||||
helpers this only applies to the 'push', if supported.
|
||||
|
||||
Documentation
|
||||
-------------
|
||||
|
|
1
Makefile
1
Makefile
|
@ -388,6 +388,7 @@ PROGRAMS += git-show-index$X
|
|||
PROGRAMS += git-unpack-file$X
|
||||
PROGRAMS += git-upload-pack$X
|
||||
PROGRAMS += git-var$X
|
||||
PROGRAMS += git-http-backend$X
|
||||
|
||||
# List built-in command $C whose implementation cmd_$C() is not in
|
||||
# builtin-$C.o but is linked in as part of some other command.
|
||||
|
|
|
@ -157,6 +157,66 @@ static const unsigned char *get_rev(void)
|
|||
return commit->object.sha1;
|
||||
}
|
||||
|
||||
enum ack_type {
|
||||
NAK = 0,
|
||||
ACK,
|
||||
ACK_continue,
|
||||
ACK_common,
|
||||
ACK_ready
|
||||
};
|
||||
|
||||
static void consume_shallow_list(int fd)
|
||||
{
|
||||
if (args.stateless_rpc && args.depth > 0) {
|
||||
/* If we sent a depth we will get back "duplicate"
|
||||
* shallow and unshallow commands every time there
|
||||
* is a block of have lines exchanged.
|
||||
*/
|
||||
char line[1000];
|
||||
while (packet_read_line(fd, line, sizeof(line))) {
|
||||
if (!prefixcmp(line, "shallow "))
|
||||
continue;
|
||||
if (!prefixcmp(line, "unshallow "))
|
||||
continue;
|
||||
die("git fetch-pack: expected shallow list");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static enum ack_type get_ack(int fd, unsigned char *result_sha1)
|
||||
{
|
||||
static char line[1000];
|
||||
int len = packet_read_line(fd, line, sizeof(line));
|
||||
|
||||
if (!len)
|
||||
die("git fetch-pack: expected ACK/NAK, got EOF");
|
||||
if (line[len-1] == '\n')
|
||||
line[--len] = 0;
|
||||
if (!strcmp(line, "NAK"))
|
||||
return NAK;
|
||||
if (!prefixcmp(line, "ACK ")) {
|
||||
if (!get_sha1_hex(line+4, result_sha1)) {
|
||||
if (strstr(line+45, "continue"))
|
||||
return ACK_continue;
|
||||
if (strstr(line+45, "common"))
|
||||
return ACK_common;
|
||||
if (strstr(line+45, "ready"))
|
||||
return ACK_ready;
|
||||
return ACK;
|
||||
}
|
||||
}
|
||||
die("git fetch_pack: expected ACK/NAK, got '%s'", line);
|
||||
}
|
||||
|
||||
static void send_request(int fd, struct strbuf *buf)
|
||||
{
|
||||
if (args.stateless_rpc) {
|
||||
send_sideband(fd, -1, buf->buf, buf->len, LARGE_PACKET_MAX);
|
||||
packet_flush(fd);
|
||||
} else
|
||||
safe_write(fd, buf->buf, buf->len);
|
||||
}
|
||||
|
||||
static int find_common(int fd[2], unsigned char *result_sha1,
|
||||
struct ref *refs)
|
||||
{
|
||||
|
@ -165,7 +225,11 @@ static int find_common(int fd[2], unsigned char *result_sha1,
|
|||
const unsigned char *sha1;
|
||||
unsigned in_vain = 0;
|
||||
int got_continue = 0;
|
||||
struct strbuf req_buf = STRBUF_INIT;
|
||||
size_t state_len = 0;
|
||||
|
||||
if (args.stateless_rpc && multi_ack == 1)
|
||||
die("--stateless-rpc requires multi_ack_detailed");
|
||||
if (marked)
|
||||
for_each_ref(clear_marks, NULL);
|
||||
marked = 1;
|
||||
|
@ -175,6 +239,7 @@ static int find_common(int fd[2], unsigned char *result_sha1,
|
|||
fetching = 0;
|
||||
for ( ; refs ; refs = refs->next) {
|
||||
unsigned char *remote = refs->old_sha1;
|
||||
const char *remote_hex;
|
||||
struct object *o;
|
||||
|
||||
/*
|
||||
|
@ -192,32 +257,42 @@ static int find_common(int fd[2], unsigned char *result_sha1,
|
|||
continue;
|
||||
}
|
||||
|
||||
if (!fetching)
|
||||
packet_write(fd[1], "want %s%s%s%s%s%s%s%s\n",
|
||||
sha1_to_hex(remote),
|
||||
(multi_ack ? " multi_ack" : ""),
|
||||
(use_sideband == 2 ? " side-band-64k" : ""),
|
||||
(use_sideband == 1 ? " side-band" : ""),
|
||||
(args.use_thin_pack ? " thin-pack" : ""),
|
||||
(args.no_progress ? " no-progress" : ""),
|
||||
(args.include_tag ? " include-tag" : ""),
|
||||
(prefer_ofs_delta ? " ofs-delta" : ""));
|
||||
else
|
||||
packet_write(fd[1], "want %s\n", sha1_to_hex(remote));
|
||||
remote_hex = sha1_to_hex(remote);
|
||||
if (!fetching) {
|
||||
struct strbuf c = STRBUF_INIT;
|
||||
if (multi_ack == 2) strbuf_addstr(&c, " multi_ack_detailed");
|
||||
if (multi_ack == 1) strbuf_addstr(&c, " multi_ack");
|
||||
if (use_sideband == 2) strbuf_addstr(&c, " side-band-64k");
|
||||
if (use_sideband == 1) strbuf_addstr(&c, " side-band");
|
||||
if (args.use_thin_pack) strbuf_addstr(&c, " thin-pack");
|
||||
if (args.no_progress) strbuf_addstr(&c, " no-progress");
|
||||
if (args.include_tag) strbuf_addstr(&c, " include-tag");
|
||||
if (prefer_ofs_delta) strbuf_addstr(&c, " ofs-delta");
|
||||
packet_buf_write(&req_buf, "want %s%s\n", remote_hex, c.buf);
|
||||
strbuf_release(&c);
|
||||
} else
|
||||
packet_buf_write(&req_buf, "want %s\n", remote_hex);
|
||||
fetching++;
|
||||
}
|
||||
if (is_repository_shallow())
|
||||
write_shallow_commits(fd[1], 1);
|
||||
if (args.depth > 0)
|
||||
packet_write(fd[1], "deepen %d", args.depth);
|
||||
packet_flush(fd[1]);
|
||||
if (!fetching)
|
||||
|
||||
if (!fetching) {
|
||||
strbuf_release(&req_buf);
|
||||
packet_flush(fd[1]);
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (is_repository_shallow())
|
||||
write_shallow_commits(&req_buf, 1);
|
||||
if (args.depth > 0)
|
||||
packet_buf_write(&req_buf, "deepen %d", args.depth);
|
||||
packet_buf_flush(&req_buf);
|
||||
state_len = req_buf.len;
|
||||
|
||||
if (args.depth > 0) {
|
||||
char line[1024];
|
||||
unsigned char sha1[20];
|
||||
|
||||
send_request(fd[1], &req_buf);
|
||||
while (packet_read_line(fd[0], line, sizeof(line))) {
|
||||
if (!prefixcmp(line, "shallow ")) {
|
||||
if (get_sha1_hex(line + 8, sha1))
|
||||
|
@ -239,45 +314,73 @@ static int find_common(int fd[2], unsigned char *result_sha1,
|
|||
}
|
||||
die("expected shallow/unshallow, got %s", line);
|
||||
}
|
||||
} else if (!args.stateless_rpc)
|
||||
send_request(fd[1], &req_buf);
|
||||
|
||||
if (!args.stateless_rpc) {
|
||||
/* If we aren't using the stateless-rpc interface
|
||||
* we don't need to retain the headers.
|
||||
*/
|
||||
strbuf_setlen(&req_buf, 0);
|
||||
state_len = 0;
|
||||
}
|
||||
|
||||
flushes = 0;
|
||||
retval = -1;
|
||||
while ((sha1 = get_rev())) {
|
||||
packet_write(fd[1], "have %s\n", sha1_to_hex(sha1));
|
||||
packet_buf_write(&req_buf, "have %s\n", sha1_to_hex(sha1));
|
||||
if (args.verbose)
|
||||
fprintf(stderr, "have %s\n", sha1_to_hex(sha1));
|
||||
in_vain++;
|
||||
if (!(31 & ++count)) {
|
||||
int ack;
|
||||
|
||||
packet_flush(fd[1]);
|
||||
packet_buf_flush(&req_buf);
|
||||
send_request(fd[1], &req_buf);
|
||||
strbuf_setlen(&req_buf, state_len);
|
||||
flushes++;
|
||||
|
||||
/*
|
||||
* We keep one window "ahead" of the other side, and
|
||||
* will wait for an ACK only on the next one
|
||||
*/
|
||||
if (count == 32)
|
||||
if (!args.stateless_rpc && count == 32)
|
||||
continue;
|
||||
|
||||
consume_shallow_list(fd[0]);
|
||||
do {
|
||||
ack = get_ack(fd[0], result_sha1);
|
||||
if (args.verbose && ack)
|
||||
fprintf(stderr, "got ack %d %s\n", ack,
|
||||
sha1_to_hex(result_sha1));
|
||||
if (ack == 1) {
|
||||
switch (ack) {
|
||||
case ACK:
|
||||
flushes = 0;
|
||||
multi_ack = 0;
|
||||
retval = 0;
|
||||
goto done;
|
||||
} else if (ack == 2) {
|
||||
case ACK_common:
|
||||
case ACK_ready:
|
||||
case ACK_continue: {
|
||||
struct commit *commit =
|
||||
lookup_commit(result_sha1);
|
||||
if (args.stateless_rpc
|
||||
&& ack == ACK_common
|
||||
&& !(commit->object.flags & COMMON)) {
|
||||
/* We need to replay the have for this object
|
||||
* on the next RPC request so the peer knows
|
||||
* it is in common with us.
|
||||
*/
|
||||
const char *hex = sha1_to_hex(result_sha1);
|
||||
packet_buf_write(&req_buf, "have %s\n", hex);
|
||||
state_len = req_buf.len;
|
||||
}
|
||||
mark_common(commit, 0, 1);
|
||||
retval = 0;
|
||||
in_vain = 0;
|
||||
got_continue = 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
} while (ack);
|
||||
flushes--;
|
||||
|
@ -289,20 +392,24 @@ static int find_common(int fd[2], unsigned char *result_sha1,
|
|||
}
|
||||
}
|
||||
done:
|
||||
packet_write(fd[1], "done\n");
|
||||
packet_buf_write(&req_buf, "done\n");
|
||||
send_request(fd[1], &req_buf);
|
||||
if (args.verbose)
|
||||
fprintf(stderr, "done\n");
|
||||
if (retval != 0) {
|
||||
multi_ack = 0;
|
||||
flushes++;
|
||||
}
|
||||
strbuf_release(&req_buf);
|
||||
|
||||
consume_shallow_list(fd[0]);
|
||||
while (flushes || multi_ack) {
|
||||
int ack = get_ack(fd[0], result_sha1);
|
||||
if (ack) {
|
||||
if (args.verbose)
|
||||
fprintf(stderr, "got ack (%d) %s\n", ack,
|
||||
sha1_to_hex(result_sha1));
|
||||
if (ack == 1)
|
||||
if (ack == ACK)
|
||||
return 0;
|
||||
multi_ack = 1;
|
||||
continue;
|
||||
|
@ -584,7 +691,12 @@ static struct ref *do_fetch_pack(int fd[2],
|
|||
|
||||
if (is_repository_shallow() && !server_supports("shallow"))
|
||||
die("Server does not support shallow clients");
|
||||
if (server_supports("multi_ack")) {
|
||||
if (server_supports("multi_ack_detailed")) {
|
||||
if (args.verbose)
|
||||
fprintf(stderr, "Server supports multi_ack_detailed\n");
|
||||
multi_ack = 2;
|
||||
}
|
||||
else if (server_supports("multi_ack")) {
|
||||
if (args.verbose)
|
||||
fprintf(stderr, "Server supports multi_ack\n");
|
||||
multi_ack = 1;
|
||||
|
@ -615,6 +727,8 @@ static struct ref *do_fetch_pack(int fd[2],
|
|||
*/
|
||||
warning("no common commits");
|
||||
|
||||
if (args.stateless_rpc)
|
||||
packet_flush(fd[1]);
|
||||
if (get_pack(fd, pack_lockfile))
|
||||
die("git fetch-pack: fetch failed.");
|
||||
|
||||
|
@ -685,6 +799,8 @@ int cmd_fetch_pack(int argc, const char **argv, const char *prefix)
|
|||
struct ref *ref = NULL;
|
||||
char *dest = NULL, **heads;
|
||||
int fd[2];
|
||||
char *pack_lockfile = NULL;
|
||||
char **pack_lockfile_ptr = NULL;
|
||||
struct child_process *conn;
|
||||
|
||||
nr_heads = 0;
|
||||
|
@ -734,6 +850,15 @@ int cmd_fetch_pack(int argc, const char **argv, const char *prefix)
|
|||
args.no_progress = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp("--stateless-rpc", arg)) {
|
||||
args.stateless_rpc = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp("--lock-pack", arg)) {
|
||||
args.lock_pack = 1;
|
||||
pack_lockfile_ptr = &pack_lockfile;
|
||||
continue;
|
||||
}
|
||||
usage(fetch_pack_usage);
|
||||
}
|
||||
dest = (char *)arg;
|
||||
|
@ -744,19 +869,27 @@ int cmd_fetch_pack(int argc, const char **argv, const char *prefix)
|
|||
if (!dest)
|
||||
usage(fetch_pack_usage);
|
||||
|
||||
conn = git_connect(fd, (char *)dest, args.uploadpack,
|
||||
args.verbose ? CONNECT_VERBOSE : 0);
|
||||
if (conn) {
|
||||
get_remote_heads(fd[0], &ref, 0, NULL, 0, NULL);
|
||||
|
||||
ref = fetch_pack(&args, fd, conn, ref, dest, nr_heads, heads, NULL);
|
||||
close(fd[0]);
|
||||
close(fd[1]);
|
||||
if (finish_connect(conn))
|
||||
ref = NULL;
|
||||
if (args.stateless_rpc) {
|
||||
conn = NULL;
|
||||
fd[0] = 0;
|
||||
fd[1] = 1;
|
||||
} else {
|
||||
ref = NULL;
|
||||
conn = git_connect(fd, (char *)dest, args.uploadpack,
|
||||
args.verbose ? CONNECT_VERBOSE : 0);
|
||||
}
|
||||
|
||||
get_remote_heads(fd[0], &ref, 0, NULL, 0, NULL);
|
||||
|
||||
ref = fetch_pack(&args, fd, conn, ref, dest,
|
||||
nr_heads, heads, pack_lockfile_ptr);
|
||||
if (pack_lockfile) {
|
||||
printf("lock %s\n", pack_lockfile);
|
||||
fflush(stdout);
|
||||
}
|
||||
close(fd[0]);
|
||||
close(fd[1]);
|
||||
if (finish_connect(conn))
|
||||
ref = NULL;
|
||||
ret = !ref;
|
||||
|
||||
if (!ret && nr_heads) {
|
||||
|
@ -809,6 +942,7 @@ struct ref *fetch_pack(struct fetch_pack_args *my_args,
|
|||
|
||||
if (args.depth > 0) {
|
||||
struct cache_time mtime;
|
||||
struct strbuf sb = STRBUF_INIT;
|
||||
char *shallow = git_path("shallow");
|
||||
int fd;
|
||||
|
||||
|
@ -826,12 +960,14 @@ struct ref *fetch_pack(struct fetch_pack_args *my_args,
|
|||
|
||||
fd = hold_lock_file_for_update(&lock, shallow,
|
||||
LOCK_DIE_ON_ERROR);
|
||||
if (!write_shallow_commits(fd, 0)) {
|
||||
if (!write_shallow_commits(&sb, 0)
|
||||
|| write_in_full(fd, sb.buf, sb.len) != sb.len) {
|
||||
unlink_or_warn(shallow);
|
||||
rollback_lock_file(&lock);
|
||||
} else {
|
||||
commit_lock_file(&lock);
|
||||
}
|
||||
strbuf_release(&sb);
|
||||
}
|
||||
|
||||
reprepare_packed_git();
|
||||
|
|
|
@ -717,7 +717,7 @@ int cmd_fetch(int argc, const char **argv, const char *prefix)
|
|||
|
||||
transport = transport_get(remote, remote->url[0]);
|
||||
if (verbosity >= 2)
|
||||
transport->verbose = 1;
|
||||
transport->verbose = verbosity <= 3 ? verbosity : 3;
|
||||
if (verbosity < 0)
|
||||
transport->verbose = -1;
|
||||
if (upload_pack)
|
||||
|
|
|
@ -627,6 +627,8 @@ static void add_alternate_refs(void)
|
|||
|
||||
int cmd_receive_pack(int argc, const char **argv, const char *prefix)
|
||||
{
|
||||
int advertise_refs = 0;
|
||||
int stateless_rpc = 0;
|
||||
int i;
|
||||
char *dir = NULL;
|
||||
|
||||
|
@ -635,7 +637,15 @@ int cmd_receive_pack(int argc, const char **argv, const char *prefix)
|
|||
const char *arg = *argv++;
|
||||
|
||||
if (*arg == '-') {
|
||||
/* Do flag handling here */
|
||||
if (!strcmp(arg, "--advertise-refs")) {
|
||||
advertise_refs = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--stateless-rpc")) {
|
||||
stateless_rpc = 1;
|
||||
continue;
|
||||
}
|
||||
|
||||
usage(receive_pack_usage);
|
||||
}
|
||||
if (dir)
|
||||
|
@ -664,12 +674,16 @@ int cmd_receive_pack(int argc, const char **argv, const char *prefix)
|
|||
" report-status delete-refs ofs-delta " :
|
||||
" report-status delete-refs ";
|
||||
|
||||
add_alternate_refs();
|
||||
write_head_info();
|
||||
clear_extra_refs();
|
||||
if (advertise_refs || !stateless_rpc) {
|
||||
add_alternate_refs();
|
||||
write_head_info();
|
||||
clear_extra_refs();
|
||||
|
||||
/* EOF */
|
||||
packet_flush(1);
|
||||
/* EOF */
|
||||
packet_flush(1);
|
||||
}
|
||||
if (advertise_refs)
|
||||
return 0;
|
||||
|
||||
read_head_info();
|
||||
if (commands) {
|
||||
|
|
|
@ -2,9 +2,11 @@
|
|||
#include "commit.h"
|
||||
#include "refs.h"
|
||||
#include "pkt-line.h"
|
||||
#include "sideband.h"
|
||||
#include "run-command.h"
|
||||
#include "remote.h"
|
||||
#include "send-pack.h"
|
||||
#include "quote.h"
|
||||
|
||||
static const char send_pack_usage[] =
|
||||
"git send-pack [--all | --mirror] [--dry-run] [--force] [--receive-pack=<git-receive-pack>] [--verbose] [--thin] [<host>:]<directory> [<ref>...]\n"
|
||||
|
@ -59,7 +61,7 @@ static int pack_objects(int fd, struct ref *refs, struct extra_have_objects *ext
|
|||
memset(&po, 0, sizeof(po));
|
||||
po.argv = argv;
|
||||
po.in = -1;
|
||||
po.out = fd;
|
||||
po.out = args->stateless_rpc ? -1 : fd;
|
||||
po.git_cmd = 1;
|
||||
if (start_command(&po))
|
||||
die_errno("git pack-objects failed");
|
||||
|
@ -83,6 +85,20 @@ static int pack_objects(int fd, struct ref *refs, struct extra_have_objects *ext
|
|||
}
|
||||
|
||||
close(po.in);
|
||||
|
||||
if (args->stateless_rpc) {
|
||||
char *buf = xmalloc(LARGE_PACKET_MAX);
|
||||
while (1) {
|
||||
ssize_t n = xread(po.out, buf, LARGE_PACKET_MAX);
|
||||
if (n <= 0)
|
||||
break;
|
||||
send_sideband(fd, -1, buf, n, LARGE_PACKET_MAX);
|
||||
}
|
||||
free(buf);
|
||||
close(po.out);
|
||||
po.out = -1;
|
||||
}
|
||||
|
||||
if (finish_command(&po))
|
||||
return error("pack-objects died with strange error");
|
||||
return 0;
|
||||
|
@ -303,6 +319,59 @@ static int refs_pushed(struct ref *ref)
|
|||
return 0;
|
||||
}
|
||||
|
||||
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;
|
||||
|
||||
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');
|
||||
|
||||
safe_write(1, buf.buf, buf.len);
|
||||
}
|
||||
strbuf_release(&buf);
|
||||
}
|
||||
|
||||
int send_pack(struct send_pack_args *args,
|
||||
int fd[], struct child_process *conn,
|
||||
struct ref *remote_refs,
|
||||
|
@ -310,6 +379,7 @@ int send_pack(struct send_pack_args *args,
|
|||
{
|
||||
int in = fd[0];
|
||||
int out = fd[1];
|
||||
struct strbuf req_buf = STRBUF_INIT;
|
||||
struct ref *ref;
|
||||
int new_refs;
|
||||
int ask_for_status_report = 0;
|
||||
|
@ -391,14 +461,14 @@ int send_pack(struct send_pack_args *args,
|
|||
char *new_hex = sha1_to_hex(ref->new_sha1);
|
||||
|
||||
if (ask_for_status_report) {
|
||||
packet_write(out, "%s %s %s%c%s",
|
||||
packet_buf_write(&req_buf, "%s %s %s%c%s",
|
||||
old_hex, new_hex, ref->name, 0,
|
||||
"report-status");
|
||||
ask_for_status_report = 0;
|
||||
expect_status_report = 1;
|
||||
}
|
||||
else
|
||||
packet_write(out, "%s %s %s",
|
||||
packet_buf_write(&req_buf, "%s %s %s",
|
||||
old_hex, new_hex, ref->name);
|
||||
}
|
||||
ref->status = expect_status_report ?
|
||||
|
@ -406,7 +476,17 @@ int send_pack(struct send_pack_args *args,
|
|||
REF_STATUS_OK;
|
||||
}
|
||||
|
||||
packet_flush(out);
|
||||
if (args->stateless_rpc) {
|
||||
if (!args->dry_run) {
|
||||
packet_buf_flush(&req_buf);
|
||||
send_sideband(out, -1, req_buf.buf, req_buf.len, LARGE_PACKET_MAX);
|
||||
}
|
||||
} else {
|
||||
safe_write(out, req_buf.buf, req_buf.len);
|
||||
packet_flush(out);
|
||||
}
|
||||
strbuf_release(&req_buf);
|
||||
|
||||
if (new_refs && !args->dry_run) {
|
||||
if (pack_objects(out, remote_refs, extra_have, args) < 0) {
|
||||
for (ref = remote_refs; ref; ref = ref->next)
|
||||
|
@ -414,11 +494,15 @@ int send_pack(struct send_pack_args *args,
|
|||
return -1;
|
||||
}
|
||||
}
|
||||
if (args->stateless_rpc && !args->dry_run)
|
||||
packet_flush(out);
|
||||
|
||||
if (expect_status_report)
|
||||
ret = receive_status(in, remote_refs);
|
||||
else
|
||||
ret = 0;
|
||||
if (args->stateless_rpc)
|
||||
packet_flush(out);
|
||||
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
|
@ -478,6 +562,7 @@ int cmd_send_pack(int argc, const char **argv, const char *prefix)
|
|||
struct extra_have_objects extra_have;
|
||||
struct ref *remote_refs, *local_refs;
|
||||
int ret;
|
||||
int helper_status = 0;
|
||||
int send_all = 0;
|
||||
const char *receivepack = "git-receive-pack";
|
||||
int flags;
|
||||
|
@ -523,6 +608,14 @@ int cmd_send_pack(int argc, const char **argv, const char *prefix)
|
|||
args.use_thin_pack = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--stateless-rpc")) {
|
||||
args.stateless_rpc = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--helper-status")) {
|
||||
helper_status = 1;
|
||||
continue;
|
||||
}
|
||||
usage(send_pack_usage);
|
||||
}
|
||||
if (!dest) {
|
||||
|
@ -551,7 +644,14 @@ int cmd_send_pack(int argc, const char **argv, const char *prefix)
|
|||
}
|
||||
}
|
||||
|
||||
conn = git_connect(fd, dest, receivepack, args.verbose ? CONNECT_VERBOSE : 0);
|
||||
if (args.stateless_rpc) {
|
||||
conn = NULL;
|
||||
fd[0] = 0;
|
||||
fd[1] = 1;
|
||||
} else {
|
||||
conn = git_connect(fd, dest, receivepack,
|
||||
args.verbose ? CONNECT_VERBOSE : 0);
|
||||
}
|
||||
|
||||
memset(&extra_have, 0, sizeof(extra_have));
|
||||
|
||||
|
@ -575,12 +675,16 @@ int cmd_send_pack(int argc, const char **argv, const char *prefix)
|
|||
|
||||
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);
|
||||
|
||||
print_push_status(dest, remote_refs);
|
||||
if (!helper_status)
|
||||
print_push_status(dest, remote_refs);
|
||||
|
||||
if (!args.dry_run && remote) {
|
||||
struct ref *ref;
|
||||
|
|
2
cache.h
2
cache.h
|
@ -657,6 +657,7 @@ const char *make_relative_path(const char *abs, const char *base);
|
|||
int normalize_path_copy(char *dst, const char *src);
|
||||
int longest_ancestor_length(const char *path, const char *prefix_list);
|
||||
char *strip_path_suffix(const char *path, const char *suffix);
|
||||
int daemon_avoid_alias(const char *path);
|
||||
|
||||
/* Read and unpack a sha1 file into memory, write memory to a sha1 file */
|
||||
extern int sha1_object_info(const unsigned char *, unsigned long *);
|
||||
|
@ -859,7 +860,6 @@ extern struct ref *find_ref_by_name(const struct ref *list, const char *name);
|
|||
extern struct child_process *git_connect(int fd[2], const char *url, const char *prog, int flags);
|
||||
extern int finish_connect(struct child_process *conn);
|
||||
extern int path_match(const char *path, int nr, char **match);
|
||||
extern int get_ack(int fd, unsigned char *result_sha1);
|
||||
struct extra_have_objects {
|
||||
int nr, alloc;
|
||||
unsigned char (*array)[20];
|
||||
|
|
10
commit.c
10
commit.c
|
@ -199,7 +199,7 @@ struct commit_graft *lookup_commit_graft(const unsigned char *sha1)
|
|||
return commit_graft[pos];
|
||||
}
|
||||
|
||||
int write_shallow_commits(int fd, int use_pack_protocol)
|
||||
int write_shallow_commits(struct strbuf *out, int use_pack_protocol)
|
||||
{
|
||||
int i, count = 0;
|
||||
for (i = 0; i < commit_graft_nr; i++)
|
||||
|
@ -208,12 +208,10 @@ int write_shallow_commits(int fd, int use_pack_protocol)
|
|||
sha1_to_hex(commit_graft[i]->sha1);
|
||||
count++;
|
||||
if (use_pack_protocol)
|
||||
packet_write(fd, "shallow %s", hex);
|
||||
packet_buf_write(out, "shallow %s", hex);
|
||||
else {
|
||||
if (write_in_full(fd, hex, 40) != 40)
|
||||
break;
|
||||
if (write_str_in_full(fd, "\n") != 1)
|
||||
break;
|
||||
strbuf_addstr(out, hex);
|
||||
strbuf_addch(out, '\n');
|
||||
}
|
||||
}
|
||||
return count;
|
||||
|
|
2
commit.h
2
commit.h
|
@ -139,7 +139,7 @@ extern struct commit_list *get_octopus_merge_bases(struct commit_list *in);
|
|||
|
||||
extern int register_shallow(const unsigned char *sha1);
|
||||
extern int unregister_shallow(const unsigned char *sha1);
|
||||
extern int write_shallow_commits(int fd, int use_pack_protocol);
|
||||
extern int write_shallow_commits(struct strbuf *out, int use_pack_protocol);
|
||||
extern int is_repository_shallow(void);
|
||||
extern struct commit_list *get_shallow_commits(struct object_array *heads,
|
||||
int depth, int shallow_flag, int not_shallow_flag);
|
||||
|
|
21
connect.c
21
connect.c
|
@ -107,27 +107,6 @@ int server_supports(const char *feature)
|
|||
strstr(server_capabilities, feature) != NULL;
|
||||
}
|
||||
|
||||
int get_ack(int fd, unsigned char *result_sha1)
|
||||
{
|
||||
static char line[1000];
|
||||
int len = packet_read_line(fd, line, sizeof(line));
|
||||
|
||||
if (!len)
|
||||
die("git fetch-pack: expected ACK/NAK, got EOF");
|
||||
if (line[len-1] == '\n')
|
||||
line[--len] = 0;
|
||||
if (!strcmp(line, "NAK"))
|
||||
return 0;
|
||||
if (!prefixcmp(line, "ACK ")) {
|
||||
if (!get_sha1_hex(line+4, result_sha1)) {
|
||||
if (strstr(line+45, "continue"))
|
||||
return 2;
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
die("git fetch_pack: expected ACK/NAK, got '%s'", line);
|
||||
}
|
||||
|
||||
int path_match(const char *path, int nr, char **match)
|
||||
{
|
||||
int i;
|
||||
|
|
49
daemon.c
49
daemon.c
|
@ -101,53 +101,6 @@ static void NORETURN daemon_die(const char *err, va_list params)
|
|||
exit(1);
|
||||
}
|
||||
|
||||
static int avoid_alias(char *p)
|
||||
{
|
||||
int sl, ndot;
|
||||
|
||||
/*
|
||||
* This resurrects the belts and suspenders paranoia check by HPA
|
||||
* done in <435560F7.4080006@zytor.com> thread, now enter_repo()
|
||||
* does not do getcwd() based path canonicalizations.
|
||||
*
|
||||
* sl becomes true immediately after seeing '/' and continues to
|
||||
* be true as long as dots continue after that without intervening
|
||||
* non-dot character.
|
||||
*/
|
||||
if (!p || (*p != '/' && *p != '~'))
|
||||
return -1;
|
||||
sl = 1; ndot = 0;
|
||||
p++;
|
||||
|
||||
while (1) {
|
||||
char ch = *p++;
|
||||
if (sl) {
|
||||
if (ch == '.')
|
||||
ndot++;
|
||||
else if (ch == '/') {
|
||||
if (ndot < 3)
|
||||
/* reject //, /./ and /../ */
|
||||
return -1;
|
||||
ndot = 0;
|
||||
}
|
||||
else if (ch == 0) {
|
||||
if (0 < ndot && ndot < 3)
|
||||
/* reject /.$ and /..$ */
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
else
|
||||
sl = ndot = 0;
|
||||
}
|
||||
else if (ch == 0)
|
||||
return 0;
|
||||
else if (ch == '/') {
|
||||
sl = 1;
|
||||
ndot = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static char *path_ok(char *directory)
|
||||
{
|
||||
static char rpath[PATH_MAX];
|
||||
|
@ -157,7 +110,7 @@ static char *path_ok(char *directory)
|
|||
|
||||
dir = directory;
|
||||
|
||||
if (avoid_alias(dir)) {
|
||||
if (daemon_avoid_alias(dir)) {
|
||||
logerror("'%s': aliased", dir);
|
||||
return NULL;
|
||||
}
|
||||
|
|
|
@ -13,7 +13,8 @@ struct fetch_pack_args
|
|||
fetch_all:1,
|
||||
verbose:1,
|
||||
no_progress:1,
|
||||
include_tag:1;
|
||||
include_tag:1,
|
||||
stateless_rpc:1;
|
||||
};
|
||||
|
||||
struct ref *fetch_pack(struct fetch_pack_args *args,
|
||||
|
|
|
@ -0,0 +1,655 @@
|
|||
#include "cache.h"
|
||||
#include "refs.h"
|
||||
#include "pkt-line.h"
|
||||
#include "object.h"
|
||||
#include "tag.h"
|
||||
#include "exec_cmd.h"
|
||||
#include "run-command.h"
|
||||
#include "string-list.h"
|
||||
|
||||
static const char content_type[] = "Content-Type";
|
||||
static const char content_length[] = "Content-Length";
|
||||
static const char last_modified[] = "Last-Modified";
|
||||
static int getanyfile = 1;
|
||||
|
||||
static struct string_list *query_params;
|
||||
|
||||
struct rpc_service {
|
||||
const char *name;
|
||||
const char *config_name;
|
||||
signed enabled : 2;
|
||||
};
|
||||
|
||||
static struct rpc_service rpc_service[] = {
|
||||
{ "upload-pack", "uploadpack", 1 },
|
||||
{ "receive-pack", "receivepack", -1 },
|
||||
};
|
||||
|
||||
static int decode_char(const char *q)
|
||||
{
|
||||
int i;
|
||||
unsigned char val = 0;
|
||||
for (i = 0; i < 2; i++) {
|
||||
unsigned char c = *q++;
|
||||
val <<= 4;
|
||||
if (c >= '0' && c <= '9')
|
||||
val += c - '0';
|
||||
else if (c >= 'a' && c <= 'f')
|
||||
val += c - 'a' + 10;
|
||||
else if (c >= 'A' && c <= 'F')
|
||||
val += c - 'A' + 10;
|
||||
else
|
||||
return -1;
|
||||
}
|
||||
return val;
|
||||
}
|
||||
|
||||
static char *decode_parameter(const char **query, int is_name)
|
||||
{
|
||||
const char *q = *query;
|
||||
struct strbuf out;
|
||||
|
||||
strbuf_init(&out, 16);
|
||||
do {
|
||||
unsigned char c = *q;
|
||||
|
||||
if (!c)
|
||||
break;
|
||||
if (c == '&' || (is_name && c == '=')) {
|
||||
q++;
|
||||
break;
|
||||
}
|
||||
|
||||
if (c == '%') {
|
||||
int val = decode_char(q + 1);
|
||||
if (0 <= val) {
|
||||
strbuf_addch(&out, val);
|
||||
q += 3;
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
if (c == '+')
|
||||
strbuf_addch(&out, ' ');
|
||||
else
|
||||
strbuf_addch(&out, c);
|
||||
q++;
|
||||
} while (1);
|
||||
*query = q;
|
||||
return strbuf_detach(&out, NULL);
|
||||
}
|
||||
|
||||
static struct string_list *get_parameters(void)
|
||||
{
|
||||
if (!query_params) {
|
||||
const char *query = getenv("QUERY_STRING");
|
||||
|
||||
query_params = xcalloc(1, sizeof(*query_params));
|
||||
while (query && *query) {
|
||||
char *name = decode_parameter(&query, 1);
|
||||
char *value = decode_parameter(&query, 0);
|
||||
struct string_list_item *i;
|
||||
|
||||
i = string_list_lookup(name, query_params);
|
||||
if (!i)
|
||||
i = string_list_insert(name, query_params);
|
||||
else
|
||||
free(i->util);
|
||||
i->util = value;
|
||||
}
|
||||
}
|
||||
return query_params;
|
||||
}
|
||||
|
||||
static const char *get_parameter(const char *name)
|
||||
{
|
||||
struct string_list_item *i;
|
||||
i = string_list_lookup(name, get_parameters());
|
||||
return i ? i->util : NULL;
|
||||
}
|
||||
|
||||
__attribute__((format (printf, 2, 3)))
|
||||
static void format_write(int fd, const char *fmt, ...)
|
||||
{
|
||||
static char buffer[1024];
|
||||
|
||||
va_list args;
|
||||
unsigned n;
|
||||
|
||||
va_start(args, fmt);
|
||||
n = vsnprintf(buffer, sizeof(buffer), fmt, args);
|
||||
va_end(args);
|
||||
if (n >= sizeof(buffer))
|
||||
die("protocol error: impossibly long line");
|
||||
|
||||
safe_write(fd, buffer, n);
|
||||
}
|
||||
|
||||
static void http_status(unsigned code, const char *msg)
|
||||
{
|
||||
format_write(1, "Status: %u %s\r\n", code, msg);
|
||||
}
|
||||
|
||||
static void hdr_str(const char *name, const char *value)
|
||||
{
|
||||
format_write(1, "%s: %s\r\n", name, value);
|
||||
}
|
||||
|
||||
static void hdr_int(const char *name, uintmax_t value)
|
||||
{
|
||||
format_write(1, "%s: %" PRIuMAX "\r\n", name, value);
|
||||
}
|
||||
|
||||
static void hdr_date(const char *name, unsigned long when)
|
||||
{
|
||||
const char *value = show_date(when, 0, DATE_RFC2822);
|
||||
hdr_str(name, value);
|
||||
}
|
||||
|
||||
static void hdr_nocache(void)
|
||||
{
|
||||
hdr_str("Expires", "Fri, 01 Jan 1980 00:00:00 GMT");
|
||||
hdr_str("Pragma", "no-cache");
|
||||
hdr_str("Cache-Control", "no-cache, max-age=0, must-revalidate");
|
||||
}
|
||||
|
||||
static void hdr_cache_forever(void)
|
||||
{
|
||||
unsigned long now = time(NULL);
|
||||
hdr_date("Date", now);
|
||||
hdr_date("Expires", now + 31536000);
|
||||
hdr_str("Cache-Control", "public, max-age=31536000");
|
||||
}
|
||||
|
||||
static void end_headers(void)
|
||||
{
|
||||
safe_write(1, "\r\n", 2);
|
||||
}
|
||||
|
||||
__attribute__((format (printf, 1, 2)))
|
||||
static NORETURN void not_found(const char *err, ...)
|
||||
{
|
||||
va_list params;
|
||||
|
||||
http_status(404, "Not Found");
|
||||
hdr_nocache();
|
||||
end_headers();
|
||||
|
||||
va_start(params, err);
|
||||
if (err && *err)
|
||||
vfprintf(stderr, err, params);
|
||||
va_end(params);
|
||||
exit(0);
|
||||
}
|
||||
|
||||
__attribute__((format (printf, 1, 2)))
|
||||
static NORETURN void forbidden(const char *err, ...)
|
||||
{
|
||||
va_list params;
|
||||
|
||||
http_status(403, "Forbidden");
|
||||
hdr_nocache();
|
||||
end_headers();
|
||||
|
||||
va_start(params, err);
|
||||
if (err && *err)
|
||||
vfprintf(stderr, err, params);
|
||||
va_end(params);
|
||||
exit(0);
|
||||
}
|
||||
|
||||
static void select_getanyfile(void)
|
||||
{
|
||||
if (!getanyfile)
|
||||
forbidden("Unsupported service: getanyfile");
|
||||
}
|
||||
|
||||
static void send_strbuf(const char *type, struct strbuf *buf)
|
||||
{
|
||||
hdr_int(content_length, buf->len);
|
||||
hdr_str(content_type, type);
|
||||
end_headers();
|
||||
safe_write(1, buf->buf, buf->len);
|
||||
}
|
||||
|
||||
static void send_local_file(const char *the_type, const char *name)
|
||||
{
|
||||
const char *p = git_path("%s", name);
|
||||
size_t buf_alloc = 8192;
|
||||
char *buf = xmalloc(buf_alloc);
|
||||
int fd;
|
||||
struct stat sb;
|
||||
|
||||
fd = open(p, O_RDONLY);
|
||||
if (fd < 0)
|
||||
not_found("Cannot open '%s': %s", p, strerror(errno));
|
||||
if (fstat(fd, &sb) < 0)
|
||||
die_errno("Cannot stat '%s'", p);
|
||||
|
||||
hdr_int(content_length, sb.st_size);
|
||||
hdr_str(content_type, the_type);
|
||||
hdr_date(last_modified, sb.st_mtime);
|
||||
end_headers();
|
||||
|
||||
for (;;) {
|
||||
ssize_t n = xread(fd, buf, buf_alloc);
|
||||
if (n < 0)
|
||||
die_errno("Cannot read '%s'", p);
|
||||
if (!n)
|
||||
break;
|
||||
safe_write(1, buf, n);
|
||||
}
|
||||
close(fd);
|
||||
free(buf);
|
||||
}
|
||||
|
||||
static void get_text_file(char *name)
|
||||
{
|
||||
select_getanyfile();
|
||||
hdr_nocache();
|
||||
send_local_file("text/plain", name);
|
||||
}
|
||||
|
||||
static void get_loose_object(char *name)
|
||||
{
|
||||
select_getanyfile();
|
||||
hdr_cache_forever();
|
||||
send_local_file("application/x-git-loose-object", name);
|
||||
}
|
||||
|
||||
static void get_pack_file(char *name)
|
||||
{
|
||||
select_getanyfile();
|
||||
hdr_cache_forever();
|
||||
send_local_file("application/x-git-packed-objects", name);
|
||||
}
|
||||
|
||||
static void get_idx_file(char *name)
|
||||
{
|
||||
select_getanyfile();
|
||||
hdr_cache_forever();
|
||||
send_local_file("application/x-git-packed-objects-toc", name);
|
||||
}
|
||||
|
||||
static int http_config(const char *var, const char *value, void *cb)
|
||||
{
|
||||
if (!strcmp(var, "http.getanyfile")) {
|
||||
getanyfile = git_config_bool(var, value);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (!prefixcmp(var, "http.")) {
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(rpc_service); i++) {
|
||||
struct rpc_service *svc = &rpc_service[i];
|
||||
if (!strcmp(var + 5, svc->config_name)) {
|
||||
svc->enabled = git_config_bool(var, value);
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* we are not interested in parsing any other configuration here */
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct rpc_service *select_service(const char *name)
|
||||
{
|
||||
struct rpc_service *svc = NULL;
|
||||
int i;
|
||||
|
||||
if (prefixcmp(name, "git-"))
|
||||
forbidden("Unsupported service: '%s'", name);
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(rpc_service); i++) {
|
||||
struct rpc_service *s = &rpc_service[i];
|
||||
if (!strcmp(s->name, name + 4)) {
|
||||
svc = s;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!svc)
|
||||
forbidden("Unsupported service: '%s'", name);
|
||||
|
||||
if (svc->enabled < 0) {
|
||||
const char *user = getenv("REMOTE_USER");
|
||||
svc->enabled = (user && *user) ? 1 : 0;
|
||||
}
|
||||
if (!svc->enabled)
|
||||
forbidden("Service not enabled: '%s'", svc->name);
|
||||
return svc;
|
||||
}
|
||||
|
||||
static void inflate_request(const char *prog_name, int out)
|
||||
{
|
||||
z_stream stream;
|
||||
unsigned char in_buf[8192];
|
||||
unsigned char out_buf[8192];
|
||||
unsigned long cnt = 0;
|
||||
int ret;
|
||||
|
||||
memset(&stream, 0, sizeof(stream));
|
||||
ret = inflateInit2(&stream, (15 + 16));
|
||||
if (ret != Z_OK)
|
||||
die("cannot start zlib inflater, zlib err %d", ret);
|
||||
|
||||
while (1) {
|
||||
ssize_t n = xread(0, in_buf, sizeof(in_buf));
|
||||
if (n <= 0)
|
||||
die("request ended in the middle of the gzip stream");
|
||||
|
||||
stream.next_in = in_buf;
|
||||
stream.avail_in = n;
|
||||
|
||||
while (0 < stream.avail_in) {
|
||||
int ret;
|
||||
|
||||
stream.next_out = out_buf;
|
||||
stream.avail_out = sizeof(out_buf);
|
||||
|
||||
ret = inflate(&stream, Z_NO_FLUSH);
|
||||
if (ret != Z_OK && ret != Z_STREAM_END)
|
||||
die("zlib error inflating request, result %d", ret);
|
||||
|
||||
n = stream.total_out - cnt;
|
||||
if (write_in_full(out, out_buf, n) != n)
|
||||
die("%s aborted reading request", prog_name);
|
||||
cnt += n;
|
||||
|
||||
if (ret == Z_STREAM_END)
|
||||
goto done;
|
||||
}
|
||||
}
|
||||
|
||||
done:
|
||||
inflateEnd(&stream);
|
||||
close(out);
|
||||
}
|
||||
|
||||
static void run_service(const char **argv)
|
||||
{
|
||||
const char *encoding = getenv("HTTP_CONTENT_ENCODING");
|
||||
const char *user = getenv("REMOTE_USER");
|
||||
const char *host = getenv("REMOTE_ADDR");
|
||||
char *env[3];
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
int gzipped_request = 0;
|
||||
struct child_process cld;
|
||||
|
||||
if (encoding && !strcmp(encoding, "gzip"))
|
||||
gzipped_request = 1;
|
||||
else if (encoding && !strcmp(encoding, "x-gzip"))
|
||||
gzipped_request = 1;
|
||||
|
||||
if (!user || !*user)
|
||||
user = "anonymous";
|
||||
if (!host || !*host)
|
||||
host = "(none)";
|
||||
|
||||
memset(&env, 0, sizeof(env));
|
||||
strbuf_addf(&buf, "GIT_COMMITTER_NAME=%s", user);
|
||||
env[0] = strbuf_detach(&buf, NULL);
|
||||
|
||||
strbuf_addf(&buf, "GIT_COMMITTER_EMAIL=%s@http.%s", user, host);
|
||||
env[1] = strbuf_detach(&buf, NULL);
|
||||
env[2] = NULL;
|
||||
|
||||
memset(&cld, 0, sizeof(cld));
|
||||
cld.argv = argv;
|
||||
cld.env = (const char *const *)env;
|
||||
if (gzipped_request)
|
||||
cld.in = -1;
|
||||
cld.git_cmd = 1;
|
||||
if (start_command(&cld))
|
||||
exit(1);
|
||||
|
||||
close(1);
|
||||
if (gzipped_request)
|
||||
inflate_request(argv[0], cld.in);
|
||||
else
|
||||
close(0);
|
||||
|
||||
if (finish_command(&cld))
|
||||
exit(1);
|
||||
free(env[0]);
|
||||
free(env[1]);
|
||||
strbuf_release(&buf);
|
||||
}
|
||||
|
||||
static int show_text_ref(const char *name, const unsigned char *sha1,
|
||||
int flag, void *cb_data)
|
||||
{
|
||||
struct strbuf *buf = cb_data;
|
||||
struct object *o = parse_object(sha1);
|
||||
if (!o)
|
||||
return 0;
|
||||
|
||||
strbuf_addf(buf, "%s\t%s\n", sha1_to_hex(sha1), name);
|
||||
if (o->type == OBJ_TAG) {
|
||||
o = deref_tag(o, name, 0);
|
||||
if (!o)
|
||||
return 0;
|
||||
strbuf_addf(buf, "%s\t%s^{}\n", sha1_to_hex(o->sha1), name);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void get_info_refs(char *arg)
|
||||
{
|
||||
const char *service_name = get_parameter("service");
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
|
||||
hdr_nocache();
|
||||
|
||||
if (service_name) {
|
||||
const char *argv[] = {NULL /* service name */,
|
||||
"--stateless-rpc", "--advertise-refs",
|
||||
".", NULL};
|
||||
struct rpc_service *svc = select_service(service_name);
|
||||
|
||||
strbuf_addf(&buf, "application/x-git-%s-advertisement",
|
||||
svc->name);
|
||||
hdr_str(content_type, buf.buf);
|
||||
end_headers();
|
||||
|
||||
packet_write(1, "# service=git-%s\n", svc->name);
|
||||
packet_flush(1);
|
||||
|
||||
argv[0] = svc->name;
|
||||
run_service(argv);
|
||||
|
||||
} else {
|
||||
select_getanyfile();
|
||||
for_each_ref(show_text_ref, &buf);
|
||||
send_strbuf("text/plain", &buf);
|
||||
}
|
||||
strbuf_release(&buf);
|
||||
}
|
||||
|
||||
static void get_info_packs(char *arg)
|
||||
{
|
||||
size_t objdirlen = strlen(get_object_directory());
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
struct packed_git *p;
|
||||
size_t cnt = 0;
|
||||
|
||||
select_getanyfile();
|
||||
prepare_packed_git();
|
||||
for (p = packed_git; p; p = p->next) {
|
||||
if (p->pack_local)
|
||||
cnt++;
|
||||
}
|
||||
|
||||
strbuf_grow(&buf, cnt * 53 + 2);
|
||||
for (p = packed_git; p; p = p->next) {
|
||||
if (p->pack_local)
|
||||
strbuf_addf(&buf, "P %s\n", p->pack_name + objdirlen + 6);
|
||||
}
|
||||
strbuf_addch(&buf, '\n');
|
||||
|
||||
hdr_nocache();
|
||||
send_strbuf("text/plain; charset=utf-8", &buf);
|
||||
strbuf_release(&buf);
|
||||
}
|
||||
|
||||
static void check_content_type(const char *accepted_type)
|
||||
{
|
||||
const char *actual_type = getenv("CONTENT_TYPE");
|
||||
|
||||
if (!actual_type)
|
||||
actual_type = "";
|
||||
|
||||
if (strcmp(actual_type, accepted_type)) {
|
||||
http_status(415, "Unsupported Media Type");
|
||||
hdr_nocache();
|
||||
end_headers();
|
||||
format_write(1,
|
||||
"Expected POST with Content-Type '%s',"
|
||||
" but received '%s' instead.\n",
|
||||
accepted_type, actual_type);
|
||||
exit(0);
|
||||
}
|
||||
}
|
||||
|
||||
static void service_rpc(char *service_name)
|
||||
{
|
||||
const char *argv[] = {NULL, "--stateless-rpc", ".", NULL};
|
||||
struct rpc_service *svc = select_service(service_name);
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
|
||||
strbuf_reset(&buf);
|
||||
strbuf_addf(&buf, "application/x-git-%s-request", svc->name);
|
||||
check_content_type(buf.buf);
|
||||
|
||||
hdr_nocache();
|
||||
|
||||
strbuf_reset(&buf);
|
||||
strbuf_addf(&buf, "application/x-git-%s-result", svc->name);
|
||||
hdr_str(content_type, buf.buf);
|
||||
|
||||
end_headers();
|
||||
|
||||
argv[0] = svc->name;
|
||||
run_service(argv);
|
||||
strbuf_release(&buf);
|
||||
}
|
||||
|
||||
static NORETURN void die_webcgi(const char *err, va_list params)
|
||||
{
|
||||
char buffer[1000];
|
||||
|
||||
http_status(500, "Internal Server Error");
|
||||
hdr_nocache();
|
||||
end_headers();
|
||||
|
||||
vsnprintf(buffer, sizeof(buffer), err, params);
|
||||
fprintf(stderr, "fatal: %s\n", buffer);
|
||||
exit(0);
|
||||
}
|
||||
|
||||
static char* getdir(void)
|
||||
{
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
char *pathinfo = getenv("PATH_INFO");
|
||||
char *root = getenv("GIT_PROJECT_ROOT");
|
||||
char *path = getenv("PATH_TRANSLATED");
|
||||
|
||||
if (root && *root) {
|
||||
if (!pathinfo || !*pathinfo)
|
||||
die("GIT_PROJECT_ROOT is set but PATH_INFO is not");
|
||||
if (daemon_avoid_alias(pathinfo))
|
||||
die("'%s': aliased", pathinfo);
|
||||
strbuf_addstr(&buf, root);
|
||||
if (buf.buf[buf.len - 1] != '/')
|
||||
strbuf_addch(&buf, '/');
|
||||
if (pathinfo[0] == '/')
|
||||
pathinfo++;
|
||||
strbuf_addstr(&buf, pathinfo);
|
||||
return strbuf_detach(&buf, NULL);
|
||||
} else if (path && *path) {
|
||||
return xstrdup(path);
|
||||
} else
|
||||
die("No GIT_PROJECT_ROOT or PATH_TRANSLATED from server");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static struct service_cmd {
|
||||
const char *method;
|
||||
const char *pattern;
|
||||
void (*imp)(char *);
|
||||
} services[] = {
|
||||
{"GET", "/HEAD$", get_text_file},
|
||||
{"GET", "/info/refs$", get_info_refs},
|
||||
{"GET", "/objects/info/alternates$", get_text_file},
|
||||
{"GET", "/objects/info/http-alternates$", get_text_file},
|
||||
{"GET", "/objects/info/packs$", get_info_packs},
|
||||
{"GET", "/objects/[0-9a-f]{2}/[0-9a-f]{38}$", get_loose_object},
|
||||
{"GET", "/objects/pack/pack-[0-9a-f]{40}\\.pack$", get_pack_file},
|
||||
{"GET", "/objects/pack/pack-[0-9a-f]{40}\\.idx$", get_idx_file},
|
||||
|
||||
{"POST", "/git-upload-pack$", service_rpc},
|
||||
{"POST", "/git-receive-pack$", service_rpc}
|
||||
};
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
char *method = getenv("REQUEST_METHOD");
|
||||
char *dir;
|
||||
struct service_cmd *cmd = NULL;
|
||||
char *cmd_arg = NULL;
|
||||
int i;
|
||||
|
||||
git_extract_argv0_path(argv[0]);
|
||||
set_die_routine(die_webcgi);
|
||||
|
||||
if (!method)
|
||||
die("No REQUEST_METHOD from server");
|
||||
if (!strcmp(method, "HEAD"))
|
||||
method = "GET";
|
||||
dir = getdir();
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(services); i++) {
|
||||
struct service_cmd *c = &services[i];
|
||||
regex_t re;
|
||||
regmatch_t out[1];
|
||||
|
||||
if (regcomp(&re, c->pattern, REG_EXTENDED))
|
||||
die("Bogus regex in service table: %s", c->pattern);
|
||||
if (!regexec(&re, dir, 1, out, 0)) {
|
||||
size_t n;
|
||||
|
||||
if (strcmp(method, c->method)) {
|
||||
const char *proto = getenv("SERVER_PROTOCOL");
|
||||
if (proto && !strcmp(proto, "HTTP/1.1"))
|
||||
http_status(405, "Method Not Allowed");
|
||||
else
|
||||
http_status(400, "Bad Request");
|
||||
hdr_nocache();
|
||||
end_headers();
|
||||
return 0;
|
||||
}
|
||||
|
||||
cmd = c;
|
||||
n = out[0].rm_eo - out[0].rm_so;
|
||||
cmd_arg = xmalloc(n);
|
||||
memcpy(cmd_arg, dir + out[0].rm_so + 1, n-1);
|
||||
cmd_arg[n-1] = '\0';
|
||||
dir[out[0].rm_so] = 0;
|
||||
break;
|
||||
}
|
||||
regfree(&re);
|
||||
}
|
||||
|
||||
if (!cmd)
|
||||
not_found("Request not supported: '%s'", dir);
|
||||
|
||||
setup_path();
|
||||
if (!enter_repo(dir, 0))
|
||||
not_found("Not a git repository: '%s'", dir);
|
||||
|
||||
git_config(http_config, NULL);
|
||||
cmd->imp(cmd_arg);
|
||||
return 0;
|
||||
}
|
31
http-push.c
31
http-push.c
|
@ -78,6 +78,7 @@ static int push_verbosely;
|
|||
static int push_all = MATCH_REFS_NONE;
|
||||
static int force_all;
|
||||
static int dry_run;
|
||||
static int helper_status;
|
||||
|
||||
static struct object_list *objects;
|
||||
|
||||
|
@ -604,7 +605,7 @@ static void finish_request(struct transfer_request *request)
|
|||
preq = (struct http_pack_request *)request->userData;
|
||||
|
||||
if (preq) {
|
||||
if (finish_http_pack_request(preq) > 0)
|
||||
if (finish_http_pack_request(preq) == 0)
|
||||
fail = 0;
|
||||
release_http_pack_request(preq);
|
||||
}
|
||||
|
@ -1811,6 +1812,10 @@ int main(int argc, char **argv)
|
|||
dry_run = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--helper-status")) {
|
||||
helper_status = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--verbose")) {
|
||||
push_verbosely = 1;
|
||||
http_is_verbose = 1;
|
||||
|
@ -1913,9 +1918,12 @@ int main(int argc, char **argv)
|
|||
|
||||
/* Remove a remote branch if -d or -D was specified */
|
||||
if (delete_branch) {
|
||||
if (delete_remote_branch(refspec[0], force_delete) == -1)
|
||||
if (delete_remote_branch(refspec[0], force_delete) == -1) {
|
||||
fprintf(stderr, "Unable to delete remote branch %s\n",
|
||||
refspec[0]);
|
||||
if (helper_status)
|
||||
printf("error %s cannot remove\n", refspec[0]);
|
||||
}
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
|
@ -1927,6 +1935,8 @@ int main(int argc, char **argv)
|
|||
}
|
||||
if (!remote_refs) {
|
||||
fprintf(stderr, "No refs in common and none specified; doing nothing.\n");
|
||||
if (helper_status)
|
||||
printf("error null no match\n");
|
||||
rc = 0;
|
||||
goto cleanup;
|
||||
}
|
||||
|
@ -1944,8 +1954,12 @@ int main(int argc, char **argv)
|
|||
if (is_null_sha1(ref->peer_ref->new_sha1)) {
|
||||
if (delete_remote_branch(ref->name, 1) == -1) {
|
||||
error("Could not remove %s", ref->name);
|
||||
if (helper_status)
|
||||
printf("error %s cannot remove\n", ref->name);
|
||||
rc = -4;
|
||||
}
|
||||
else if (helper_status)
|
||||
printf("ok %s\n", ref->name);
|
||||
new_refs++;
|
||||
continue;
|
||||
}
|
||||
|
@ -1953,6 +1967,8 @@ int main(int argc, char **argv)
|
|||
if (!hashcmp(ref->old_sha1, ref->peer_ref->new_sha1)) {
|
||||
if (push_verbosely || 1)
|
||||
fprintf(stderr, "'%s': up-to-date\n", ref->name);
|
||||
if (helper_status)
|
||||
printf("ok %s up to date\n", ref->name);
|
||||
continue;
|
||||
}
|
||||
|
||||
|
@ -1976,6 +1992,8 @@ int main(int argc, char **argv)
|
|||
"need to pull first?",
|
||||
ref->name,
|
||||
ref->peer_ref->name);
|
||||
if (helper_status)
|
||||
printf("error %s non-fast forward\n", ref->name);
|
||||
rc = -2;
|
||||
continue;
|
||||
}
|
||||
|
@ -1989,14 +2007,19 @@ int main(int argc, char **argv)
|
|||
if (strcmp(ref->name, ref->peer_ref->name))
|
||||
fprintf(stderr, " using '%s'", ref->peer_ref->name);
|
||||
fprintf(stderr, "\n from %s\n to %s\n", old_hex, new_hex);
|
||||
if (dry_run)
|
||||
if (dry_run) {
|
||||
if (helper_status)
|
||||
printf("ok %s\n", ref->name);
|
||||
continue;
|
||||
}
|
||||
|
||||
/* Lock remote branch ref */
|
||||
ref_lock = lock_remote(ref->name, LOCK_TIME);
|
||||
if (ref_lock == NULL) {
|
||||
fprintf(stderr, "Unable to lock remote branch %s\n",
|
||||
ref->name);
|
||||
if (helper_status)
|
||||
printf("error %s lock error\n", ref->name);
|
||||
rc = 1;
|
||||
continue;
|
||||
}
|
||||
|
@ -2047,6 +2070,8 @@ int main(int argc, char **argv)
|
|||
|
||||
if (!rc)
|
||||
fprintf(stderr, " done\n");
|
||||
if (helper_status)
|
||||
printf("%s %s\n", !rc ? "ok" : "error", ref->name);
|
||||
unlock_remote(ref_lock);
|
||||
check_locks();
|
||||
}
|
||||
|
|
13
http.c
13
http.c
|
@ -1,9 +1,11 @@
|
|||
#include "http.h"
|
||||
#include "pack.h"
|
||||
#include "sideband.h"
|
||||
|
||||
int data_received;
|
||||
int active_requests;
|
||||
int http_is_verbose;
|
||||
size_t http_post_buffer = 16 * LARGE_PACKET_MAX;
|
||||
|
||||
#ifdef USE_CURL_MULTI
|
||||
static int max_requests = -1;
|
||||
|
@ -97,8 +99,6 @@ size_t fwrite_null(const void *ptr, size_t eltsize, size_t nmemb, void *strbuf)
|
|||
return eltsize * nmemb;
|
||||
}
|
||||
|
||||
static void finish_active_slot(struct active_request_slot *slot);
|
||||
|
||||
#ifdef USE_CURL_MULTI
|
||||
static void process_curl_messages(void)
|
||||
{
|
||||
|
@ -174,6 +174,13 @@ static int http_options(const char *var, const char *value, void *cb)
|
|||
if (!strcmp("http.proxy", var))
|
||||
return git_config_string(&curl_http_proxy, var, value);
|
||||
|
||||
if (!strcmp("http.postbuffer", var)) {
|
||||
http_post_buffer = git_config_int(var, value);
|
||||
if (http_post_buffer < LARGE_PACKET_MAX)
|
||||
http_post_buffer = LARGE_PACKET_MAX;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Fall back on the default ones */
|
||||
return git_default_config(var, value, cb);
|
||||
}
|
||||
|
@ -638,7 +645,7 @@ void release_active_slot(struct active_request_slot *slot)
|
|||
#endif
|
||||
}
|
||||
|
||||
static void finish_active_slot(struct active_request_slot *slot)
|
||||
void finish_active_slot(struct active_request_slot *slot)
|
||||
{
|
||||
closedown_active_slot(slot);
|
||||
curl_easy_getinfo(slot->curl, CURLINFO_HTTP_CODE, &slot->http_code);
|
||||
|
|
2
http.h
2
http.h
|
@ -79,6 +79,7 @@ extern curlioerr ioctl_buffer(CURL *handle, int cmd, void *clientp);
|
|||
extern struct active_request_slot *get_active_slot(void);
|
||||
extern int start_active_slot(struct active_request_slot *slot);
|
||||
extern void run_active_slot(struct active_request_slot *slot);
|
||||
extern void finish_active_slot(struct active_request_slot *slot);
|
||||
extern void finish_all_active_slots(void);
|
||||
extern void release_active_slot(struct active_request_slot *slot);
|
||||
|
||||
|
@ -94,6 +95,7 @@ extern void http_cleanup(void);
|
|||
extern int data_received;
|
||||
extern int active_requests;
|
||||
extern int http_is_verbose;
|
||||
extern size_t http_post_buffer;
|
||||
|
||||
extern char curl_errorstr[CURL_ERROR_SIZE];
|
||||
|
||||
|
|
47
path.c
47
path.c
|
@ -564,3 +564,50 @@ char *strip_path_suffix(const char *path, const char *suffix)
|
|||
return NULL;
|
||||
return xstrndup(path, chomp_trailing_dir_sep(path, path_len));
|
||||
}
|
||||
|
||||
int daemon_avoid_alias(const char *p)
|
||||
{
|
||||
int sl, ndot;
|
||||
|
||||
/*
|
||||
* This resurrects the belts and suspenders paranoia check by HPA
|
||||
* done in <435560F7.4080006@zytor.com> thread, now enter_repo()
|
||||
* does not do getcwd() based path canonicalizations.
|
||||
*
|
||||
* sl becomes true immediately after seeing '/' and continues to
|
||||
* be true as long as dots continue after that without intervening
|
||||
* non-dot character.
|
||||
*/
|
||||
if (!p || (*p != '/' && *p != '~'))
|
||||
return -1;
|
||||
sl = 1; ndot = 0;
|
||||
p++;
|
||||
|
||||
while (1) {
|
||||
char ch = *p++;
|
||||
if (sl) {
|
||||
if (ch == '.')
|
||||
ndot++;
|
||||
else if (ch == '/') {
|
||||
if (ndot < 3)
|
||||
/* reject //, /./ and /../ */
|
||||
return -1;
|
||||
ndot = 0;
|
||||
}
|
||||
else if (ch == 0) {
|
||||
if (0 < ndot && ndot < 3)
|
||||
/* reject /.$ and /..$ */
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
else
|
||||
sl = ndot = 0;
|
||||
}
|
||||
else if (ch == 0)
|
||||
return 0;
|
||||
else if (ch == '/') {
|
||||
sl = 1;
|
||||
ndot = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
86
pkt-line.c
86
pkt-line.c
|
@ -42,17 +42,19 @@ void packet_flush(int fd)
|
|||
safe_write(fd, "0000", 4);
|
||||
}
|
||||
|
||||
#define hex(a) (hexchar[(a) & 15])
|
||||
void packet_write(int fd, const char *fmt, ...)
|
||||
void packet_buf_flush(struct strbuf *buf)
|
||||
{
|
||||
strbuf_add(buf, "0000", 4);
|
||||
}
|
||||
|
||||
#define hex(a) (hexchar[(a) & 15])
|
||||
static char buffer[1000];
|
||||
static unsigned format_packet(const char *fmt, va_list args)
|
||||
{
|
||||
static char buffer[1000];
|
||||
static char hexchar[] = "0123456789abcdef";
|
||||
va_list args;
|
||||
unsigned n;
|
||||
|
||||
va_start(args, fmt);
|
||||
n = vsnprintf(buffer + 4, sizeof(buffer) - 4, fmt, args);
|
||||
va_end(args);
|
||||
if (n >= sizeof(buffer)-4)
|
||||
die("protocol error: impossibly long line");
|
||||
n += 4;
|
||||
|
@ -60,9 +62,31 @@ void packet_write(int fd, const char *fmt, ...)
|
|||
buffer[1] = hex(n >> 8);
|
||||
buffer[2] = hex(n >> 4);
|
||||
buffer[3] = hex(n);
|
||||
return n;
|
||||
}
|
||||
|
||||
void packet_write(int fd, const char *fmt, ...)
|
||||
{
|
||||
va_list args;
|
||||
unsigned n;
|
||||
|
||||
va_start(args, fmt);
|
||||
n = format_packet(fmt, args);
|
||||
va_end(args);
|
||||
safe_write(fd, buffer, n);
|
||||
}
|
||||
|
||||
void packet_buf_write(struct strbuf *buf, const char *fmt, ...)
|
||||
{
|
||||
va_list args;
|
||||
unsigned n;
|
||||
|
||||
va_start(args, fmt);
|
||||
n = format_packet(fmt, args);
|
||||
va_end(args);
|
||||
strbuf_add(buf, buffer, n);
|
||||
}
|
||||
|
||||
static void safe_read(int fd, void *buffer, unsigned size)
|
||||
{
|
||||
ssize_t ret = read_in_full(fd, buffer, size);
|
||||
|
@ -72,15 +96,11 @@ static void safe_read(int fd, void *buffer, unsigned size)
|
|||
die("The remote end hung up unexpectedly");
|
||||
}
|
||||
|
||||
int packet_read_line(int fd, char *buffer, unsigned size)
|
||||
static int packet_length(const char *linelen)
|
||||
{
|
||||
int n;
|
||||
unsigned len;
|
||||
char linelen[4];
|
||||
int len = 0;
|
||||
|
||||
safe_read(fd, linelen, 4);
|
||||
|
||||
len = 0;
|
||||
for (n = 0; n < 4; n++) {
|
||||
unsigned char c = linelen[n];
|
||||
len <<= 4;
|
||||
|
@ -96,8 +116,20 @@ int packet_read_line(int fd, char *buffer, unsigned size)
|
|||
len += c - 'A' + 10;
|
||||
continue;
|
||||
}
|
||||
die("protocol error: bad line length character");
|
||||
return -1;
|
||||
}
|
||||
return len;
|
||||
}
|
||||
|
||||
int packet_read_line(int fd, char *buffer, unsigned size)
|
||||
{
|
||||
int len;
|
||||
char linelen[4];
|
||||
|
||||
safe_read(fd, linelen, 4);
|
||||
len = packet_length(linelen);
|
||||
if (len < 0)
|
||||
die("protocol error: bad line length character: %.4s", linelen);
|
||||
if (!len)
|
||||
return 0;
|
||||
len -= 4;
|
||||
|
@ -107,3 +139,31 @@ int packet_read_line(int fd, char *buffer, unsigned size)
|
|||
buffer[len] = 0;
|
||||
return len;
|
||||
}
|
||||
|
||||
int packet_get_line(struct strbuf *out,
|
||||
char **src_buf, size_t *src_len)
|
||||
{
|
||||
int len;
|
||||
|
||||
if (*src_len < 4)
|
||||
return -1;
|
||||
len = packet_length(*src_buf);
|
||||
if (len < 0)
|
||||
return -1;
|
||||
if (!len) {
|
||||
*src_buf += 4;
|
||||
*src_len -= 4;
|
||||
return 0;
|
||||
}
|
||||
if (*src_len < len)
|
||||
return -2;
|
||||
|
||||
*src_buf += 4;
|
||||
*src_len -= 4;
|
||||
len -= 4;
|
||||
|
||||
strbuf_add(out, *src_buf, len);
|
||||
*src_buf += len;
|
||||
*src_len -= len;
|
||||
return len;
|
||||
}
|
||||
|
|
|
@ -2,14 +2,18 @@
|
|||
#define PKTLINE_H
|
||||
|
||||
#include "git-compat-util.h"
|
||||
#include "strbuf.h"
|
||||
|
||||
/*
|
||||
* Silly packetized line writing interface
|
||||
*/
|
||||
void packet_flush(int fd);
|
||||
void packet_write(int fd, const char *fmt, ...) __attribute__((format (printf, 2, 3)));
|
||||
void packet_buf_flush(struct strbuf *buf);
|
||||
void packet_buf_write(struct strbuf *buf, const char *fmt, ...) __attribute__((format (printf, 2, 3)));
|
||||
|
||||
int packet_read_line(int fd, char *buffer, unsigned size);
|
||||
int packet_get_line(struct strbuf *out, char **src_buf, size_t *src_len);
|
||||
ssize_t safe_write(int, const void *, ssize_t);
|
||||
|
||||
#endif
|
||||
|
|
759
remote-curl.c
759
remote-curl.c
|
@ -4,23 +4,122 @@
|
|||
#include "walker.h"
|
||||
#include "http.h"
|
||||
#include "exec_cmd.h"
|
||||
#include "run-command.h"
|
||||
#include "pkt-line.h"
|
||||
#include "sideband.h"
|
||||
|
||||
static struct ref *get_refs(struct walker *walker, const char *url)
|
||||
static struct remote *remote;
|
||||
static const char *url;
|
||||
static struct walker *walker;
|
||||
|
||||
struct options {
|
||||
int verbosity;
|
||||
unsigned long depth;
|
||||
unsigned progress : 1,
|
||||
followtags : 1,
|
||||
dry_run : 1,
|
||||
thin : 1;
|
||||
};
|
||||
static struct options options;
|
||||
|
||||
static void init_walker(void)
|
||||
{
|
||||
if (!walker)
|
||||
walker = get_http_walker(url, remote);
|
||||
}
|
||||
|
||||
static int set_option(const char *name, const char *value)
|
||||
{
|
||||
if (!strcmp(name, "verbosity")) {
|
||||
char *end;
|
||||
int v = strtol(value, &end, 10);
|
||||
if (value == end || *end)
|
||||
return -1;
|
||||
options.verbosity = v;
|
||||
return 0;
|
||||
}
|
||||
else if (!strcmp(name, "progress")) {
|
||||
if (!strcmp(value, "true"))
|
||||
options.progress = 1;
|
||||
else if (!strcmp(value, "false"))
|
||||
options.progress = 0;
|
||||
else
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
else if (!strcmp(name, "depth")) {
|
||||
char *end;
|
||||
unsigned long v = strtoul(value, &end, 10);
|
||||
if (value == end || *end)
|
||||
return -1;
|
||||
options.depth = v;
|
||||
return 0;
|
||||
}
|
||||
else if (!strcmp(name, "followtags")) {
|
||||
if (!strcmp(value, "true"))
|
||||
options.followtags = 1;
|
||||
else if (!strcmp(value, "false"))
|
||||
options.followtags = 0;
|
||||
else
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
else if (!strcmp(name, "dry-run")) {
|
||||
if (!strcmp(value, "true"))
|
||||
options.dry_run = 1;
|
||||
else if (!strcmp(value, "false"))
|
||||
options.dry_run = 0;
|
||||
else
|
||||
return -1;
|
||||
return 0;
|
||||
}
|
||||
else {
|
||||
return 1 /* unsupported */;
|
||||
}
|
||||
}
|
||||
|
||||
struct discovery {
|
||||
const char *service;
|
||||
char *buf_alloc;
|
||||
char *buf;
|
||||
size_t len;
|
||||
unsigned proto_git : 1;
|
||||
};
|
||||
static struct discovery *last_discovery;
|
||||
|
||||
static void free_discovery(struct discovery *d)
|
||||
{
|
||||
if (d) {
|
||||
if (d == last_discovery)
|
||||
last_discovery = NULL;
|
||||
free(d->buf_alloc);
|
||||
free(d);
|
||||
}
|
||||
}
|
||||
|
||||
static struct discovery* discover_refs(const char *service)
|
||||
{
|
||||
struct strbuf buffer = STRBUF_INIT;
|
||||
char *data, *start, *mid;
|
||||
char *ref_name;
|
||||
struct discovery *last = last_discovery;
|
||||
char *refs_url;
|
||||
int i = 0;
|
||||
int http_ret;
|
||||
int http_ret, is_http = 0;
|
||||
|
||||
struct ref *refs = NULL;
|
||||
struct ref *ref = NULL;
|
||||
struct ref *last_ref = NULL;
|
||||
if (last && !strcmp(service, last->service))
|
||||
return last;
|
||||
free_discovery(last);
|
||||
|
||||
refs_url = xmalloc(strlen(url) + 11);
|
||||
sprintf(refs_url, "%s/info/refs", url);
|
||||
strbuf_addf(&buffer, "%s/info/refs", url);
|
||||
if (!prefixcmp(url, "http://") || !prefixcmp(url, "https://")) {
|
||||
is_http = 1;
|
||||
if (!strchr(url, '?'))
|
||||
strbuf_addch(&buffer, '?');
|
||||
else
|
||||
strbuf_addch(&buffer, '&');
|
||||
strbuf_addf(&buffer, "service=%s", service);
|
||||
}
|
||||
refs_url = strbuf_detach(&buffer, NULL);
|
||||
|
||||
init_walker();
|
||||
http_ret = http_get_strbuf(refs_url, &buffer, HTTP_NO_CACHE);
|
||||
switch (http_ret) {
|
||||
case HTTP_OK:
|
||||
|
@ -33,10 +132,86 @@ static struct ref *get_refs(struct walker *walker, const char *url)
|
|||
die("HTTP request failed");
|
||||
}
|
||||
|
||||
data = buffer.buf;
|
||||
last= xcalloc(1, sizeof(*last_discovery));
|
||||
last->service = service;
|
||||
last->buf_alloc = strbuf_detach(&buffer, &last->len);
|
||||
last->buf = last->buf_alloc;
|
||||
|
||||
if (is_http && 5 <= last->len && last->buf[4] == '#') {
|
||||
/* smart HTTP response; validate that the service
|
||||
* pkt-line matches our request.
|
||||
*/
|
||||
struct strbuf exp = STRBUF_INIT;
|
||||
|
||||
if (packet_get_line(&buffer, &last->buf, &last->len) <= 0)
|
||||
die("%s has invalid packet header", refs_url);
|
||||
if (buffer.len && buffer.buf[buffer.len - 1] == '\n')
|
||||
strbuf_setlen(&buffer, buffer.len - 1);
|
||||
|
||||
strbuf_addf(&exp, "# service=%s", service);
|
||||
if (strbuf_cmp(&exp, &buffer))
|
||||
die("invalid server response; got '%s'", buffer.buf);
|
||||
strbuf_release(&exp);
|
||||
|
||||
/* The header can include additional metadata lines, up
|
||||
* until a packet flush marker. Ignore these now, but
|
||||
* in the future we might start to scan them.
|
||||
*/
|
||||
strbuf_reset(&buffer);
|
||||
while (packet_get_line(&buffer, &last->buf, &last->len) > 0)
|
||||
strbuf_reset(&buffer);
|
||||
|
||||
last->proto_git = 1;
|
||||
}
|
||||
|
||||
free(refs_url);
|
||||
strbuf_release(&buffer);
|
||||
last_discovery = last;
|
||||
return last;
|
||||
}
|
||||
|
||||
static int write_discovery(int fd, void *data)
|
||||
{
|
||||
struct discovery *heads = data;
|
||||
int err = 0;
|
||||
if (write_in_full(fd, heads->buf, heads->len) != heads->len)
|
||||
err = 1;
|
||||
close(fd);
|
||||
return err;
|
||||
}
|
||||
|
||||
static struct ref *parse_git_refs(struct discovery *heads)
|
||||
{
|
||||
struct ref *list = NULL;
|
||||
struct async async;
|
||||
|
||||
memset(&async, 0, sizeof(async));
|
||||
async.proc = write_discovery;
|
||||
async.data = heads;
|
||||
|
||||
if (start_async(&async))
|
||||
die("cannot start thread to parse advertised refs");
|
||||
get_remote_heads(async.out, &list, 0, NULL, 0, NULL);
|
||||
close(async.out);
|
||||
if (finish_async(&async))
|
||||
die("ref parsing thread failed");
|
||||
return list;
|
||||
}
|
||||
|
||||
static struct ref *parse_info_refs(struct discovery *heads)
|
||||
{
|
||||
char *data, *start, *mid;
|
||||
char *ref_name;
|
||||
int i = 0;
|
||||
|
||||
struct ref *refs = NULL;
|
||||
struct ref *ref = NULL;
|
||||
struct ref *last_ref = NULL;
|
||||
|
||||
data = heads->buf;
|
||||
start = NULL;
|
||||
mid = data;
|
||||
while (i < buffer.len) {
|
||||
while (i < heads->len) {
|
||||
if (!start) {
|
||||
start = &data[i];
|
||||
}
|
||||
|
@ -60,8 +235,7 @@ static struct ref *get_refs(struct walker *walker, const char *url)
|
|||
i++;
|
||||
}
|
||||
|
||||
strbuf_release(&buffer);
|
||||
|
||||
init_walker();
|
||||
ref = alloc_ref("HEAD");
|
||||
if (!walker->fetch_ref(walker, ref) &&
|
||||
!resolve_remote_symref(ref, refs)) {
|
||||
|
@ -71,17 +245,506 @@ static struct ref *get_refs(struct walker *walker, const char *url)
|
|||
free(ref);
|
||||
}
|
||||
|
||||
strbuf_release(&buffer);
|
||||
free(refs_url);
|
||||
return refs;
|
||||
}
|
||||
|
||||
static struct ref *get_refs(int for_push)
|
||||
{
|
||||
struct discovery *heads;
|
||||
|
||||
if (for_push)
|
||||
heads = discover_refs("git-receive-pack");
|
||||
else
|
||||
heads = discover_refs("git-upload-pack");
|
||||
|
||||
if (heads->proto_git)
|
||||
return parse_git_refs(heads);
|
||||
return parse_info_refs(heads);
|
||||
}
|
||||
|
||||
static void output_refs(struct ref *refs)
|
||||
{
|
||||
struct ref *posn;
|
||||
for (posn = refs; posn; posn = posn->next) {
|
||||
if (posn->symref)
|
||||
printf("@%s %s\n", posn->symref, posn->name);
|
||||
else
|
||||
printf("%s %s\n", sha1_to_hex(posn->old_sha1), posn->name);
|
||||
}
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
free_refs(refs);
|
||||
}
|
||||
|
||||
struct rpc_state {
|
||||
const char *service_name;
|
||||
const char **argv;
|
||||
char *service_url;
|
||||
char *hdr_content_type;
|
||||
char *hdr_accept;
|
||||
char *buf;
|
||||
size_t alloc;
|
||||
size_t len;
|
||||
size_t pos;
|
||||
int in;
|
||||
int out;
|
||||
struct strbuf result;
|
||||
unsigned gzip_request : 1;
|
||||
};
|
||||
|
||||
static size_t rpc_out(void *ptr, size_t eltsize,
|
||||
size_t nmemb, void *buffer_)
|
||||
{
|
||||
size_t max = eltsize * nmemb;
|
||||
struct rpc_state *rpc = buffer_;
|
||||
size_t avail = rpc->len - rpc->pos;
|
||||
|
||||
if (!avail) {
|
||||
avail = packet_read_line(rpc->out, rpc->buf, rpc->alloc);
|
||||
if (!avail)
|
||||
return 0;
|
||||
rpc->pos = 0;
|
||||
rpc->len = avail;
|
||||
}
|
||||
|
||||
if (max < avail);
|
||||
avail = max;
|
||||
memcpy(ptr, rpc->buf + rpc->pos, avail);
|
||||
rpc->pos += avail;
|
||||
return avail;
|
||||
}
|
||||
|
||||
static size_t rpc_in(const void *ptr, size_t eltsize,
|
||||
size_t nmemb, void *buffer_)
|
||||
{
|
||||
size_t size = eltsize * nmemb;
|
||||
struct rpc_state *rpc = buffer_;
|
||||
write_or_die(rpc->in, ptr, size);
|
||||
return size;
|
||||
}
|
||||
|
||||
static int post_rpc(struct rpc_state *rpc)
|
||||
{
|
||||
struct active_request_slot *slot;
|
||||
struct slot_results results;
|
||||
struct curl_slist *headers = NULL;
|
||||
int use_gzip = rpc->gzip_request;
|
||||
char *gzip_body = NULL;
|
||||
int err = 0, large_request = 0;
|
||||
|
||||
/* Try to load the entire request, if we can fit it into the
|
||||
* allocated buffer space we can use HTTP/1.0 and avoid the
|
||||
* chunked encoding mess.
|
||||
*/
|
||||
while (1) {
|
||||
size_t left = rpc->alloc - rpc->len;
|
||||
char *buf = rpc->buf + rpc->len;
|
||||
int n;
|
||||
|
||||
if (left < LARGE_PACKET_MAX) {
|
||||
large_request = 1;
|
||||
use_gzip = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
n = packet_read_line(rpc->out, buf, left);
|
||||
if (!n)
|
||||
break;
|
||||
rpc->len += n;
|
||||
}
|
||||
|
||||
slot = get_active_slot();
|
||||
slot->results = &results;
|
||||
|
||||
curl_easy_setopt(slot->curl, CURLOPT_POST, 1);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_NOBODY, 0);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_URL, rpc->service_url);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_ENCODING, "");
|
||||
|
||||
headers = curl_slist_append(headers, rpc->hdr_content_type);
|
||||
headers = curl_slist_append(headers, rpc->hdr_accept);
|
||||
|
||||
if (large_request) {
|
||||
/* The request body is large and the size cannot be predicted.
|
||||
* We must use chunked encoding to send it.
|
||||
*/
|
||||
headers = curl_slist_append(headers, "Expect: 100-continue");
|
||||
headers = curl_slist_append(headers, "Transfer-Encoding: chunked");
|
||||
curl_easy_setopt(slot->curl, CURLOPT_READFUNCTION, rpc_out);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_INFILE, rpc);
|
||||
if (options.verbosity > 1) {
|
||||
fprintf(stderr, "POST %s (chunked)\n", rpc->service_name);
|
||||
fflush(stderr);
|
||||
}
|
||||
|
||||
} else if (use_gzip && 1024 < rpc->len) {
|
||||
/* The client backend isn't giving us compressed data so
|
||||
* we can try to deflate it ourselves, this may save on.
|
||||
* the transfer time.
|
||||
*/
|
||||
size_t size;
|
||||
z_stream stream;
|
||||
int ret;
|
||||
|
||||
memset(&stream, 0, sizeof(stream));
|
||||
ret = deflateInit2(&stream, Z_BEST_COMPRESSION,
|
||||
Z_DEFLATED, (15 + 16),
|
||||
8, Z_DEFAULT_STRATEGY);
|
||||
if (ret != Z_OK)
|
||||
die("cannot deflate request; zlib init error %d", ret);
|
||||
size = deflateBound(&stream, rpc->len);
|
||||
gzip_body = xmalloc(size);
|
||||
|
||||
stream.next_in = (unsigned char *)rpc->buf;
|
||||
stream.avail_in = rpc->len;
|
||||
stream.next_out = (unsigned char *)gzip_body;
|
||||
stream.avail_out = size;
|
||||
|
||||
ret = deflate(&stream, Z_FINISH);
|
||||
if (ret != Z_STREAM_END)
|
||||
die("cannot deflate request; zlib deflate error %d", ret);
|
||||
|
||||
ret = deflateEnd(&stream);
|
||||
if (ret != Z_OK)
|
||||
die("cannot deflate request; zlib end error %d", ret);
|
||||
|
||||
size = stream.total_out;
|
||||
|
||||
headers = curl_slist_append(headers, "Content-Encoding: gzip");
|
||||
curl_easy_setopt(slot->curl, CURLOPT_POSTFIELDS, gzip_body);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_POSTFIELDSIZE, size);
|
||||
|
||||
if (options.verbosity > 1) {
|
||||
fprintf(stderr, "POST %s (gzip %lu to %lu bytes)\n",
|
||||
rpc->service_name,
|
||||
(unsigned long)rpc->len, (unsigned long)size);
|
||||
fflush(stderr);
|
||||
}
|
||||
} else {
|
||||
/* We know the complete request size in advance, use the
|
||||
* more normal Content-Length approach.
|
||||
*/
|
||||
curl_easy_setopt(slot->curl, CURLOPT_POSTFIELDS, rpc->buf);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_POSTFIELDSIZE, rpc->len);
|
||||
if (options.verbosity > 1) {
|
||||
fprintf(stderr, "POST %s (%lu bytes)\n",
|
||||
rpc->service_name, (unsigned long)rpc->len);
|
||||
fflush(stderr);
|
||||
}
|
||||
}
|
||||
|
||||
curl_easy_setopt(slot->curl, CURLOPT_HTTPHEADER, headers);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_WRITEFUNCTION, rpc_in);
|
||||
curl_easy_setopt(slot->curl, CURLOPT_FILE, rpc);
|
||||
|
||||
slot->curl_result = curl_easy_perform(slot->curl);
|
||||
finish_active_slot(slot);
|
||||
|
||||
if (results.curl_result != CURLE_OK) {
|
||||
err |= error("RPC failed; result=%d, HTTP code = %ld",
|
||||
results.curl_result, results.http_code);
|
||||
}
|
||||
|
||||
curl_slist_free_all(headers);
|
||||
free(gzip_body);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int rpc_service(struct rpc_state *rpc, struct discovery *heads)
|
||||
{
|
||||
const char *svc = rpc->service_name;
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
struct child_process client;
|
||||
int err = 0;
|
||||
|
||||
init_walker();
|
||||
memset(&client, 0, sizeof(client));
|
||||
client.in = -1;
|
||||
client.out = -1;
|
||||
client.git_cmd = 1;
|
||||
client.argv = rpc->argv;
|
||||
if (start_command(&client))
|
||||
exit(1);
|
||||
if (heads)
|
||||
write_or_die(client.in, heads->buf, heads->len);
|
||||
|
||||
rpc->alloc = http_post_buffer;
|
||||
rpc->buf = xmalloc(rpc->alloc);
|
||||
rpc->in = client.in;
|
||||
rpc->out = client.out;
|
||||
strbuf_init(&rpc->result, 0);
|
||||
|
||||
strbuf_addf(&buf, "%s/%s", url, svc);
|
||||
rpc->service_url = strbuf_detach(&buf, NULL);
|
||||
|
||||
strbuf_addf(&buf, "Content-Type: application/x-%s-request", svc);
|
||||
rpc->hdr_content_type = strbuf_detach(&buf, NULL);
|
||||
|
||||
strbuf_addf(&buf, "Accept: application/x-%s-response", svc);
|
||||
rpc->hdr_accept = strbuf_detach(&buf, NULL);
|
||||
|
||||
while (!err) {
|
||||
int n = packet_read_line(rpc->out, rpc->buf, rpc->alloc);
|
||||
if (!n)
|
||||
break;
|
||||
rpc->pos = 0;
|
||||
rpc->len = n;
|
||||
err |= post_rpc(rpc);
|
||||
}
|
||||
strbuf_read(&rpc->result, client.out, 0);
|
||||
|
||||
close(client.in);
|
||||
close(client.out);
|
||||
client.in = -1;
|
||||
client.out = -1;
|
||||
|
||||
err |= finish_command(&client);
|
||||
free(rpc->service_url);
|
||||
free(rpc->hdr_content_type);
|
||||
free(rpc->hdr_accept);
|
||||
free(rpc->buf);
|
||||
strbuf_release(&buf);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int fetch_dumb(int nr_heads, struct ref **to_fetch)
|
||||
{
|
||||
char **targets = xmalloc(nr_heads * sizeof(char*));
|
||||
int ret, i;
|
||||
|
||||
if (options.depth)
|
||||
die("dumb http transport does not support --depth");
|
||||
for (i = 0; i < nr_heads; i++)
|
||||
targets[i] = xstrdup(sha1_to_hex(to_fetch[i]->old_sha1));
|
||||
|
||||
init_walker();
|
||||
walker->get_all = 1;
|
||||
walker->get_tree = 1;
|
||||
walker->get_history = 1;
|
||||
walker->get_verbosely = options.verbosity >= 3;
|
||||
walker->get_recover = 0;
|
||||
ret = walker_fetch(walker, nr_heads, targets, NULL, NULL);
|
||||
|
||||
for (i = 0; i < nr_heads; i++)
|
||||
free(targets[i]);
|
||||
free(targets);
|
||||
|
||||
return ret ? error("Fetch failed.") : 0;
|
||||
}
|
||||
|
||||
static int fetch_git(struct discovery *heads,
|
||||
int nr_heads, struct ref **to_fetch)
|
||||
{
|
||||
struct rpc_state rpc;
|
||||
char *depth_arg = NULL;
|
||||
const char **argv;
|
||||
int argc = 0, i, err;
|
||||
|
||||
argv = xmalloc((15 + nr_heads) * sizeof(char*));
|
||||
argv[argc++] = "fetch-pack";
|
||||
argv[argc++] = "--stateless-rpc";
|
||||
argv[argc++] = "--lock-pack";
|
||||
if (options.followtags)
|
||||
argv[argc++] = "--include-tag";
|
||||
if (options.thin)
|
||||
argv[argc++] = "--thin";
|
||||
if (options.verbosity >= 3) {
|
||||
argv[argc++] = "-v";
|
||||
argv[argc++] = "-v";
|
||||
}
|
||||
if (!options.progress)
|
||||
argv[argc++] = "--no-progress";
|
||||
if (options.depth) {
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
strbuf_addf(&buf, "--depth=%lu", options.depth);
|
||||
depth_arg = strbuf_detach(&buf, NULL);
|
||||
argv[argc++] = depth_arg;
|
||||
}
|
||||
argv[argc++] = url;
|
||||
for (i = 0; i < nr_heads; i++) {
|
||||
struct ref *ref = to_fetch[i];
|
||||
if (!ref->name || !*ref->name)
|
||||
die("cannot fetch by sha1 over smart http");
|
||||
argv[argc++] = ref->name;
|
||||
}
|
||||
argv[argc++] = NULL;
|
||||
|
||||
memset(&rpc, 0, sizeof(rpc));
|
||||
rpc.service_name = "git-upload-pack",
|
||||
rpc.argv = argv;
|
||||
rpc.gzip_request = 1;
|
||||
|
||||
err = rpc_service(&rpc, heads);
|
||||
if (rpc.result.len)
|
||||
safe_write(1, rpc.result.buf, rpc.result.len);
|
||||
strbuf_release(&rpc.result);
|
||||
free(argv);
|
||||
free(depth_arg);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int fetch(int nr_heads, struct ref **to_fetch)
|
||||
{
|
||||
struct discovery *d = discover_refs("git-upload-pack");
|
||||
if (d->proto_git)
|
||||
return fetch_git(d, nr_heads, to_fetch);
|
||||
else
|
||||
return fetch_dumb(nr_heads, to_fetch);
|
||||
}
|
||||
|
||||
static void parse_fetch(struct strbuf *buf)
|
||||
{
|
||||
struct ref **to_fetch = NULL;
|
||||
struct ref *list_head = NULL;
|
||||
struct ref **list = &list_head;
|
||||
int alloc_heads = 0, nr_heads = 0;
|
||||
|
||||
do {
|
||||
if (!prefixcmp(buf->buf, "fetch ")) {
|
||||
char *p = buf->buf + strlen("fetch ");
|
||||
char *name;
|
||||
struct ref *ref;
|
||||
unsigned char old_sha1[20];
|
||||
|
||||
if (strlen(p) < 40 || get_sha1_hex(p, old_sha1))
|
||||
die("protocol error: expected sha/ref, got %s'", p);
|
||||
if (p[40] == ' ')
|
||||
name = p + 41;
|
||||
else if (!p[40])
|
||||
name = "";
|
||||
else
|
||||
die("protocol error: expected sha/ref, got %s'", p);
|
||||
|
||||
ref = alloc_ref(name);
|
||||
hashcpy(ref->old_sha1, old_sha1);
|
||||
|
||||
*list = ref;
|
||||
list = &ref->next;
|
||||
|
||||
ALLOC_GROW(to_fetch, nr_heads + 1, alloc_heads);
|
||||
to_fetch[nr_heads++] = ref;
|
||||
}
|
||||
else
|
||||
die("http transport does not support %s", buf->buf);
|
||||
|
||||
strbuf_reset(buf);
|
||||
if (strbuf_getline(buf, stdin, '\n') == EOF)
|
||||
return;
|
||||
if (!*buf->buf)
|
||||
break;
|
||||
} while (1);
|
||||
|
||||
if (fetch(nr_heads, to_fetch))
|
||||
exit(128); /* error already reported */
|
||||
free_refs(list_head);
|
||||
free(to_fetch);
|
||||
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
strbuf_reset(buf);
|
||||
}
|
||||
|
||||
static int push_dav(int nr_spec, char **specs)
|
||||
{
|
||||
const char **argv = xmalloc((10 + nr_spec) * sizeof(char*));
|
||||
int argc = 0, i;
|
||||
|
||||
argv[argc++] = "http-push";
|
||||
argv[argc++] = "--helper-status";
|
||||
if (options.dry_run)
|
||||
argv[argc++] = "--dry-run";
|
||||
if (options.verbosity > 1)
|
||||
argv[argc++] = "--verbose";
|
||||
argv[argc++] = url;
|
||||
for (i = 0; i < nr_spec; i++)
|
||||
argv[argc++] = specs[i];
|
||||
argv[argc++] = NULL;
|
||||
|
||||
if (run_command_v_opt(argv, RUN_GIT_CMD))
|
||||
die("git-%s failed", argv[0]);
|
||||
free(argv);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int push_git(struct discovery *heads, int nr_spec, char **specs)
|
||||
{
|
||||
struct rpc_state rpc;
|
||||
const char **argv;
|
||||
int argc = 0, i, err;
|
||||
|
||||
argv = xmalloc((10 + nr_spec) * sizeof(char*));
|
||||
argv[argc++] = "send-pack";
|
||||
argv[argc++] = "--stateless-rpc";
|
||||
argv[argc++] = "--helper-status";
|
||||
if (options.thin)
|
||||
argv[argc++] = "--thin";
|
||||
if (options.dry_run)
|
||||
argv[argc++] = "--dry-run";
|
||||
if (options.verbosity > 1)
|
||||
argv[argc++] = "--verbose";
|
||||
argv[argc++] = url;
|
||||
for (i = 0; i < nr_spec; i++)
|
||||
argv[argc++] = specs[i];
|
||||
argv[argc++] = NULL;
|
||||
|
||||
memset(&rpc, 0, sizeof(rpc));
|
||||
rpc.service_name = "git-receive-pack",
|
||||
rpc.argv = argv;
|
||||
|
||||
err = rpc_service(&rpc, heads);
|
||||
if (rpc.result.len)
|
||||
safe_write(1, rpc.result.buf, rpc.result.len);
|
||||
strbuf_release(&rpc.result);
|
||||
free(argv);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int push(int nr_spec, char **specs)
|
||||
{
|
||||
struct discovery *heads = discover_refs("git-receive-pack");
|
||||
int ret;
|
||||
|
||||
if (heads->proto_git)
|
||||
ret = push_git(heads, nr_spec, specs);
|
||||
else
|
||||
ret = push_dav(nr_spec, specs);
|
||||
free_discovery(heads);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void parse_push(struct strbuf *buf)
|
||||
{
|
||||
char **specs = NULL;
|
||||
int alloc_spec = 0, nr_spec = 0, i;
|
||||
|
||||
do {
|
||||
if (!prefixcmp(buf->buf, "push ")) {
|
||||
ALLOC_GROW(specs, nr_spec + 1, alloc_spec);
|
||||
specs[nr_spec++] = xstrdup(buf->buf + 5);
|
||||
}
|
||||
else
|
||||
die("http transport does not support %s", buf->buf);
|
||||
|
||||
strbuf_reset(buf);
|
||||
if (strbuf_getline(buf, stdin, '\n') == EOF)
|
||||
return;
|
||||
if (!*buf->buf)
|
||||
break;
|
||||
} while (1);
|
||||
|
||||
if (push(nr_spec, specs))
|
||||
exit(128); /* error already reported */
|
||||
for (i = 0; i < nr_spec; i++)
|
||||
free(specs[i]);
|
||||
free(specs);
|
||||
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
}
|
||||
|
||||
int main(int argc, const char **argv)
|
||||
{
|
||||
struct remote *remote;
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
const char *url;
|
||||
struct walker *walker = NULL;
|
||||
int nongit;
|
||||
|
||||
git_extract_argv0_path(argv[0]);
|
||||
|
@ -91,6 +754,10 @@ int main(int argc, const char **argv)
|
|||
return 1;
|
||||
}
|
||||
|
||||
options.verbosity = 1;
|
||||
options.progress = !!isatty(2);
|
||||
options.thin = 1;
|
||||
|
||||
remote = remote_get(argv[1]);
|
||||
|
||||
if (argc > 2) {
|
||||
|
@ -103,36 +770,40 @@ int main(int argc, const char **argv)
|
|||
if (strbuf_getline(&buf, stdin, '\n') == EOF)
|
||||
break;
|
||||
if (!prefixcmp(buf.buf, "fetch ")) {
|
||||
char *obj = buf.buf + strlen("fetch ");
|
||||
if (nongit)
|
||||
die("Fetch attempted without a local repo");
|
||||
if (!walker)
|
||||
walker = get_http_walker(url, remote);
|
||||
walker->get_all = 1;
|
||||
walker->get_tree = 1;
|
||||
walker->get_history = 1;
|
||||
walker->get_verbosely = 0;
|
||||
walker->get_recover = 0;
|
||||
if (walker_fetch(walker, 1, &obj, NULL, NULL))
|
||||
die("Fetch failed.");
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
} else if (!strcmp(buf.buf, "list")) {
|
||||
struct ref *refs;
|
||||
struct ref *posn;
|
||||
if (!walker)
|
||||
walker = get_http_walker(url, remote);
|
||||
refs = get_refs(walker, url);
|
||||
for (posn = refs; posn; posn = posn->next) {
|
||||
if (posn->symref)
|
||||
printf("@%s %s\n", posn->symref, posn->name);
|
||||
else
|
||||
printf("%s %s\n", sha1_to_hex(posn->old_sha1), posn->name);
|
||||
}
|
||||
printf("\n");
|
||||
parse_fetch(&buf);
|
||||
|
||||
} else if (!strcmp(buf.buf, "list") || !prefixcmp(buf.buf, "list ")) {
|
||||
int for_push = !!strstr(buf.buf + 4, "for-push");
|
||||
output_refs(get_refs(for_push));
|
||||
|
||||
} else if (!prefixcmp(buf.buf, "push ")) {
|
||||
parse_push(&buf);
|
||||
|
||||
} else if (!prefixcmp(buf.buf, "option ")) {
|
||||
char *name = buf.buf + strlen("option ");
|
||||
char *value = strchr(name, ' ');
|
||||
int result;
|
||||
|
||||
if (value)
|
||||
*value++ = '\0';
|
||||
else
|
||||
value = "true";
|
||||
|
||||
result = set_option(name, value);
|
||||
if (!result)
|
||||
printf("ok\n");
|
||||
else if (result < 0)
|
||||
printf("error invalid value\n");
|
||||
else
|
||||
printf("unsupported\n");
|
||||
fflush(stdout);
|
||||
|
||||
} else if (!strcmp(buf.buf, "capabilities")) {
|
||||
printf("fetch\n");
|
||||
printf("option\n");
|
||||
printf("push\n");
|
||||
printf("\n");
|
||||
fflush(stdout);
|
||||
} else {
|
||||
|
|
|
@ -8,7 +8,8 @@ struct send_pack_args {
|
|||
force_update:1,
|
||||
use_thin_pack:1,
|
||||
use_ofs_delta:1,
|
||||
dry_run:1;
|
||||
dry_run:1,
|
||||
stateless_rpc:1;
|
||||
};
|
||||
|
||||
int send_pack(struct send_pack_args *args,
|
||||
|
|
11
sideband.c
11
sideband.c
|
@ -135,9 +135,14 @@ ssize_t send_sideband(int fd, int band, const char *data, ssize_t sz, int packet
|
|||
n = sz;
|
||||
if (packet_max - 5 < n)
|
||||
n = packet_max - 5;
|
||||
sprintf(hdr, "%04x", n + 5);
|
||||
hdr[4] = band;
|
||||
safe_write(fd, hdr, 5);
|
||||
if (0 <= band) {
|
||||
sprintf(hdr, "%04x", n + 5);
|
||||
hdr[4] = band;
|
||||
safe_write(fd, hdr, 5);
|
||||
} else {
|
||||
sprintf(hdr, "%04x", n + 4);
|
||||
safe_write(fd, hdr, 4);
|
||||
}
|
||||
safe_write(fd, p, n);
|
||||
p += n;
|
||||
sz -= n;
|
||||
|
|
|
@ -8,6 +8,28 @@ ErrorLog error.log
|
|||
<IfModule !mod_log_config.c>
|
||||
LoadModule log_config_module modules/mod_log_config.so
|
||||
</IfModule>
|
||||
<IfModule !mod_alias.c>
|
||||
LoadModule alias_module modules/mod_alias.so
|
||||
</IfModule>
|
||||
<IfModule !mod_cgi.c>
|
||||
LoadModule cgi_module modules/mod_cgi.so
|
||||
</IfModule>
|
||||
<IfModule !mod_env.c>
|
||||
LoadModule env_module modules/mod_env.so
|
||||
</IfModule>
|
||||
|
||||
Alias /dumb/ www/
|
||||
|
||||
<Location /smart/>
|
||||
SetEnv GIT_EXEC_PATH ${GIT_EXEC_PATH}
|
||||
</Location>
|
||||
ScriptAlias /smart/ ${GIT_EXEC_PATH}/git-http-backend/
|
||||
<Directory ${GIT_EXEC_PATH}>
|
||||
Options None
|
||||
</Directory>
|
||||
<Files ${GIT_EXEC_PATH}/git-http-backend>
|
||||
Options ExecCGI
|
||||
</Files>
|
||||
|
||||
<IfDefine SSL>
|
||||
LoadModule ssl_module modules/mod_ssl.so
|
||||
|
@ -26,7 +48,7 @@ SSLEngine On
|
|||
LoadModule dav_fs_module modules/mod_dav_fs.so
|
||||
|
||||
DAVLockDB DAVLock
|
||||
<Location />
|
||||
<Location /dumb/>
|
||||
Dav on
|
||||
</Location>
|
||||
</IfDefine>
|
||||
|
|
|
@ -3,23 +3,22 @@
|
|||
# Copyright (c) 2008 Clemens Buchacher <drizzd@aon.at>
|
||||
#
|
||||
|
||||
test_description='test http-push
|
||||
test_description='test WebDAV http-push
|
||||
|
||||
This test runs various sanity checks on http-push.'
|
||||
|
||||
. ./test-lib.sh
|
||||
|
||||
ROOT_PATH="$PWD"
|
||||
LIB_HTTPD_DAV=t
|
||||
LIB_HTTPD_PORT=${LIB_HTTPD_PORT-'5540'}
|
||||
|
||||
if git http-push > /dev/null 2>&1 || [ $? -eq 128 ]
|
||||
then
|
||||
say "skipping test, USE_CURL_MULTI is not defined"
|
||||
test_done
|
||||
fi
|
||||
|
||||
LIB_HTTPD_DAV=t
|
||||
LIB_HTTPD_PORT=${LIB_HTTPD_PORT-'5540'}
|
||||
. "$TEST_DIRECTORY"/lib-httpd.sh
|
||||
ROOT_PATH="$PWD"
|
||||
start_httpd
|
||||
|
||||
test_expect_success 'setup remote repository' '
|
||||
|
@ -36,16 +35,17 @@ test_expect_success 'setup remote repository' '
|
|||
cd test_repo.git &&
|
||||
git --bare update-server-info &&
|
||||
mv hooks/post-update.sample hooks/post-update &&
|
||||
ORIG_HEAD=$(git rev-parse --verify HEAD) &&
|
||||
cd - &&
|
||||
mv test_repo.git "$HTTPD_DOCUMENT_ROOT_PATH"
|
||||
'
|
||||
|
||||
test_expect_success 'clone remote repository' '
|
||||
cd "$ROOT_PATH" &&
|
||||
git clone $HTTPD_URL/test_repo.git test_repo_clone
|
||||
git clone $HTTPD_URL/dumb/test_repo.git test_repo_clone
|
||||
'
|
||||
|
||||
test_expect_failure 'push to remote repository with packed refs' '
|
||||
test_expect_success 'push to remote repository with packed refs' '
|
||||
cd "$ROOT_PATH"/test_repo_clone &&
|
||||
: >path2 &&
|
||||
git add path2 &&
|
||||
|
@ -57,11 +57,15 @@ test_expect_failure 'push to remote repository with packed refs' '
|
|||
test $HEAD = $(git rev-parse --verify HEAD))
|
||||
'
|
||||
|
||||
test_expect_success ' push to remote repository with unpacked refs' '
|
||||
test_expect_success 'push already up-to-date' '
|
||||
git push
|
||||
'
|
||||
|
||||
test_expect_success 'push to remote repository with unpacked refs' '
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH"/test_repo.git &&
|
||||
rm packed-refs &&
|
||||
git update-ref refs/heads/master \
|
||||
0c973ae9bd51902a28466f3850b543fa66a6aaf4) &&
|
||||
git update-ref refs/heads/master $ORIG_HEAD &&
|
||||
git --bare update-server-info) &&
|
||||
git push &&
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH"/test_repo.git &&
|
||||
test $HEAD = $(git rev-parse --verify HEAD))
|
||||
|
@ -71,7 +75,7 @@ test_expect_success 'http-push fetches unpacked objects' '
|
|||
cp -R "$HTTPD_DOCUMENT_ROOT_PATH"/test_repo.git \
|
||||
"$HTTPD_DOCUMENT_ROOT_PATH"/test_repo_unpacked.git &&
|
||||
|
||||
git clone $HTTPD_URL/test_repo_unpacked.git \
|
||||
git clone $HTTPD_URL/dumb/test_repo_unpacked.git \
|
||||
"$ROOT_PATH"/fetch_unpacked &&
|
||||
|
||||
# By reset, we force git to retrieve the object
|
||||
|
@ -80,14 +84,14 @@ test_expect_success 'http-push fetches unpacked objects' '
|
|||
git remote rm origin &&
|
||||
git reflog expire --expire=0 --all &&
|
||||
git prune &&
|
||||
git push -f -v $HTTPD_URL/test_repo_unpacked.git master)
|
||||
git push -f -v $HTTPD_URL/dumb/test_repo_unpacked.git master)
|
||||
'
|
||||
|
||||
test_expect_success 'http-push fetches packed objects' '
|
||||
cp -R "$HTTPD_DOCUMENT_ROOT_PATH"/test_repo.git \
|
||||
"$HTTPD_DOCUMENT_ROOT_PATH"/test_repo_packed.git &&
|
||||
|
||||
git clone $HTTPD_URL/test_repo_packed.git \
|
||||
git clone $HTTPD_URL/dumb/test_repo_packed.git \
|
||||
"$ROOT_PATH"/test_repo_clone_packed &&
|
||||
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH"/test_repo_packed.git &&
|
||||
|
@ -100,7 +104,7 @@ test_expect_success 'http-push fetches packed objects' '
|
|||
git remote rm origin &&
|
||||
git reflog expire --expire=0 --all &&
|
||||
git prune &&
|
||||
git push -f -v $HTTPD_URL/test_repo_packed.git master)
|
||||
git push -f -v $HTTPD_URL/dumb/test_repo_packed.git master)
|
||||
'
|
||||
|
||||
test_expect_success 'create and delete remote branch' '
|
||||
|
@ -111,10 +115,7 @@ test_expect_success 'create and delete remote branch' '
|
|||
test_tick &&
|
||||
git commit -m dev &&
|
||||
git push origin dev &&
|
||||
git fetch &&
|
||||
git push origin :dev &&
|
||||
git branch -d -r origin/dev &&
|
||||
git fetch &&
|
||||
test_must_fail git show-ref --verify refs/remotes/origin/dev
|
||||
'
|
||||
|
||||
|
|
|
@ -0,0 +1,92 @@
|
|||
#!/bin/sh
|
||||
#
|
||||
# Copyright (c) 2008 Clemens Buchacher <drizzd@aon.at>
|
||||
#
|
||||
|
||||
test_description='test smart pushing over http via http-backend'
|
||||
. ./test-lib.sh
|
||||
|
||||
if test -n "$NO_CURL"; then
|
||||
say 'skipping test, git built without http support'
|
||||
test_done
|
||||
fi
|
||||
|
||||
ROOT_PATH="$PWD"
|
||||
LIB_HTTPD_PORT=${LIB_HTTPD_PORT-'5541'}
|
||||
. "$TEST_DIRECTORY"/lib-httpd.sh
|
||||
start_httpd
|
||||
|
||||
test_expect_success 'setup remote repository' '
|
||||
cd "$ROOT_PATH" &&
|
||||
mkdir test_repo &&
|
||||
cd test_repo &&
|
||||
git init &&
|
||||
: >path1 &&
|
||||
git add path1 &&
|
||||
test_tick &&
|
||||
git commit -m initial &&
|
||||
cd - &&
|
||||
git clone --bare test_repo test_repo.git &&
|
||||
cd test_repo.git &&
|
||||
git config http.receivepack true &&
|
||||
ORIG_HEAD=$(git rev-parse --verify HEAD) &&
|
||||
cd - &&
|
||||
mv test_repo.git "$HTTPD_DOCUMENT_ROOT_PATH"
|
||||
'
|
||||
|
||||
test_expect_success 'clone remote repository' '
|
||||
cd "$ROOT_PATH" &&
|
||||
git clone $HTTPD_URL/smart/test_repo.git test_repo_clone
|
||||
'
|
||||
|
||||
test_expect_success 'push to remote repository' '
|
||||
cd "$ROOT_PATH"/test_repo_clone &&
|
||||
: >path2 &&
|
||||
git add path2 &&
|
||||
test_tick &&
|
||||
git commit -m path2 &&
|
||||
HEAD=$(git rev-parse --verify HEAD) &&
|
||||
git push &&
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH"/test_repo.git &&
|
||||
test $HEAD = $(git rev-parse --verify HEAD))
|
||||
'
|
||||
|
||||
test_expect_success 'push already up-to-date' '
|
||||
git push
|
||||
'
|
||||
|
||||
test_expect_success 'create and delete remote branch' '
|
||||
cd "$ROOT_PATH"/test_repo_clone &&
|
||||
git checkout -b dev &&
|
||||
: >path3 &&
|
||||
git add path3 &&
|
||||
test_tick &&
|
||||
git commit -m dev &&
|
||||
git push origin dev &&
|
||||
git push origin :dev &&
|
||||
test_must_fail git show-ref --verify refs/remotes/origin/dev
|
||||
'
|
||||
|
||||
cat >exp <<EOF
|
||||
GET /smart/test_repo.git/info/refs?service=git-upload-pack HTTP/1.1 200
|
||||
POST /smart/test_repo.git/git-upload-pack HTTP/1.1 200
|
||||
GET /smart/test_repo.git/info/refs?service=git-receive-pack HTTP/1.1 200
|
||||
POST /smart/test_repo.git/git-receive-pack HTTP/1.1 200
|
||||
GET /smart/test_repo.git/info/refs?service=git-receive-pack HTTP/1.1 200
|
||||
GET /smart/test_repo.git/info/refs?service=git-receive-pack HTTP/1.1 200
|
||||
POST /smart/test_repo.git/git-receive-pack HTTP/1.1 200
|
||||
GET /smart/test_repo.git/info/refs?service=git-receive-pack HTTP/1.1 200
|
||||
POST /smart/test_repo.git/git-receive-pack HTTP/1.1 200
|
||||
EOF
|
||||
test_expect_success 'used receive-pack service' '
|
||||
sed -e "
|
||||
s/^.* \"//
|
||||
s/\"//
|
||||
s/ [1-9][0-9]*\$//
|
||||
s/^GET /GET /
|
||||
" >act <"$HTTPD_ROOT_PATH"/access.log &&
|
||||
test_cmp exp act
|
||||
'
|
||||
|
||||
stop_httpd
|
||||
test_done
|
|
@ -1,6 +1,6 @@
|
|||
#!/bin/sh
|
||||
|
||||
test_description='test fetching over http'
|
||||
test_description='test dumb fetching over http via static file'
|
||||
. ./test-lib.sh
|
||||
|
||||
if test -n "$NO_CURL"; then
|
||||
|
@ -30,7 +30,7 @@ test_expect_success 'create http-accessible bare repository' '
|
|||
'
|
||||
|
||||
test_expect_success 'clone http repository' '
|
||||
git clone $HTTPD_URL/repo.git clone &&
|
||||
git clone $HTTPD_URL/dumb/repo.git clone &&
|
||||
test_cmp file clone/file
|
||||
'
|
||||
|
||||
|
@ -58,7 +58,13 @@ test_expect_success 'fetch packed objects' '
|
|||
cd "$HTTPD_DOCUMENT_ROOT_PATH"/repo_pack.git &&
|
||||
git --bare repack &&
|
||||
git --bare prune-packed &&
|
||||
git clone $HTTPD_URL/repo_pack.git
|
||||
git clone $HTTPD_URL/dumb/repo_pack.git
|
||||
'
|
||||
|
||||
test_expect_success 'did not use upload-pack service' '
|
||||
grep '/git-upload-pack' <"$HTTPD_ROOT_PATH"/access.log >act
|
||||
: >exp
|
||||
test_cmp exp act
|
||||
'
|
||||
|
||||
stop_httpd
|
||||
|
|
|
@ -0,0 +1,105 @@
|
|||
#!/bin/sh
|
||||
|
||||
test_description='test smart fetching over http via http-backend'
|
||||
. ./test-lib.sh
|
||||
|
||||
if test -n "$NO_CURL"; then
|
||||
say 'skipping test, git built without http support'
|
||||
test_done
|
||||
fi
|
||||
|
||||
LIB_HTTPD_PORT=${LIB_HTTPD_PORT-'5551'}
|
||||
. "$TEST_DIRECTORY"/lib-httpd.sh
|
||||
start_httpd
|
||||
|
||||
test_expect_success 'setup repository' '
|
||||
echo content >file &&
|
||||
git add file &&
|
||||
git commit -m one
|
||||
'
|
||||
|
||||
test_expect_success 'create http-accessible bare repository' '
|
||||
mkdir "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
git --bare init
|
||||
) &&
|
||||
git remote add public "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
git push public master:master
|
||||
'
|
||||
|
||||
cat >exp <<EOF
|
||||
> GET /smart/repo.git/info/refs?service=git-upload-pack HTTP/1.1
|
||||
> Accept: */*
|
||||
> Pragma: no-cache
|
||||
< HTTP/1.1 200 OK
|
||||
< Pragma: no-cache
|
||||
< Cache-Control: no-cache, max-age=0, must-revalidate
|
||||
< Content-Type: application/x-git-upload-pack-advertisement
|
||||
> POST /smart/repo.git/git-upload-pack HTTP/1.1
|
||||
> Accept-Encoding: deflate, gzip
|
||||
> Content-Type: application/x-git-upload-pack-request
|
||||
> Accept: application/x-git-upload-pack-response
|
||||
> Content-Length: xxx
|
||||
< HTTP/1.1 200 OK
|
||||
< Pragma: no-cache
|
||||
< Cache-Control: no-cache, max-age=0, must-revalidate
|
||||
< Content-Type: application/x-git-upload-pack-result
|
||||
EOF
|
||||
test_expect_success 'clone http repository' '
|
||||
GIT_CURL_VERBOSE=1 git clone --quiet $HTTPD_URL/smart/repo.git clone 2>err &&
|
||||
test_cmp file clone/file &&
|
||||
tr '\''\015'\'' Q <err |
|
||||
sed -e "
|
||||
s/Q\$//
|
||||
/^[*] /d
|
||||
/^$/d
|
||||
/^< $/d
|
||||
|
||||
/^[^><]/{
|
||||
s/^/> /
|
||||
}
|
||||
|
||||
/^> User-Agent: /d
|
||||
/^> Host: /d
|
||||
/^> POST /,$ {
|
||||
/^> Accept: [*]\\/[*]/d
|
||||
}
|
||||
s/^> Content-Length: .*/> Content-Length: xxx/
|
||||
/^> 00..want /d
|
||||
/^> 00.*done/d
|
||||
|
||||
/^< Server: /d
|
||||
/^< Expires: /d
|
||||
/^< Date: /d
|
||||
/^< Content-Length: /d
|
||||
/^< Transfer-Encoding: /d
|
||||
" >act &&
|
||||
test_cmp exp act
|
||||
'
|
||||
|
||||
test_expect_success 'fetch changes via http' '
|
||||
echo content >>file &&
|
||||
git commit -a -m two &&
|
||||
git push public
|
||||
(cd clone && git pull) &&
|
||||
test_cmp file clone/file
|
||||
'
|
||||
|
||||
cat >exp <<EOF
|
||||
GET /smart/repo.git/info/refs?service=git-upload-pack HTTP/1.1 200
|
||||
POST /smart/repo.git/git-upload-pack HTTP/1.1 200
|
||||
GET /smart/repo.git/info/refs?service=git-upload-pack HTTP/1.1 200
|
||||
POST /smart/repo.git/git-upload-pack HTTP/1.1 200
|
||||
EOF
|
||||
test_expect_success 'used upload-pack service' '
|
||||
sed -e "
|
||||
s/^.* \"//
|
||||
s/\"//
|
||||
s/ [1-9][0-9]*\$//
|
||||
s/^GET /GET /
|
||||
" >act <"$HTTPD_ROOT_PATH"/access.log &&
|
||||
test_cmp exp act
|
||||
'
|
||||
|
||||
stop_httpd
|
||||
test_done
|
|
@ -0,0 +1,260 @@
|
|||
#!/bin/sh
|
||||
|
||||
test_description='test git-http-backend'
|
||||
. ./test-lib.sh
|
||||
|
||||
if test -n "$NO_CURL"; then
|
||||
say 'skipping test, git built without http support'
|
||||
test_done
|
||||
fi
|
||||
|
||||
LIB_HTTPD_PORT=${LIB_HTTPD_PORT-'5560'}
|
||||
. "$TEST_DIRECTORY"/lib-httpd.sh
|
||||
start_httpd
|
||||
|
||||
find_file() {
|
||||
cd "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
find $1 -type f |
|
||||
sed -e 1q
|
||||
}
|
||||
|
||||
config() {
|
||||
git --git-dir="$HTTPD_DOCUMENT_ROOT_PATH/repo.git" config $1 $2
|
||||
}
|
||||
|
||||
GET() {
|
||||
curl --include "$HTTPD_URL/smart/repo.git/$1" >out 2>/dev/null &&
|
||||
tr '\015' Q <out |
|
||||
sed '
|
||||
s/Q$//
|
||||
1q
|
||||
' >act &&
|
||||
echo "HTTP/1.1 $2" >exp &&
|
||||
test_cmp exp act
|
||||
}
|
||||
|
||||
POST() {
|
||||
curl --include --data "$2" \
|
||||
--header "Content-Type: application/x-$1-request" \
|
||||
"$HTTPD_URL/smart/repo.git/$1" >out 2>/dev/null &&
|
||||
tr '\015' Q <out |
|
||||
sed '
|
||||
s/Q$//
|
||||
1q
|
||||
' >act &&
|
||||
echo "HTTP/1.1 $3" >exp &&
|
||||
test_cmp exp act
|
||||
}
|
||||
|
||||
log_div() {
|
||||
echo >>"$HTTPD_ROOT_PATH"/access.log
|
||||
echo "### $1" >>"$HTTPD_ROOT_PATH"/access.log
|
||||
echo "###" >>"$HTTPD_ROOT_PATH"/access.log
|
||||
}
|
||||
|
||||
test_expect_success 'setup repository' '
|
||||
echo content >file &&
|
||||
git add file &&
|
||||
git commit -m one &&
|
||||
|
||||
mkdir "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
git --bare init &&
|
||||
: >objects/info/alternates &&
|
||||
: >objects/info/http-alternates
|
||||
) &&
|
||||
git remote add public "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
git push public master:master &&
|
||||
|
||||
(cd "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
|
||||
git repack -a -d
|
||||
) &&
|
||||
|
||||
echo other >file &&
|
||||
git add file &&
|
||||
git commit -m two &&
|
||||
git push public master:master &&
|
||||
|
||||
LOOSE_URL=$(find_file objects/??) &&
|
||||
PACK_URL=$(find_file objects/pack/*.pack) &&
|
||||
IDX_URL=$(find_file objects/pack/*.idx)
|
||||
'
|
||||
|
||||
get_static_files() {
|
||||
GET HEAD "$1" &&
|
||||
GET info/refs "$1" &&
|
||||
GET objects/info/packs "$1" &&
|
||||
GET objects/info/alternates "$1" &&
|
||||
GET objects/info/http-alternates "$1" &&
|
||||
GET $LOOSE_URL "$1" &&
|
||||
GET $PACK_URL "$1" &&
|
||||
GET $IDX_URL "$1"
|
||||
}
|
||||
|
||||
test_expect_success 'direct refs/heads/master not found' '
|
||||
log_div "refs/heads/master"
|
||||
GET refs/heads/master "404 Not Found"
|
||||
'
|
||||
test_expect_success 'static file is ok' '
|
||||
log_div "getanyfile default"
|
||||
get_static_files "200 OK"
|
||||
'
|
||||
test_expect_success 'static file if http.getanyfile true is ok' '
|
||||
log_div "getanyfile true"
|
||||
config http.getanyfile true &&
|
||||
get_static_files "200 OK"
|
||||
'
|
||||
test_expect_success 'static file if http.getanyfile false fails' '
|
||||
log_div "getanyfile false"
|
||||
config http.getanyfile false &&
|
||||
get_static_files "403 Forbidden"
|
||||
'
|
||||
|
||||
test_expect_success 'http.uploadpack default enabled' '
|
||||
log_div "uploadpack default"
|
||||
GET info/refs?service=git-upload-pack "200 OK" &&
|
||||
POST git-upload-pack 0000 "200 OK"
|
||||
'
|
||||
test_expect_success 'http.uploadpack true' '
|
||||
log_div "uploadpack true"
|
||||
config http.uploadpack true &&
|
||||
GET info/refs?service=git-upload-pack "200 OK" &&
|
||||
POST git-upload-pack 0000 "200 OK"
|
||||
'
|
||||
test_expect_success 'http.uploadpack false' '
|
||||
log_div "uploadpack false"
|
||||
config http.uploadpack false &&
|
||||
GET info/refs?service=git-upload-pack "403 Forbidden" &&
|
||||
POST git-upload-pack 0000 "403 Forbidden"
|
||||
'
|
||||
|
||||
test_expect_success 'http.receivepack default disabled' '
|
||||
log_div "receivepack default"
|
||||
GET info/refs?service=git-receive-pack "403 Forbidden" &&
|
||||
POST git-receive-pack 0000 "403 Forbidden"
|
||||
'
|
||||
test_expect_success 'http.receivepack true' '
|
||||
log_div "receivepack true"
|
||||
config http.receivepack true &&
|
||||
GET info/refs?service=git-receive-pack "200 OK" &&
|
||||
POST git-receive-pack 0000 "200 OK"
|
||||
'
|
||||
test_expect_success 'http.receivepack false' '
|
||||
log_div "receivepack false"
|
||||
config http.receivepack false &&
|
||||
GET info/refs?service=git-receive-pack "403 Forbidden" &&
|
||||
POST git-receive-pack 0000 "403 Forbidden"
|
||||
'
|
||||
|
||||
run_backend() {
|
||||
REQUEST_METHOD=GET \
|
||||
GIT_PROJECT_ROOT="$HTTPD_DOCUMENT_ROOT_PATH" \
|
||||
PATH_INFO="$2" \
|
||||
git http-backend >act.out 2>act.err
|
||||
}
|
||||
|
||||
path_info() {
|
||||
if test $1 = 0; then
|
||||
run_backend "$2"
|
||||
else
|
||||
test_must_fail run_backend "$2" &&
|
||||
echo "fatal: '$2': aliased" >exp.err &&
|
||||
test_cmp exp.err act.err
|
||||
fi
|
||||
}
|
||||
|
||||
test_expect_success 'http-backend blocks bad PATH_INFO' '
|
||||
config http.getanyfile true &&
|
||||
|
||||
run_backend 0 /repo.git/HEAD &&
|
||||
|
||||
run_backend 1 /repo.git/../HEAD &&
|
||||
run_backend 1 /../etc/passwd &&
|
||||
run_backend 1 ../etc/passwd &&
|
||||
run_backend 1 /etc//passwd &&
|
||||
run_backend 1 /etc/./passwd &&
|
||||
run_backend 1 /etc/.../passwd &&
|
||||
run_backend 1 //domain/data.txt
|
||||
'
|
||||
|
||||
cat >exp <<EOF
|
||||
|
||||
### refs/heads/master
|
||||
###
|
||||
GET /smart/repo.git/refs/heads/master HTTP/1.1 404 -
|
||||
|
||||
### getanyfile default
|
||||
###
|
||||
GET /smart/repo.git/HEAD HTTP/1.1 200
|
||||
GET /smart/repo.git/info/refs HTTP/1.1 200
|
||||
GET /smart/repo.git/objects/info/packs HTTP/1.1 200
|
||||
GET /smart/repo.git/objects/info/alternates HTTP/1.1 200 -
|
||||
GET /smart/repo.git/objects/info/http-alternates HTTP/1.1 200 -
|
||||
GET /smart/repo.git/$LOOSE_URL HTTP/1.1 200
|
||||
GET /smart/repo.git/$PACK_URL HTTP/1.1 200
|
||||
GET /smart/repo.git/$IDX_URL HTTP/1.1 200
|
||||
|
||||
### getanyfile true
|
||||
###
|
||||
GET /smart/repo.git/HEAD HTTP/1.1 200
|
||||
GET /smart/repo.git/info/refs HTTP/1.1 200
|
||||
GET /smart/repo.git/objects/info/packs HTTP/1.1 200
|
||||
GET /smart/repo.git/objects/info/alternates HTTP/1.1 200 -
|
||||
GET /smart/repo.git/objects/info/http-alternates HTTP/1.1 200 -
|
||||
GET /smart/repo.git/$LOOSE_URL HTTP/1.1 200
|
||||
GET /smart/repo.git/$PACK_URL HTTP/1.1 200
|
||||
GET /smart/repo.git/$IDX_URL HTTP/1.1 200
|
||||
|
||||
### getanyfile false
|
||||
###
|
||||
GET /smart/repo.git/HEAD HTTP/1.1 403 -
|
||||
GET /smart/repo.git/info/refs HTTP/1.1 403 -
|
||||
GET /smart/repo.git/objects/info/packs HTTP/1.1 403 -
|
||||
GET /smart/repo.git/objects/info/alternates HTTP/1.1 403 -
|
||||
GET /smart/repo.git/objects/info/http-alternates HTTP/1.1 403 -
|
||||
GET /smart/repo.git/$LOOSE_URL HTTP/1.1 403 -
|
||||
GET /smart/repo.git/$PACK_URL HTTP/1.1 403 -
|
||||
GET /smart/repo.git/$IDX_URL HTTP/1.1 403 -
|
||||
|
||||
### uploadpack default
|
||||
###
|
||||
GET /smart/repo.git/info/refs?service=git-upload-pack HTTP/1.1 200
|
||||
POST /smart/repo.git/git-upload-pack HTTP/1.1 200 -
|
||||
|
||||
### uploadpack true
|
||||
###
|
||||
GET /smart/repo.git/info/refs?service=git-upload-pack HTTP/1.1 200
|
||||
POST /smart/repo.git/git-upload-pack HTTP/1.1 200 -
|
||||
|
||||
### uploadpack false
|
||||
###
|
||||
GET /smart/repo.git/info/refs?service=git-upload-pack HTTP/1.1 403 -
|
||||
POST /smart/repo.git/git-upload-pack HTTP/1.1 403 -
|
||||
|
||||
### receivepack default
|
||||
###
|
||||
GET /smart/repo.git/info/refs?service=git-receive-pack HTTP/1.1 403 -
|
||||
POST /smart/repo.git/git-receive-pack HTTP/1.1 403 -
|
||||
|
||||
### receivepack true
|
||||
###
|
||||
GET /smart/repo.git/info/refs?service=git-receive-pack HTTP/1.1 200
|
||||
POST /smart/repo.git/git-receive-pack HTTP/1.1 200 -
|
||||
|
||||
### receivepack false
|
||||
###
|
||||
GET /smart/repo.git/info/refs?service=git-receive-pack HTTP/1.1 403 -
|
||||
POST /smart/repo.git/git-receive-pack HTTP/1.1 403 -
|
||||
EOF
|
||||
test_expect_success 'server request log matches test results' '
|
||||
sed -e "
|
||||
s/^.* \"//
|
||||
s/\"//
|
||||
s/ [1-9][0-9]*\$//
|
||||
s/^GET /GET /
|
||||
" >act <"$HTTPD_ROOT_PATH"/access.log &&
|
||||
test_cmp exp act
|
||||
'
|
||||
|
||||
stop_httpd
|
||||
test_done
|
|
@ -1,16 +1,20 @@
|
|||
#include "cache.h"
|
||||
#include "transport.h"
|
||||
|
||||
#include "quote.h"
|
||||
#include "run-command.h"
|
||||
#include "commit.h"
|
||||
#include "diff.h"
|
||||
#include "revision.h"
|
||||
#include "quote.h"
|
||||
|
||||
struct helper_data
|
||||
{
|
||||
const char *name;
|
||||
struct child_process *helper;
|
||||
unsigned fetch : 1;
|
||||
FILE *out;
|
||||
unsigned fetch : 1,
|
||||
option : 1,
|
||||
push : 1;
|
||||
};
|
||||
|
||||
static struct child_process *get_helper(struct transport *transport)
|
||||
|
@ -18,7 +22,6 @@ static struct child_process *get_helper(struct transport *transport)
|
|||
struct helper_data *data = transport->data;
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
struct child_process *helper;
|
||||
FILE *file;
|
||||
|
||||
if (data->helper)
|
||||
return data->helper;
|
||||
|
@ -39,15 +42,19 @@ static struct child_process *get_helper(struct transport *transport)
|
|||
|
||||
write_str_in_full(helper->in, "capabilities\n");
|
||||
|
||||
file = xfdopen(helper->out, "r");
|
||||
data->out = xfdopen(helper->out, "r");
|
||||
while (1) {
|
||||
if (strbuf_getline(&buf, file, '\n') == EOF)
|
||||
if (strbuf_getline(&buf, data->out, '\n') == EOF)
|
||||
exit(128); /* child died, message supplied already */
|
||||
|
||||
if (!*buf.buf)
|
||||
break;
|
||||
if (!strcmp(buf.buf, "fetch"))
|
||||
data->fetch = 1;
|
||||
if (!strcmp(buf.buf, "option"))
|
||||
data->option = 1;
|
||||
if (!strcmp(buf.buf, "push"))
|
||||
data->push = 1;
|
||||
}
|
||||
return data->helper;
|
||||
}
|
||||
|
@ -58,23 +65,104 @@ static int disconnect_helper(struct transport *transport)
|
|||
if (data->helper) {
|
||||
write_str_in_full(data->helper->in, "\n");
|
||||
close(data->helper->in);
|
||||
fclose(data->out);
|
||||
finish_command(data->helper);
|
||||
free((char *)data->helper->argv[0]);
|
||||
free(data->helper->argv);
|
||||
free(data->helper);
|
||||
data->helper = NULL;
|
||||
}
|
||||
free(data);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const char *unsupported_options[] = {
|
||||
TRANS_OPT_UPLOADPACK,
|
||||
TRANS_OPT_RECEIVEPACK,
|
||||
TRANS_OPT_THIN,
|
||||
TRANS_OPT_KEEP
|
||||
};
|
||||
static const char *boolean_options[] = {
|
||||
TRANS_OPT_THIN,
|
||||
TRANS_OPT_KEEP,
|
||||
TRANS_OPT_FOLLOWTAGS
|
||||
};
|
||||
|
||||
static int set_helper_option(struct transport *transport,
|
||||
const char *name, const char *value)
|
||||
{
|
||||
struct helper_data *data = transport->data;
|
||||
struct child_process *helper = get_helper(transport);
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
int i, ret, is_bool = 0;
|
||||
|
||||
if (!data->option)
|
||||
return 1;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(unsupported_options); i++) {
|
||||
if (!strcmp(name, unsupported_options[i]))
|
||||
return 1;
|
||||
}
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(boolean_options); i++) {
|
||||
if (!strcmp(name, boolean_options[i])) {
|
||||
is_bool = 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
strbuf_addf(&buf, "option %s ", name);
|
||||
if (is_bool)
|
||||
strbuf_addstr(&buf, value ? "true" : "false");
|
||||
else
|
||||
quote_c_style(value, &buf, NULL, 0);
|
||||
strbuf_addch(&buf, '\n');
|
||||
|
||||
if (write_in_full(helper->in, buf.buf, buf.len) != buf.len)
|
||||
die_errno("cannot send option to %s", data->name);
|
||||
|
||||
strbuf_reset(&buf);
|
||||
if (strbuf_getline(&buf, data->out, '\n') == EOF)
|
||||
exit(128); /* child died, message supplied already */
|
||||
|
||||
if (!strcmp(buf.buf, "ok"))
|
||||
ret = 0;
|
||||
else if (!prefixcmp(buf.buf, "error")) {
|
||||
ret = -1;
|
||||
} else if (!strcmp(buf.buf, "unsupported"))
|
||||
ret = 1;
|
||||
else {
|
||||
warning("%s unexpectedly said: '%s'", data->name, buf.buf);
|
||||
ret = 1;
|
||||
}
|
||||
strbuf_release(&buf);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static void standard_options(struct transport *t)
|
||||
{
|
||||
char buf[16];
|
||||
int n;
|
||||
int v = t->verbose;
|
||||
int no_progress = v < 0 || (!t->progress && !isatty(1));
|
||||
|
||||
set_helper_option(t, "progress", !no_progress ? "true" : "false");
|
||||
|
||||
n = snprintf(buf, sizeof(buf), "%d", v + 1);
|
||||
if (n >= sizeof(buf))
|
||||
die("impossibly large verbosity value");
|
||||
set_helper_option(t, "verbosity", buf);
|
||||
}
|
||||
|
||||
static int fetch_with_fetch(struct transport *transport,
|
||||
int nr_heads, const struct ref **to_fetch)
|
||||
{
|
||||
struct child_process *helper = get_helper(transport);
|
||||
FILE *file = xfdopen(helper->out, "r");
|
||||
struct helper_data *data = transport->data;
|
||||
int i;
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
|
||||
standard_options(transport);
|
||||
|
||||
for (i = 0; i < nr_heads; i++) {
|
||||
const struct ref *posn = to_fetch[i];
|
||||
if (posn->status & REF_STATUS_UPTODATE)
|
||||
|
@ -82,12 +170,30 @@ static int fetch_with_fetch(struct transport *transport,
|
|||
|
||||
strbuf_addf(&buf, "fetch %s %s\n",
|
||||
sha1_to_hex(posn->old_sha1), posn->name);
|
||||
write_in_full(helper->in, buf.buf, buf.len);
|
||||
strbuf_reset(&buf);
|
||||
|
||||
if (strbuf_getline(&buf, file, '\n') == EOF)
|
||||
exit(128); /* child died, message supplied already */
|
||||
}
|
||||
|
||||
strbuf_addch(&buf, '\n');
|
||||
if (write_in_full(data->helper->in, buf.buf, buf.len) != buf.len)
|
||||
die_errno("cannot send fetch to %s", data->name);
|
||||
|
||||
while (1) {
|
||||
strbuf_reset(&buf);
|
||||
if (strbuf_getline(&buf, data->out, '\n') == EOF)
|
||||
exit(128); /* child died, message supplied already */
|
||||
|
||||
if (!prefixcmp(buf.buf, "lock ")) {
|
||||
const char *name = buf.buf + 5;
|
||||
if (transport->pack_lockfile)
|
||||
warning("%s also locked %s", data->name, name);
|
||||
else
|
||||
transport->pack_lockfile = xstrdup(name);
|
||||
}
|
||||
else if (!buf.len)
|
||||
break;
|
||||
else
|
||||
warning("%s unexpectedly said: '%s'", data->name, buf.buf);
|
||||
}
|
||||
strbuf_release(&buf);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -111,23 +217,151 @@ static int fetch(struct transport *transport,
|
|||
return -1;
|
||||
}
|
||||
|
||||
static int push_refs(struct transport *transport,
|
||||
struct ref *remote_refs, int flags)
|
||||
{
|
||||
int force_all = flags & TRANSPORT_PUSH_FORCE;
|
||||
int mirror = flags & TRANSPORT_PUSH_MIRROR;
|
||||
struct helper_data *data = transport->data;
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
struct child_process *helper;
|
||||
struct ref *ref;
|
||||
|
||||
if (!remote_refs)
|
||||
return 0;
|
||||
|
||||
helper = get_helper(transport);
|
||||
if (!data->push)
|
||||
return 1;
|
||||
|
||||
for (ref = remote_refs; ref; ref = ref->next) {
|
||||
if (ref->peer_ref)
|
||||
hashcpy(ref->new_sha1, ref->peer_ref->new_sha1);
|
||||
else if (!mirror)
|
||||
continue;
|
||||
|
||||
ref->deletion = is_null_sha1(ref->new_sha1);
|
||||
if (!ref->deletion &&
|
||||
!hashcmp(ref->old_sha1, ref->new_sha1)) {
|
||||
ref->status = REF_STATUS_UPTODATE;
|
||||
continue;
|
||||
}
|
||||
|
||||
if (force_all)
|
||||
ref->force = 1;
|
||||
|
||||
strbuf_addstr(&buf, "push ");
|
||||
if (!ref->deletion) {
|
||||
if (ref->force)
|
||||
strbuf_addch(&buf, '+');
|
||||
if (ref->peer_ref)
|
||||
strbuf_addstr(&buf, ref->peer_ref->name);
|
||||
else
|
||||
strbuf_addstr(&buf, sha1_to_hex(ref->new_sha1));
|
||||
}
|
||||
strbuf_addch(&buf, ':');
|
||||
strbuf_addstr(&buf, ref->name);
|
||||
strbuf_addch(&buf, '\n');
|
||||
}
|
||||
if (buf.len == 0)
|
||||
return 0;
|
||||
|
||||
transport->verbose = flags & TRANSPORT_PUSH_VERBOSE ? 1 : 0;
|
||||
standard_options(transport);
|
||||
|
||||
if (flags & TRANSPORT_PUSH_DRY_RUN) {
|
||||
if (set_helper_option(transport, "dry-run", "true") != 0)
|
||||
die("helper %s does not support dry-run", data->name);
|
||||
}
|
||||
|
||||
strbuf_addch(&buf, '\n');
|
||||
if (write_in_full(helper->in, buf.buf, buf.len) != buf.len)
|
||||
exit(128);
|
||||
|
||||
ref = remote_refs;
|
||||
while (1) {
|
||||
char *refname, *msg;
|
||||
int status;
|
||||
|
||||
strbuf_reset(&buf);
|
||||
if (strbuf_getline(&buf, data->out, '\n') == EOF)
|
||||
exit(128); /* child died, message supplied already */
|
||||
if (!buf.len)
|
||||
break;
|
||||
|
||||
if (!prefixcmp(buf.buf, "ok ")) {
|
||||
status = REF_STATUS_OK;
|
||||
refname = buf.buf + 3;
|
||||
} else if (!prefixcmp(buf.buf, "error ")) {
|
||||
status = REF_STATUS_REMOTE_REJECT;
|
||||
refname = buf.buf + 6;
|
||||
} else
|
||||
die("expected ok/error, helper said '%s'\n", buf.buf);
|
||||
|
||||
msg = strchr(refname, ' ');
|
||||
if (msg) {
|
||||
struct strbuf msg_buf = STRBUF_INIT;
|
||||
const char *end;
|
||||
|
||||
*msg++ = '\0';
|
||||
if (!unquote_c_style(&msg_buf, msg, &end))
|
||||
msg = strbuf_detach(&msg_buf, NULL);
|
||||
else
|
||||
msg = xstrdup(msg);
|
||||
strbuf_release(&msg_buf);
|
||||
|
||||
if (!strcmp(msg, "no match")) {
|
||||
status = REF_STATUS_NONE;
|
||||
free(msg);
|
||||
msg = NULL;
|
||||
}
|
||||
else if (!strcmp(msg, "up to date")) {
|
||||
status = REF_STATUS_UPTODATE;
|
||||
free(msg);
|
||||
msg = NULL;
|
||||
}
|
||||
else if (!strcmp(msg, "non-fast forward")) {
|
||||
status = REF_STATUS_REJECT_NONFASTFORWARD;
|
||||
free(msg);
|
||||
msg = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
if (ref)
|
||||
ref = find_ref_by_name(ref, refname);
|
||||
if (!ref)
|
||||
ref = find_ref_by_name(remote_refs, refname);
|
||||
if (!ref) {
|
||||
warning("helper reported unexpected status of %s", refname);
|
||||
continue;
|
||||
}
|
||||
|
||||
ref->status = status;
|
||||
ref->remote_status = msg;
|
||||
}
|
||||
strbuf_release(&buf);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct ref *get_refs_list(struct transport *transport, int for_push)
|
||||
{
|
||||
struct helper_data *data = transport->data;
|
||||
struct child_process *helper;
|
||||
struct ref *ret = NULL;
|
||||
struct ref **tail = &ret;
|
||||
struct ref *posn;
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
FILE *file;
|
||||
|
||||
helper = get_helper(transport);
|
||||
|
||||
write_str_in_full(helper->in, "list\n");
|
||||
if (data->push && for_push)
|
||||
write_str_in_full(helper->in, "list for-push\n");
|
||||
else
|
||||
write_str_in_full(helper->in, "list\n");
|
||||
|
||||
file = xfdopen(helper->out, "r");
|
||||
while (1) {
|
||||
char *eov, *eon;
|
||||
if (strbuf_getline(&buf, file, '\n') == EOF)
|
||||
if (strbuf_getline(&buf, data->out, '\n') == EOF)
|
||||
exit(128); /* child died, message supplied already */
|
||||
|
||||
if (!*buf.buf)
|
||||
|
@ -161,8 +395,10 @@ int transport_helper_init(struct transport *transport, const char *name)
|
|||
data->name = name;
|
||||
|
||||
transport->data = data;
|
||||
transport->set_option = set_helper_option;
|
||||
transport->get_refs_list = get_refs_list;
|
||||
transport->fetch = fetch;
|
||||
transport->push_refs = push_refs;
|
||||
transport->disconnect = disconnect_helper;
|
||||
return 0;
|
||||
}
|
||||
|
|
32
transport.c
32
transport.c
|
@ -349,35 +349,6 @@ static int rsync_transport_push(struct transport *transport,
|
|||
return result;
|
||||
}
|
||||
|
||||
#ifndef NO_CURL
|
||||
static int curl_transport_push(struct transport *transport, int refspec_nr, const char **refspec, int flags)
|
||||
{
|
||||
const char **argv;
|
||||
int argc;
|
||||
|
||||
if (flags & TRANSPORT_PUSH_MIRROR)
|
||||
return error("http transport does not support mirror mode");
|
||||
|
||||
argv = xmalloc((refspec_nr + 12) * sizeof(char *));
|
||||
argv[0] = "http-push";
|
||||
argc = 1;
|
||||
if (flags & TRANSPORT_PUSH_ALL)
|
||||
argv[argc++] = "--all";
|
||||
if (flags & TRANSPORT_PUSH_FORCE)
|
||||
argv[argc++] = "--force";
|
||||
if (flags & TRANSPORT_PUSH_DRY_RUN)
|
||||
argv[argc++] = "--dry-run";
|
||||
if (flags & TRANSPORT_PUSH_VERBOSE)
|
||||
argv[argc++] = "--verbose";
|
||||
argv[argc++] = transport->url;
|
||||
while (refspec_nr--)
|
||||
argv[argc++] = *refspec++;
|
||||
argv[argc] = NULL;
|
||||
return !!run_command_v_opt(argv, RUN_GIT_CMD);
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
struct bundle_transport_data {
|
||||
int fd;
|
||||
struct bundle_header header;
|
||||
|
@ -760,6 +731,7 @@ static int git_transport_push(struct transport *transport, struct ref *remote_re
|
|||
NULL);
|
||||
}
|
||||
|
||||
memset(&args, 0, sizeof(args));
|
||||
args.send_mirror = !!(flags & TRANSPORT_PUSH_MIRROR);
|
||||
args.force_update = !!(flags & TRANSPORT_PUSH_FORCE);
|
||||
args.use_thin_pack = data->thin;
|
||||
|
@ -829,8 +801,6 @@ struct transport *transport_get(struct remote *remote, const char *url)
|
|||
transport_helper_init(ret, "curl");
|
||||
#ifdef NO_CURL
|
||||
error("git was compiled without libcurl support.");
|
||||
#else
|
||||
ret->push = curl_transport_push;
|
||||
#endif
|
||||
|
||||
} else if (is_local(url) && is_file(url)) {
|
||||
|
|
|
@ -25,7 +25,7 @@ struct transport {
|
|||
|
||||
int (*disconnect)(struct transport *connection);
|
||||
char *pack_lockfile;
|
||||
signed verbose : 2;
|
||||
signed verbose : 3;
|
||||
/* Force progress even if the output is not a tty */
|
||||
unsigned progress : 1;
|
||||
};
|
||||
|
|
|
@ -39,6 +39,8 @@ static unsigned int timeout;
|
|||
*/
|
||||
static int use_sideband;
|
||||
static int debug_fd;
|
||||
static int advertise_refs;
|
||||
static int stateless_rpc;
|
||||
|
||||
static void reset_timeout(void)
|
||||
{
|
||||
|
@ -500,7 +502,7 @@ static int get_common_commits(void)
|
|||
{
|
||||
static char line[1000];
|
||||
unsigned char sha1[20];
|
||||
char hex[41], last_hex[41];
|
||||
char last_hex[41];
|
||||
|
||||
save_commit_buffer = 0;
|
||||
|
||||
|
@ -511,25 +513,30 @@ static int get_common_commits(void)
|
|||
if (!len) {
|
||||
if (have_obj.nr == 0 || multi_ack)
|
||||
packet_write(1, "NAK\n");
|
||||
if (stateless_rpc)
|
||||
exit(0);
|
||||
continue;
|
||||
}
|
||||
strip(line, len);
|
||||
if (!prefixcmp(line, "have ")) {
|
||||
switch (got_sha1(line+5, sha1)) {
|
||||
case -1: /* they have what we do not */
|
||||
if (multi_ack && ok_to_give_up())
|
||||
packet_write(1, "ACK %s continue\n",
|
||||
sha1_to_hex(sha1));
|
||||
if (multi_ack && ok_to_give_up()) {
|
||||
const char *hex = sha1_to_hex(sha1);
|
||||
if (multi_ack == 2)
|
||||
packet_write(1, "ACK %s ready\n", hex);
|
||||
else
|
||||
packet_write(1, "ACK %s continue\n", hex);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
memcpy(hex, sha1_to_hex(sha1), 41);
|
||||
if (multi_ack) {
|
||||
const char *msg = "ACK %s continue\n";
|
||||
packet_write(1, msg, hex);
|
||||
memcpy(last_hex, hex, 41);
|
||||
}
|
||||
memcpy(last_hex, sha1_to_hex(sha1), 41);
|
||||
if (multi_ack == 2)
|
||||
packet_write(1, "ACK %s common\n", last_hex);
|
||||
else if (multi_ack)
|
||||
packet_write(1, "ACK %s continue\n", last_hex);
|
||||
else if (have_obj.nr == 1)
|
||||
packet_write(1, "ACK %s\n", hex);
|
||||
packet_write(1, "ACK %s\n", last_hex);
|
||||
break;
|
||||
}
|
||||
continue;
|
||||
|
@ -589,7 +596,9 @@ static void receive_needs(void)
|
|||
get_sha1_hex(line+5, sha1_buf))
|
||||
die("git upload-pack: protocol error, "
|
||||
"expected to get sha, not '%s'", line);
|
||||
if (strstr(line+45, "multi_ack"))
|
||||
if (strstr(line+45, "multi_ack_detailed"))
|
||||
multi_ack = 2;
|
||||
else if (strstr(line+45, "multi_ack"))
|
||||
multi_ack = 1;
|
||||
if (strstr(line+45, "thin-pack"))
|
||||
use_thin_pack = 1;
|
||||
|
@ -683,7 +692,7 @@ static int send_ref(const char *refname, const unsigned char *sha1, int flag, vo
|
|||
{
|
||||
static const char *capabilities = "multi_ack thin-pack side-band"
|
||||
" side-band-64k ofs-delta shallow no-progress"
|
||||
" include-tag";
|
||||
" include-tag multi_ack_detailed";
|
||||
struct object *o = parse_object(sha1);
|
||||
|
||||
if (!o)
|
||||
|
@ -707,12 +716,32 @@ static int send_ref(const char *refname, const unsigned char *sha1, int flag, vo
|
|||
return 0;
|
||||
}
|
||||
|
||||
static int mark_our_ref(const char *refname, const unsigned char *sha1, int flag, void *cb_data)
|
||||
{
|
||||
struct object *o = parse_object(sha1);
|
||||
if (!o)
|
||||
die("git upload-pack: cannot find object %s:", sha1_to_hex(sha1));
|
||||
if (!(o->flags & OUR_REF)) {
|
||||
o->flags |= OUR_REF;
|
||||
nr_our_refs++;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void upload_pack(void)
|
||||
{
|
||||
reset_timeout();
|
||||
head_ref(send_ref, NULL);
|
||||
for_each_ref(send_ref, NULL);
|
||||
packet_flush(1);
|
||||
if (advertise_refs || !stateless_rpc) {
|
||||
reset_timeout();
|
||||
head_ref(send_ref, NULL);
|
||||
for_each_ref(send_ref, NULL);
|
||||
packet_flush(1);
|
||||
} else {
|
||||
head_ref(mark_our_ref, NULL);
|
||||
for_each_ref(mark_our_ref, NULL);
|
||||
}
|
||||
if (advertise_refs)
|
||||
return;
|
||||
|
||||
receive_needs();
|
||||
if (want_obj.nr) {
|
||||
get_common_commits();
|
||||
|
@ -734,6 +763,14 @@ int main(int argc, char **argv)
|
|||
|
||||
if (arg[0] != '-')
|
||||
break;
|
||||
if (!strcmp(arg, "--advertise-refs")) {
|
||||
advertise_refs = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--stateless-rpc")) {
|
||||
stateless_rpc = 1;
|
||||
continue;
|
||||
}
|
||||
if (!strcmp(arg, "--strict")) {
|
||||
strict = 1;
|
||||
continue;
|
||||
|
|
Loading…
Reference in New Issue