2018-04-24 09:31:23 +00:00
|
|
|
//
|
|
|
|
// WebResponseStream.cs
|
|
|
|
//
|
|
|
|
// Author:
|
|
|
|
// Martin Baulig <mabaul@microsoft.com>
|
|
|
|
//
|
|
|
|
// Copyright (c) 2017 Xamarin Inc. (http://www.xamarin.com)
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
|
|
|
using System.IO;
|
|
|
|
using System.Text;
|
|
|
|
using System.Collections;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Threading;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using System.Runtime.ExceptionServices;
|
|
|
|
using System.Net.Sockets;
|
|
|
|
|
|
|
|
namespace System.Net
|
|
|
|
{
|
|
|
|
class WebResponseStream : WebConnectionStream
|
|
|
|
{
|
2018-08-07 15:19:03 +00:00
|
|
|
WebReadStream innerStream;
|
2018-04-24 09:31:23 +00:00
|
|
|
bool nextReadCalled;
|
2018-08-07 15:19:03 +00:00
|
|
|
bool bufferedEntireContent;
|
2018-04-24 09:31:23 +00:00
|
|
|
WebCompletionSource pendingRead;
|
|
|
|
object locker = new object ();
|
|
|
|
int nestedRead;
|
|
|
|
bool read_eof;
|
|
|
|
|
|
|
|
public WebRequestStream RequestStream {
|
|
|
|
get;
|
|
|
|
}
|
|
|
|
|
|
|
|
public WebHeaderCollection Headers {
|
|
|
|
get;
|
|
|
|
private set;
|
|
|
|
}
|
|
|
|
|
|
|
|
public HttpStatusCode StatusCode {
|
|
|
|
get;
|
|
|
|
private set;
|
|
|
|
}
|
|
|
|
|
|
|
|
public string StatusDescription {
|
|
|
|
get;
|
|
|
|
private set;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Version Version {
|
|
|
|
get;
|
|
|
|
private set;
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool KeepAlive {
|
|
|
|
get;
|
|
|
|
private set;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal readonly string ME;
|
|
|
|
|
|
|
|
public WebResponseStream (WebRequestStream request)
|
2018-08-07 15:19:03 +00:00
|
|
|
: base (request.Connection, request.Operation)
|
2018-04-24 09:31:23 +00:00
|
|
|
{
|
|
|
|
RequestStream = request;
|
|
|
|
|
|
|
|
#if MONO_WEB_DEBUG
|
|
|
|
ME = $"WRP(Cnc={Connection.ID}, Op={Operation.ID})";
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
public override bool CanRead => true;
|
|
|
|
|
|
|
|
public override bool CanWrite => false;
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
bool ChunkedRead {
|
|
|
|
get; set;
|
2018-04-24 09:31:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public override async Task<int> ReadAsync (byte[] buffer, int offset, int count, CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
WebConnection.Debug ($"{ME} READ ASYNC");
|
|
|
|
|
|
|
|
cancellationToken.ThrowIfCancellationRequested ();
|
|
|
|
|
|
|
|
if (buffer == null)
|
|
|
|
throw new ArgumentNullException (nameof (buffer));
|
|
|
|
|
|
|
|
int length = buffer.Length;
|
|
|
|
if (offset < 0 || length < offset)
|
|
|
|
throw new ArgumentOutOfRangeException (nameof (offset));
|
|
|
|
if (count < 0 || (length - offset) < count)
|
|
|
|
throw new ArgumentOutOfRangeException (nameof (count));
|
|
|
|
|
|
|
|
if (Interlocked.CompareExchange (ref nestedRead, 1, 0) != 0)
|
|
|
|
throw new InvalidOperationException ("Invalid nested call.");
|
|
|
|
|
|
|
|
var completion = new WebCompletionSource ();
|
|
|
|
while (!cancellationToken.IsCancellationRequested) {
|
|
|
|
/*
|
|
|
|
* 'currentRead' is set by ReadAllAsync().
|
|
|
|
*/
|
|
|
|
var oldCompletion = Interlocked.CompareExchange (ref pendingRead, completion, null);
|
|
|
|
WebConnection.Debug ($"{ME} READ ASYNC #1: {oldCompletion != null}");
|
|
|
|
if (oldCompletion == null)
|
|
|
|
break;
|
|
|
|
await oldCompletion.WaitForCompletion ().ConfigureAwait (false);
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
WebConnection.Debug ($"{ME} READ ASYNC #2");
|
2018-04-24 09:31:23 +00:00
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
int nbytes = 0;
|
2018-04-24 09:31:23 +00:00
|
|
|
Exception throwMe = null;
|
|
|
|
|
|
|
|
try {
|
2018-08-07 15:19:03 +00:00
|
|
|
nbytes = await ProcessRead (buffer, offset, count, cancellationToken).ConfigureAwait (false);
|
2018-04-24 09:31:23 +00:00
|
|
|
} catch (Exception e) {
|
|
|
|
throwMe = GetReadException (WebExceptionStatus.ReceiveFailure, e, "ReadAsync");
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
WebConnection.Debug ($"{ME} READ ASYNC #3: {nbytes} {throwMe?.Message}");
|
2018-04-24 09:31:23 +00:00
|
|
|
|
|
|
|
if (throwMe != null) {
|
|
|
|
lock (locker) {
|
|
|
|
completion.TrySetException (throwMe);
|
|
|
|
pendingRead = null;
|
|
|
|
nestedRead = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
closed = true;
|
|
|
|
Operation.Finish (false, throwMe);
|
|
|
|
throw throwMe;
|
|
|
|
}
|
|
|
|
|
|
|
|
lock (locker) {
|
2018-08-07 15:19:03 +00:00
|
|
|
completion.TrySetCompleted ();
|
2018-04-24 09:31:23 +00:00
|
|
|
pendingRead = null;
|
|
|
|
nestedRead = 0;
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
if (nbytes <= 0 && !read_eof) {
|
2018-04-24 09:31:23 +00:00
|
|
|
read_eof = true;
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
if (!nextReadCalled) {
|
|
|
|
WebConnection.Debug ($"{ME} READ ASYNC - READ COMPLETE: {nbytes} - {nextReadCalled}");
|
|
|
|
if (!nextReadCalled) {
|
|
|
|
nextReadCalled = true;
|
|
|
|
Operation.Finish (true);
|
|
|
|
}
|
2018-04-24 09:31:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nbytes;
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
Task<int> ProcessRead (byte[] buffer, int offset, int size, CancellationToken cancellationToken)
|
2018-04-24 09:31:23 +00:00
|
|
|
{
|
2018-08-07 15:19:03 +00:00
|
|
|
if (read_eof)
|
|
|
|
return Task.FromResult (0);
|
|
|
|
if (cancellationToken.IsCancellationRequested)
|
|
|
|
return Task.FromCanceled<int> (cancellationToken);
|
|
|
|
return HttpWebRequest.RunWithTimeout (
|
|
|
|
ct => innerStream.ReadAsync (buffer, offset, size, ct),
|
|
|
|
ReadTimeout,
|
|
|
|
() => {
|
|
|
|
Operation.Abort ();
|
|
|
|
innerStream.Dispose ();
|
|
|
|
}, () => Operation.Aborted, cancellationToken);
|
2018-04-24 09:31:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CheckAuthHeader (string headerName)
|
|
|
|
{
|
|
|
|
var authHeader = Headers[headerName];
|
|
|
|
return (authHeader != null && authHeader.IndexOf ("NTLM", StringComparison.Ordinal) != -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ExpectContent {
|
|
|
|
get {
|
|
|
|
if (Request.Method == "HEAD")
|
|
|
|
return false;
|
|
|
|
return ((int)StatusCode >= 200 && (int)StatusCode != 204 && (int)StatusCode != 304);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
void Initialize (BufferOffsetSize buffer)
|
2018-04-24 09:31:23 +00:00
|
|
|
{
|
|
|
|
WebConnection.Debug ($"{ME} INIT: status={(int)StatusCode} bos={buffer.Offset}/{buffer.Size}");
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
long contentLength;
|
2018-04-24 09:31:23 +00:00
|
|
|
string contentType = Headers["Transfer-Encoding"];
|
|
|
|
bool chunkedRead = (contentType != null && contentType.IndexOf ("chunked", StringComparison.OrdinalIgnoreCase) != -1);
|
|
|
|
string clength = Headers["Content-Length"];
|
|
|
|
if (!chunkedRead && !string.IsNullOrEmpty (clength)) {
|
|
|
|
if (!long.TryParse (clength, out contentLength))
|
|
|
|
contentLength = Int64.MaxValue;
|
|
|
|
} else {
|
|
|
|
contentLength = Int64.MaxValue;
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
string tencoding = null;
|
|
|
|
if (ExpectContent)
|
|
|
|
tencoding = Headers["Transfer-Encoding"];
|
|
|
|
|
|
|
|
ChunkedRead = (tencoding != null && tencoding.IndexOf ("chunked", StringComparison.OrdinalIgnoreCase) != -1);
|
|
|
|
|
2018-04-24 09:31:23 +00:00
|
|
|
if (Version == HttpVersion.Version11 && RequestStream.KeepAlive) {
|
|
|
|
KeepAlive = true;
|
|
|
|
var cncHeader = Headers[ServicePoint.UsesProxy ? "Proxy-Connection" : "Connection"];
|
|
|
|
if (cncHeader != null) {
|
|
|
|
cncHeader = cncHeader.ToLower ();
|
|
|
|
KeepAlive = cncHeader.IndexOf ("keep-alive", StringComparison.Ordinal) != -1;
|
|
|
|
if (cncHeader.IndexOf ("close", StringComparison.Ordinal) != -1)
|
|
|
|
KeepAlive = false;
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
if (!ChunkedRead && contentLength == Int64.MaxValue) {
|
|
|
|
/*
|
|
|
|
* This is a violation of the HTTP Spec - the server neither send a
|
|
|
|
* "Content-Length:" nor a "Transfer-Encoding: chunked" header.
|
|
|
|
* The only way to recover from this is to keep reading until the
|
|
|
|
* remote closes the connection, so we can't reuse it.
|
|
|
|
*/
|
|
|
|
KeepAlive = false;
|
|
|
|
}
|
|
|
|
}
|
2018-04-24 09:31:23 +00:00
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
/*
|
|
|
|
* Inner layer:
|
|
|
|
* We may have read a few extra bytes while parsing the headers, these will be
|
|
|
|
* passed to us in the @buffer parameter and we need to read these before
|
|
|
|
* reading from the `InnerStream`.
|
|
|
|
*/
|
|
|
|
Stream networkStream;
|
|
|
|
if (!ExpectContent || (!ChunkedRead && buffer.Size >= contentLength)) {
|
|
|
|
bufferedEntireContent = true;
|
|
|
|
innerStream = new BufferedReadStream (Operation, null, buffer);
|
|
|
|
networkStream = innerStream;
|
|
|
|
} else if (buffer.Size > 0) {
|
|
|
|
networkStream = new BufferedReadStream (Operation, RequestStream.InnerStream, buffer);
|
|
|
|
} else {
|
|
|
|
networkStream = RequestStream.InnerStream;
|
|
|
|
}
|
2018-04-24 09:31:23 +00:00
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
/*
|
|
|
|
* Intermediate layer:
|
|
|
|
* - Wrap with MonoChunkStream when using chunked encoding.
|
|
|
|
* - Otherwise, we should have a Content-Length, wrap with
|
|
|
|
* FixedSizeReadStream to read exactly that many bytes.
|
|
|
|
*/
|
|
|
|
if (ChunkedRead) {
|
|
|
|
innerStream = new MonoChunkStream (Operation, networkStream, Headers);
|
|
|
|
} else if (bufferedEntireContent) {
|
|
|
|
// 'innerStream' has already been set above.
|
|
|
|
} else if (contentLength != Int64.MaxValue) {
|
|
|
|
innerStream = new FixedSizeReadStream (Operation, networkStream, contentLength);
|
2018-04-24 09:31:23 +00:00
|
|
|
} else {
|
2018-08-07 15:19:03 +00:00
|
|
|
// Violation of the HTTP Spec - neither chunked nor length.
|
|
|
|
innerStream = new BufferedReadStream (Operation, networkStream, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Outer layer:
|
|
|
|
* - Decode gzip/deflate if requested.
|
|
|
|
*/
|
|
|
|
string content_encoding = Headers["Content-Encoding"];
|
|
|
|
if (content_encoding == "gzip" && (Request.AutomaticDecompression & DecompressionMethods.GZip) != 0) {
|
|
|
|
innerStream = ContentDecodeStream.Create (Operation, innerStream, ContentDecodeStream.Mode.GZip);
|
|
|
|
Headers.Remove (HttpRequestHeader.ContentEncoding);
|
|
|
|
} else if (content_encoding == "deflate" && (Request.AutomaticDecompression & DecompressionMethods.Deflate) != 0) {
|
|
|
|
innerStream = ContentDecodeStream.Create (Operation, innerStream, ContentDecodeStream.Mode.Deflate);
|
|
|
|
Headers.Remove (HttpRequestHeader.ContentEncoding);
|
2018-04-24 09:31:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
WebConnection.Debug ($"{ME} INIT #1: - {ExpectContent} {closed} {nextReadCalled}");
|
|
|
|
|
|
|
|
if (!ExpectContent) {
|
2018-08-07 15:19:03 +00:00
|
|
|
nextReadCalled = true;
|
2018-04-24 09:31:23 +00:00
|
|
|
Operation.Finish (true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
async Task<byte[]> ReadAllAsyncInner (CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
var maximumSize = (long)HttpWebRequest.DefaultMaximumErrorResponseLength << 16;
|
|
|
|
using (var ms = new MemoryStream ()) {
|
|
|
|
while (ms.Position < maximumSize) {
|
|
|
|
cancellationToken.ThrowIfCancellationRequested ();
|
|
|
|
var buffer = new byte[16384];
|
|
|
|
var ret = await ProcessRead (buffer, 0, buffer.Length, cancellationToken).ConfigureAwait (false);
|
|
|
|
if (ret < 0)
|
|
|
|
throw new IOException ();
|
|
|
|
if (ret == 0)
|
|
|
|
break;
|
|
|
|
ms.Write (buffer, 0, ret);
|
|
|
|
}
|
|
|
|
return ms.GetBuffer ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 09:31:23 +00:00
|
|
|
internal async Task ReadAllAsync (bool resending, CancellationToken cancellationToken)
|
|
|
|
{
|
2018-08-07 15:19:03 +00:00
|
|
|
WebConnection.Debug ($"{ME} READ ALL ASYNC: resending={resending} eof={read_eof} " +
|
|
|
|
"nextReadCalled={nextReadCalled}");
|
|
|
|
if (read_eof || bufferedEntireContent || nextReadCalled) {
|
2018-04-24 09:31:23 +00:00
|
|
|
if (!nextReadCalled) {
|
|
|
|
nextReadCalled = true;
|
|
|
|
Operation.Finish (true);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var completion = new WebCompletionSource ();
|
|
|
|
var timeoutCts = new CancellationTokenSource ();
|
|
|
|
try {
|
|
|
|
var timeoutTask = Task.Delay (ReadTimeout, timeoutCts.Token);
|
|
|
|
while (true) {
|
|
|
|
/*
|
|
|
|
* 'currentRead' is set by ReadAsync().
|
|
|
|
*/
|
|
|
|
cancellationToken.ThrowIfCancellationRequested ();
|
|
|
|
var oldCompletion = Interlocked.CompareExchange (ref pendingRead, completion, null);
|
|
|
|
if (oldCompletion == null)
|
|
|
|
break;
|
|
|
|
|
|
|
|
// ReadAsync() is in progress.
|
|
|
|
var oldReadTask = oldCompletion.WaitForCompletion ();
|
|
|
|
var anyTask = await Task.WhenAny (oldReadTask, timeoutTask).ConfigureAwait (false);
|
|
|
|
if (anyTask == timeoutTask)
|
|
|
|
throw new WebException ("The operation has timed out.", WebExceptionStatus.Timeout);
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
timeoutCts.Cancel ();
|
|
|
|
timeoutCts.Dispose ();
|
|
|
|
}
|
|
|
|
|
|
|
|
WebConnection.Debug ($"{ME} READ ALL ASYNC #1");
|
|
|
|
|
|
|
|
try {
|
2018-08-07 15:19:03 +00:00
|
|
|
cancellationToken.ThrowIfCancellationRequested ();
|
2018-04-24 09:31:23 +00:00
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
/*
|
|
|
|
* We may have awaited on the 'readTcs', so check
|
|
|
|
* for eof again as ReadAsync() may have set it.
|
|
|
|
*/
|
|
|
|
if (read_eof || bufferedEntireContent)
|
|
|
|
return;
|
|
|
|
/*
|
|
|
|
* Simplify: if we're resending on a new connection,
|
|
|
|
* then we can simply close the connection here.
|
|
|
|
*/
|
|
|
|
if (resending && !KeepAlive) {
|
|
|
|
Close ();
|
|
|
|
return;
|
2018-04-24 09:31:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
var buffer = await ReadAllAsyncInner (cancellationToken).ConfigureAwait (false);
|
|
|
|
var bos = new BufferOffsetSize (buffer, 0, buffer.Length, false);
|
|
|
|
innerStream = new BufferedReadStream (Operation, null, bos);
|
2018-04-24 09:31:23 +00:00
|
|
|
|
|
|
|
nextReadCalled = true;
|
|
|
|
completion.TrySetCompleted ();
|
|
|
|
} catch (Exception ex) {
|
|
|
|
WebConnection.Debug ($"{ME} READ ALL ASYNC EX: {ex.Message}");
|
|
|
|
completion.TrySetException (ex);
|
|
|
|
throw;
|
|
|
|
} finally {
|
|
|
|
WebConnection.Debug ($"{ME} READ ALL ASYNC #2");
|
|
|
|
pendingRead = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
Operation.Finish (true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override Task WriteAsync (byte[] buffer, int offset, int count, CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
return Task.FromException (new NotSupportedException (SR.net_readonlystream));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void Close_internal (ref bool disposed)
|
|
|
|
{
|
2018-08-07 15:19:03 +00:00
|
|
|
WebConnection.Debug ($"{ME} CLOSE: disposed={disposed} closed={closed} nextReadCalled={nextReadCalled}");
|
2018-04-24 09:31:23 +00:00
|
|
|
if (!closed && !nextReadCalled) {
|
|
|
|
nextReadCalled = true;
|
2018-08-07 15:19:03 +00:00
|
|
|
WebConnection.Debug ($"{ME} CLOSE #1: read_eof={read_eof} bufferedEntireContent={bufferedEntireContent}");
|
|
|
|
if (read_eof || bufferedEntireContent) {
|
2018-04-24 09:31:23 +00:00
|
|
|
disposed = true;
|
|
|
|
Operation.Finish (true);
|
|
|
|
} else {
|
|
|
|
// If we have not read all the contents
|
|
|
|
closed = true;
|
|
|
|
disposed = true;
|
|
|
|
Operation.Finish (false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WebException GetReadException (WebExceptionStatus status, Exception error, string where)
|
|
|
|
{
|
|
|
|
error = GetException (error);
|
|
|
|
string msg = $"Error getting response stream ({where}): {status}";
|
|
|
|
if (error == null)
|
|
|
|
return new WebException ($"Error getting response stream ({where}): {status}", status);
|
|
|
|
if (error is WebException wexc)
|
|
|
|
return wexc;
|
|
|
|
if (Operation.Aborted || error is OperationCanceledException || error is ObjectDisposedException)
|
|
|
|
return HttpWebRequest.CreateRequestAbortedException ();
|
|
|
|
return new WebException ($"Error getting response stream ({where}): {status} {error.Message}", status,
|
|
|
|
WebExceptionInternalStatus.RequestFatal, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal async Task InitReadAsync (CancellationToken cancellationToken)
|
|
|
|
{
|
|
|
|
WebConnection.Debug ($"{ME} INIT READ ASYNC");
|
|
|
|
|
|
|
|
var buffer = new BufferOffsetSize (new byte[4096], false);
|
|
|
|
var state = ReadState.None;
|
|
|
|
int position = 0;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
Operation.ThrowIfClosedOrDisposed (cancellationToken);
|
|
|
|
|
|
|
|
WebConnection.Debug ($"{ME} INIT READ ASYNC LOOP: {state} {position} - {buffer.Offset}/{buffer.Size}");
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
var nread = await RequestStream.InnerStream.ReadAsync (
|
2018-04-24 09:31:23 +00:00
|
|
|
buffer.Buffer, buffer.Offset, buffer.Size, cancellationToken).ConfigureAwait (false);
|
|
|
|
|
|
|
|
WebConnection.Debug ($"{ME} INIT READ ASYNC LOOP #1: {state} {position} - {buffer.Offset}/{buffer.Size} - {nread}");
|
|
|
|
|
|
|
|
if (nread == 0)
|
|
|
|
throw GetReadException (WebExceptionStatus.ReceiveFailure, null, "ReadDoneAsync2");
|
|
|
|
|
|
|
|
if (nread < 0)
|
|
|
|
throw GetReadException (WebExceptionStatus.ServerProtocolViolation, null, "ReadDoneAsync3");
|
|
|
|
|
|
|
|
buffer.Offset += nread;
|
|
|
|
buffer.Size -= nread;
|
|
|
|
|
|
|
|
if (state == ReadState.None) {
|
|
|
|
try {
|
|
|
|
var oldPos = position;
|
|
|
|
if (!GetResponse (buffer, ref position, ref state))
|
|
|
|
position = oldPos;
|
|
|
|
} catch (Exception e) {
|
|
|
|
WebConnection.Debug ($"{ME} INIT READ ASYNC FAILED: {e.Message}\n{e}");
|
|
|
|
throw GetReadException (WebExceptionStatus.ServerProtocolViolation, e, "ReadDoneAsync4");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state == ReadState.Aborted)
|
|
|
|
throw GetReadException (WebExceptionStatus.RequestCanceled, null, "ReadDoneAsync5");
|
|
|
|
|
|
|
|
if (state == ReadState.Content) {
|
|
|
|
buffer.Size = buffer.Offset - position;
|
|
|
|
buffer.Offset = position;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
int est = nread * 2;
|
|
|
|
if (est > buffer.Size) {
|
|
|
|
var newBuffer = new byte [buffer.Buffer.Length + est];
|
|
|
|
Buffer.BlockCopy (buffer.Buffer, 0, newBuffer, 0, buffer.Offset);
|
|
|
|
buffer = new BufferOffsetSize (newBuffer, buffer.Offset, newBuffer.Length - buffer.Offset, false);
|
|
|
|
}
|
|
|
|
state = ReadState.None;
|
|
|
|
position = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
WebConnection.Debug ($"{ME} INIT READ ASYNC LOOP DONE: {buffer.Offset} {buffer.Size}");
|
|
|
|
|
|
|
|
try {
|
2018-08-07 15:19:03 +00:00
|
|
|
Initialize (buffer);
|
2018-04-24 09:31:23 +00:00
|
|
|
} catch (Exception e) {
|
|
|
|
throw GetReadException (WebExceptionStatus.ReceiveFailure, e, "ReadDoneAsync6");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetResponse (BufferOffsetSize buffer, ref int pos, ref ReadState state)
|
|
|
|
{
|
|
|
|
string line = null;
|
|
|
|
bool lineok = false;
|
|
|
|
bool isContinue = false;
|
|
|
|
bool emptyFirstLine = false;
|
|
|
|
do {
|
|
|
|
if (state == ReadState.Aborted)
|
|
|
|
throw GetReadException (WebExceptionStatus.RequestCanceled, null, "GetResponse");
|
|
|
|
|
|
|
|
if (state == ReadState.None) {
|
|
|
|
lineok = WebConnection.ReadLine (buffer.Buffer, ref pos, buffer.Offset, ref line);
|
|
|
|
if (!lineok)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (line == null) {
|
|
|
|
emptyFirstLine = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
emptyFirstLine = false;
|
|
|
|
state = ReadState.Status;
|
|
|
|
|
|
|
|
string[] parts = line.Split (' ');
|
|
|
|
if (parts.Length < 2)
|
|
|
|
throw GetReadException (WebExceptionStatus.ServerProtocolViolation, null, "GetResponse");
|
|
|
|
|
|
|
|
if (String.Compare (parts[0], "HTTP/1.1", true) == 0) {
|
|
|
|
Version = HttpVersion.Version11;
|
|
|
|
ServicePoint.SetVersion (HttpVersion.Version11);
|
|
|
|
} else {
|
|
|
|
Version = HttpVersion.Version10;
|
|
|
|
ServicePoint.SetVersion (HttpVersion.Version10);
|
|
|
|
}
|
|
|
|
|
|
|
|
StatusCode = (HttpStatusCode)UInt32.Parse (parts[1]);
|
|
|
|
if (parts.Length >= 3)
|
|
|
|
StatusDescription = String.Join (" ", parts, 2, parts.Length - 2);
|
|
|
|
else
|
|
|
|
StatusDescription = string.Empty;
|
|
|
|
|
2018-06-23 17:06:07 +00:00
|
|
|
if (pos >= buffer.Offset)
|
2018-04-24 09:31:23 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
emptyFirstLine = false;
|
|
|
|
if (state == ReadState.Status) {
|
|
|
|
state = ReadState.Headers;
|
|
|
|
Headers = new WebHeaderCollection ();
|
|
|
|
var headerList = new List<string> ();
|
|
|
|
bool finished = false;
|
|
|
|
while (!finished) {
|
|
|
|
if (WebConnection.ReadLine (buffer.Buffer, ref pos, buffer.Offset, ref line) == false)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (line == null) {
|
|
|
|
// Empty line: end of headers
|
|
|
|
finished = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (line.Length > 0 && (line[0] == ' ' || line[0] == '\t')) {
|
|
|
|
int count = headerList.Count - 1;
|
|
|
|
if (count < 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
string prev = headerList[count] + line;
|
|
|
|
headerList[count] = prev;
|
|
|
|
} else {
|
|
|
|
headerList.Add (line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!finished)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// .NET uses ParseHeaders or ParseHeadersStrict which is much better
|
|
|
|
foreach (string s in headerList) {
|
|
|
|
|
|
|
|
int pos_s = s.IndexOf (':');
|
|
|
|
if (pos_s == -1)
|
|
|
|
throw new ArgumentException ("no colon found", "header");
|
|
|
|
|
|
|
|
var header = s.Substring (0, pos_s);
|
|
|
|
var value = s.Substring (pos_s + 1).Trim ();
|
|
|
|
|
|
|
|
if (WebHeaderCollection.AllowMultiValues (header)) {
|
|
|
|
Headers.AddInternal (header, value);
|
|
|
|
} else {
|
|
|
|
Headers.SetInternal (header, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (StatusCode == HttpStatusCode.Continue) {
|
|
|
|
ServicePoint.SendContinue = true;
|
|
|
|
if (pos >= buffer.Offset)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (Request.ExpectContinue) {
|
|
|
|
Request.DoContinueDelegate ((int)StatusCode, Headers);
|
|
|
|
// Prevent double calls when getting the
|
|
|
|
// headers in several packets.
|
|
|
|
Request.ExpectContinue = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
state = ReadState.None;
|
|
|
|
isContinue = true;
|
|
|
|
} else {
|
|
|
|
state = ReadState.Content;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (emptyFirstLine || isContinue);
|
|
|
|
|
|
|
|
throw GetReadException (WebExceptionStatus.ServerProtocolViolation, null, "GetResponse");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|