diff options
| author | Scott Chacon <schacon@gmail.com> | 2010-04-01 15:14:35 -0700 | 
|---|---|---|
| committer | Junio C Hamano <gitster@pobox.com> | 2010-04-01 23:24:59 -0700 | 
| commit | 42653c09c85015addc6fa8dd4d49cb250253412e (patch) | |
| tree | cd33603246f89be8c771ff51284f9833bc565efa /http.c | |
| parent | 890a13a45285ad44858add2ce2f74eb478f549c8 (diff) | |
| download | git-42653c09c85015addc6fa8dd4d49cb250253412e.tar.gz | |
Prompt for a username when an HTTP request 401s
When an HTTP request returns a 401, Git will currently fail with a
confusing message saying that it got a 401, which is not very
descriptive.
Currently if a user wants to use Git over HTTP, they have to use one
URL with the username in the URL (e.g. "http://user@host.com/repo.git")
for write access and another without the username for unauthenticated
read access (unless they want to be prompted for the password each
time). However, since the HTTP servers will return a 401 if an action
requires authentication, we can prompt for username and password if we
see this, allowing us to use a single URL for both purposes.
This patch changes http_request to prompt for the username and password,
then return HTTP_REAUTH so http_get_strbuf can try again.  If it gets
a 401 even when a user/pass is supplied, http_request will now return
HTTP_NOAUTH which remote_curl can then use to display a more
intelligent error message that is less confusing.
Signed-off-by: Scott Chacon <schacon@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
Diffstat (limited to 'http.c')
| -rw-r--r-- | http.c | 22 | 
1 files changed, 20 insertions, 2 deletions
| @@ -815,7 +815,21 @@ static int http_request(const char *url, void *result, int target, int options)  			ret = HTTP_OK;  		else if (missing_target(&results))  			ret = HTTP_MISSING_TARGET; -		else +		else if (results.http_code == 401) { +			if (user_name) { +				ret = HTTP_NOAUTH; +			} else { +				/* +				 * git_getpass is needed here because its very likely stdin/stdout are +				 * pipes to our parent process.  So we instead need to use /dev/tty, +				 * but that is non-portable.  Using git_getpass() can at least be stubbed +				 * on other platforms with a different implementation if/when necessary. +				 */ +				user_name = xstrdup(git_getpass("Username: ")); +				init_curl_http_auth(slot->curl); +				ret = HTTP_REAUTH; +			} +		} else  			ret = HTTP_ERROR;  	} else {  		error("Unable to start HTTP request for %s", url); @@ -831,7 +845,11 @@ static int http_request(const char *url, void *result, int target, int options)  int http_get_strbuf(const char *url, struct strbuf *result, int options)  { -	return http_request(url, result, HTTP_REQUEST_STRBUF, options); +	int http_ret = http_request(url, result, HTTP_REQUEST_STRBUF, options); +	if (http_ret == HTTP_REAUTH) { +		http_ret = http_request(url, result, HTTP_REQUEST_STRBUF, options); +	} +	return http_ret;  }  /* | 
