Skip to content

Commit

Permalink
Change DefaultMaximumErrorResponseLength to KB from Byte (#105396)
Browse files Browse the repository at this point in the history
* Change DefaultMaximumErrorResponseLength to KB from Byte

* Handle overflow

* Review feedback
  • Loading branch information
liveans authored Jul 24, 2024
1 parent 4a8aca8 commit e75fc27
Show file tree
Hide file tree
Showing 2 changed files with 57 additions and 19 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -346,7 +346,7 @@ public override Stream GetResponseStream()
return contentStream;
}

return new TruncatedReadStream(contentStream, maxErrorResponseLength);
return new TruncatedReadStream(contentStream, (long)maxErrorResponseLength * 1024);
}

return Stream.Null;
Expand Down Expand Up @@ -381,8 +381,9 @@ private void CheckDisposed()

private static string GetHeaderValueAsString(IEnumerable<string> values) => string.Join(", ", values);

internal sealed class TruncatedReadStream(Stream innerStream, int maxSize) : Stream
internal sealed class TruncatedReadStream(Stream innerStream, long maxSize) : Stream
{
private long _maxRemainingLength = maxSize;
public override bool CanRead => true;
public override bool CanSeek => false;
public override bool CanWrite => false;
Expand All @@ -399,8 +400,8 @@ public override int Read(byte[] buffer, int offset, int count)

public override int Read(Span<byte> buffer)
{
int readBytes = innerStream.Read(buffer.Slice(0, Math.Min(buffer.Length, maxSize)));
maxSize -= readBytes;
int readBytes = innerStream.Read(buffer.Slice(0, (int)Math.Min(buffer.Length, _maxRemainingLength)));
_maxRemainingLength -= readBytes;
return readBytes;
}

Expand All @@ -411,9 +412,9 @@ public override Task<int> ReadAsync(byte[] buffer, int offset, int count, Cancel

public override async ValueTask<int> ReadAsync(Memory<byte> buffer, CancellationToken cancellationToken = default)
{
int readBytes = await innerStream.ReadAsync(buffer.Slice(0, Math.Min(buffer.Length, maxSize)), cancellationToken)
int readBytes = await innerStream.ReadAsync(buffer.Slice(0, (int)Math.Min(buffer.Length, _maxRemainingLength)), cancellationToken)
.ConfigureAwait(false);
maxSize -= readBytes;
_maxRemainingLength -= readBytes;
return readBytes;
}

Expand Down
63 changes: 50 additions & 13 deletions src/libraries/System.Net.Requests/tests/HttpWebRequestTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2318,33 +2318,70 @@ await server.AcceptConnectionAsync(
[ConditionalFact(typeof(RemoteExecutor), nameof(RemoteExecutor.IsSupported))]
public async Task SendHttpRequest_WhenDefaultMaximumErrorResponseLengthSet_Success()
{
await RemoteExecutor.Invoke(async (async) =>
await RemoteExecutor.Invoke(async isAsync =>
{
TaskCompletionSource tcs = new TaskCompletionSource();
await LoopbackServer.CreateClientAndServerAsync(
async (uri) =>
async uri =>
{
HttpWebRequest request = WebRequest.CreateHttp(uri);
HttpWebRequest.DefaultMaximumErrorResponseLength = 5;
var exception =
await Assert.ThrowsAsync<WebException>(() => bool.Parse(async) ? request.GetResponseAsync() : Task.Run(() => request.GetResponse()));
HttpWebRequest.DefaultMaximumErrorResponseLength = 1; // 1 KB
WebException exception =
await Assert.ThrowsAsync<WebException>(() => bool.Parse(isAsync) ? request.GetResponseAsync() : Task.Run(() => request.GetResponse()));
tcs.SetResult();
Assert.NotNull(exception.Response);
using (var responseStream = exception.Response.GetResponseStream())
using (Stream responseStream = exception.Response.GetResponseStream())
{
var buffer = new byte[10];
int readLen = responseStream.Read(buffer, 0, buffer.Length);
Assert.Equal(5, readLen);
Assert.Equal(new string('a', 5), Encoding.UTF8.GetString(buffer[0..readLen]));
Assert.Equal(0, responseStream.Read(buffer));
byte[] buffer = new byte[10 * 1024];
int totalReadLen = 0;
int readLen = 0;
while ((readLen = responseStream.Read(buffer, readLen, buffer.Length - readLen)) > 0)
{
totalReadLen += readLen;
}

Assert.Equal(1024, totalReadLen);
Assert.Equal(new string('a', 1024), Encoding.UTF8.GetString(buffer[0..totalReadLen]));
}
},
async (server) =>
async server =>
{
await server.AcceptConnectionAsync(
async connection =>
{
await connection.SendResponseAsync(statusCode: HttpStatusCode.BadRequest, content: new string('a', 10 * 1024));
await tcs.Task;
});
});
}, IsAsync.ToString()).DisposeAsync();
}

[ConditionalFact(typeof(RemoteExecutor), nameof(RemoteExecutor.IsSupported))]
public async Task SendHttpRequest_WhenDefaultMaximumErrorResponseLengthSetToIntMax_DoesNotThrow()
{
await RemoteExecutor.Invoke(async isAsync =>
{
TaskCompletionSource tcs = new TaskCompletionSource();
await LoopbackServer.CreateClientAndServerAsync(
async uri =>
{
HttpWebRequest request = WebRequest.CreateHttp(uri);
HttpWebRequest.DefaultMaximumErrorResponseLength = int.MaxValue; // KB
WebException exception =
await Assert.ThrowsAsync<WebException>(() => bool.Parse(isAsync) ? request.GetResponseAsync() : Task.Run(() => request.GetResponse()));
tcs.SetResult();
Assert.NotNull(exception.Response);
using (Stream responseStream = exception.Response.GetResponseStream())
{
Assert.Equal(1, await responseStream.ReadAsync(new byte[1]));
}
},
async server =>
{
await server.AcceptConnectionAsync(
async connection =>
{
await connection.SendResponseAsync(statusCode: HttpStatusCode.BadRequest, content: new string('a', 10));
await connection.SendResponseAsync(statusCode: HttpStatusCode.BadRequest, content: new string('a', 10 * 1024));
await tcs.Task;
});
});
Expand Down

0 comments on commit e75fc27

Please sign in to comment.