Skip to content

Commit

Permalink
Reworked redirect implementation to fully clone http request and re-u…
Browse files Browse the repository at this point in the history
…se it later

Now the skipped test from octokit#874 works!
Also had to fix the new ReturnsRenamedRepository test as the ionide repo was renamed again
  • Loading branch information
ryangribble authored and martinscholz83 committed Aug 11, 2016
1 parent 802091c commit 815f0d1
Show file tree
Hide file tree
Showing 4 changed files with 47 additions and 41 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -618,7 +618,7 @@ public async Task ReturnsRenamedRepository()

repository = await github.Repository.Get("fsprojects", "FSharp.Atom");

Assert.Equal("https://github.com/ionide/ionide-fsharp.git", repository.CloneUrl);
Assert.Equal("https://github.com/ionide/ionide-atom-fsharp.git", repository.CloneUrl);
Assert.False(repository.Private);
Assert.False(repository.Fork);

Expand Down
2 changes: 1 addition & 1 deletion Octokit.Tests.Integration/RedirectTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ public async Task ReturnsRedirectedRepository()
Assert.Equal(AccountType.User, repository.Owner.Type);
}

[IntegrationTest(Skip = "This test is super-unreliable right now - see https://github.com/octokit/octokit.net/issues/874 for discussion")]
[IntegrationTest]
public async Task CanCreateIssueOnRedirectedRepository()
{
var client = Helper.GetAuthenticatedClient();
Expand Down
2 changes: 0 additions & 2 deletions Octokit.Tests/Http/RedirectHandlerTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -122,8 +122,6 @@ public async Task Status301ShouldRedirectPOSTWithBody(HttpStatusCode statusCode)

Assert.Equal(response.RequestMessage.Method, httpRequestMessage.Method);
Assert.NotSame(response.RequestMessage, httpRequestMessage);
//TODO: response.RequestMessage.Content has been disposed so cant access it here
//Assert.Equal("Hello World", await response.RequestMessage.Content.ReadAsStringAsync());
}

// POST see other with content
Expand Down
82 changes: 45 additions & 37 deletions Octokit/Http/HttpClientAdapter.cs
Original file line number Diff line number Diff line change
Expand Up @@ -173,26 +173,17 @@ protected virtual void Dispose(bool disposing)

public async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
StreamContent savedContent = null;
bool unbuffered = false;
if (request.Content != null && request.Content.Headers.ContentLength != 0)
{
var stream = await request.Content.ReadAsStreamAsync().ConfigureAwait(false);
if (stream.CanSeek)
{
stream.Position = 0;
savedContent = new StreamContent(stream);
}
else
{
unbuffered = true;
}
}
// Clone the request/content incase we get a redirect
var clonedRequest = await CloneHttpRequestMessageAsync(request);

// Send initial response
var response = await _http.SendAsync(request, HttpCompletionOption.ResponseContentRead, cancellationToken).ConfigureAwait(false);

// Can't redirect without somewhere to redirect to. Throw?
if (response.Headers.Location == null) return response;
// Can't redirect without somewhere to redirect to.
if (response.Headers.Location == null)
{
return response;
}

// Don't redirect if we exceed max number of redirects
var redirectCount = 0;
Expand All @@ -204,7 +195,6 @@ public async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, Can
{
throw new InvalidOperationException("The redirect count for this request has been exceeded. Aborting.");
}
request.Properties[RedirectCountKey] = ++redirectCount;

if (response.StatusCode == HttpStatusCode.MovedPermanently
|| response.StatusCode == HttpStatusCode.Redirect
Expand All @@ -213,42 +203,60 @@ public async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, Can
|| response.StatusCode == HttpStatusCode.TemporaryRedirect
|| (int)response.StatusCode == 308)
{
var newRequest = CopyRequest(response.RequestMessage);

if (response.StatusCode == HttpStatusCode.SeeOther)
{
newRequest.Content = null;
newRequest.Method = HttpMethod.Get;
clonedRequest.Content = null;
clonedRequest.Method = HttpMethod.Get;
}
else
{
if (unbuffered)
{
throw new Exception("Cannot redirect a request with an unbuffered body");
}
newRequest.Content = savedContent;
}
newRequest.RequestUri = response.Headers.Location;
if (string.Compare(newRequest.RequestUri.Host, request.RequestUri.Host, StringComparison.OrdinalIgnoreCase) != 0)

// Increment the redirect count
clonedRequest.Properties[RedirectCountKey] = ++redirectCount;

// Set the new Uri based on location header
clonedRequest.RequestUri = response.Headers.Location;

// Clear authentication if redirected to a different host
if (string.Compare(clonedRequest.RequestUri.Host, request.RequestUri.Host, StringComparison.OrdinalIgnoreCase) != 0)
{
newRequest.Headers.Authorization = null;
clonedRequest.Headers.Authorization = null;
}

response = await SendAsync(newRequest, cancellationToken).ConfigureAwait(false);
// Send redirected request
response = await SendAsync(clonedRequest, cancellationToken).ConfigureAwait(false);
}

return response;
}

[SuppressMessage("Microsoft.Reliability", "CA2000:Dispose objects before losing scope")]
private static HttpRequestMessage CopyRequest(HttpRequestMessage oldRequest)
public static async Task<HttpRequestMessage> CloneHttpRequestMessageAsync(HttpRequestMessage oldRequest)
{
var newRequest = new HttpRequestMessage(oldRequest.Method, oldRequest.RequestUri);

// Copy the request's content (via a MemoryStream) into the cloned object
var ms = new MemoryStream();
if (oldRequest.Content != null)
{
await oldRequest.Content.CopyToAsync(ms).ConfigureAwait(false);
ms.Position = 0;
newRequest.Content = new StreamContent(ms);

// Copy the content headers
if (oldRequest.Content.Headers != null)
{
foreach (var h in oldRequest.Content.Headers)
{
newRequest.Content.Headers.Add(h.Key, h.Value);
}
}
}

newRequest.Version = oldRequest.Version;

foreach (var header in oldRequest.Headers)
{
newRequest.Headers.TryAddWithoutValidation(header.Key, header.Value);
}

foreach (var property in oldRequest.Properties)
{
newRequest.Properties.Add(property);
Expand All @@ -259,6 +267,6 @@ private static HttpRequestMessage CopyRequest(HttpRequestMessage oldRequest)
}

internal class RedirectHandler : DelegatingHandler
{
{
}
}

0 comments on commit 815f0d1

Please sign in to comment.