2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// <copyright file="XmlDownloadManager.cs" company="Microsoft">
|
|
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
// </copyright>
|
2017-08-21 15:34:15 +00:00
|
|
|
// <owner current="true" primary="true">Microsoft</owner>
|
2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
namespace System.Xml {
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Security;
|
|
|
|
using System.Collections;
|
|
|
|
using System.Net;
|
|
|
|
using System.Net.Cache;
|
|
|
|
using System.Runtime.Versioning;
|
|
|
|
|
|
|
|
//
|
|
|
|
// XmlDownloadManager
|
|
|
|
//
|
|
|
|
internal partial class XmlDownloadManager {
|
|
|
|
|
|
|
|
Hashtable connections;
|
|
|
|
|
|
|
|
[ResourceConsumption(ResourceScope.Machine)]
|
|
|
|
[ResourceExposure(ResourceScope.Machine)]
|
|
|
|
internal Stream GetStream(Uri uri, ICredentials credentials, IWebProxy proxy,
|
|
|
|
RequestCachePolicy cachePolicy) {
|
|
|
|
if ( uri.Scheme == "file" ) {
|
|
|
|
return new FileStream( uri.LocalPath, FileMode.Open, FileAccess.Read, FileShare.Read, 1 );
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return GetNonFileStream( uri, credentials, proxy, cachePolicy );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private Stream GetNonFileStream( Uri uri, ICredentials credentials, IWebProxy proxy,
|
|
|
|
RequestCachePolicy cachePolicy ) {
|
|
|
|
WebRequest req = WebRequest.Create( uri );
|
|
|
|
if ( credentials != null ) {
|
|
|
|
req.Credentials = credentials;
|
|
|
|
}
|
|
|
|
if ( proxy != null ) {
|
|
|
|
req.Proxy = proxy;
|
|
|
|
}
|
|
|
|
if ( cachePolicy != null ) {
|
|
|
|
req.CachePolicy = cachePolicy;
|
|
|
|
}
|
|
|
|
WebResponse resp = req.GetResponse();
|
|
|
|
HttpWebRequest webReq = req as HttpWebRequest;
|
|
|
|
if ( webReq != null ) {
|
|
|
|
lock ( this ) {
|
|
|
|
if ( connections == null ) {
|
|
|
|
connections = new Hashtable();
|
|
|
|
}
|
|
|
|
OpenedHost openedHost = (OpenedHost)connections[webReq.Address.Host];
|
|
|
|
if ( openedHost == null ) {
|
|
|
|
openedHost = new OpenedHost();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( openedHost.nonCachedConnectionsCount < webReq.ServicePoint.ConnectionLimit - 1 ) {
|
|
|
|
// we are not close to connection limit -> don't cache the stream
|
|
|
|
if ( openedHost.nonCachedConnectionsCount == 0 ) {
|
|
|
|
connections.Add( webReq.Address.Host, openedHost );
|
|
|
|
}
|
|
|
|
openedHost.nonCachedConnectionsCount++;
|
|
|
|
return new XmlRegisteredNonCachedStream( resp.GetResponseStream(), this, webReq.Address.Host );
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// cache the stream and save the connection for the next request
|
|
|
|
return new XmlCachedStream( resp.ResponseUri, resp.GetResponseStream() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return resp.GetResponseStream();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void Remove( string host ) {
|
|
|
|
lock ( this ) {
|
|
|
|
OpenedHost openedHost = (OpenedHost)connections[host];
|
|
|
|
if ( openedHost != null ) {
|
|
|
|
if ( --openedHost.nonCachedConnectionsCount == 0 ) {
|
|
|
|
connections.Remove( host );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// OpenedHost
|
|
|
|
//
|
|
|
|
internal class OpenedHost {
|
|
|
|
internal int nonCachedConnectionsCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// XmlRegisteredNonCachedStream
|
|
|
|
//
|
|
|
|
internal class XmlRegisteredNonCachedStream : Stream {
|
|
|
|
protected Stream stream;
|
|
|
|
XmlDownloadManager downloadManager;
|
|
|
|
string host;
|
|
|
|
|
|
|
|
internal XmlRegisteredNonCachedStream( Stream stream, XmlDownloadManager downloadManager, string host ) {
|
|
|
|
this.stream = stream;
|
|
|
|
this.downloadManager = downloadManager;
|
|
|
|
this.host = host;
|
|
|
|
}
|
|
|
|
|
|
|
|
~XmlRegisteredNonCachedStream() {
|
|
|
|
if ( downloadManager != null ) {
|
|
|
|
downloadManager.Remove( host );
|
|
|
|
}
|
|
|
|
stream = null;
|
|
|
|
// The base class, Stream, provides its own finalizer
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void Dispose( bool disposing ) {
|
|
|
|
try {
|
|
|
|
if ( disposing && stream != null ) {
|
|
|
|
if ( downloadManager != null ) {
|
|
|
|
downloadManager.Remove( host );
|
|
|
|
}
|
|
|
|
stream.Close();
|
|
|
|
}
|
|
|
|
stream = null;
|
|
|
|
GC.SuppressFinalize( this ); // do not call finalizer
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
base.Dispose( disposing );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Stream
|
|
|
|
//
|
|
|
|
public override IAsyncResult BeginRead( byte[] buffer, int offset, int count, AsyncCallback callback, object state ) {
|
|
|
|
return stream.BeginRead( buffer, offset, count, callback, state );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override IAsyncResult BeginWrite( byte[] buffer, int offset, int count, AsyncCallback callback, object state ) {
|
|
|
|
return stream.BeginWrite( buffer, offset, count, callback, state );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override int EndRead( IAsyncResult asyncResult ) {
|
|
|
|
return stream.EndRead( asyncResult );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void EndWrite( IAsyncResult asyncResult ) {
|
|
|
|
stream.EndWrite( asyncResult );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Flush() {
|
|
|
|
stream.Flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
public override int Read( byte[] buffer, int offset, int count ) {
|
|
|
|
return stream.Read( buffer, offset, count );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override int ReadByte() {
|
|
|
|
return stream.ReadByte();
|
|
|
|
}
|
|
|
|
|
|
|
|
public override long Seek( long offset, SeekOrigin origin ) {
|
|
|
|
return stream.Seek( offset, origin );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void SetLength( long value ) {
|
|
|
|
stream.SetLength( value );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Write( byte[] buffer, int offset, int count ) {
|
|
|
|
stream.Write( buffer, offset, count );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void WriteByte( byte value ) {
|
|
|
|
stream.WriteByte( value );
|
|
|
|
}
|
|
|
|
|
|
|
|
public override Boolean CanRead {
|
|
|
|
get { return stream.CanRead; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override Boolean CanSeek {
|
|
|
|
get { return stream.CanSeek; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override Boolean CanWrite {
|
|
|
|
get { return stream.CanWrite; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override long Length {
|
|
|
|
get { return stream.Length; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override long Position {
|
|
|
|
get { return stream.Position; }
|
|
|
|
set { stream.Position = value; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// XmlCachedStream
|
|
|
|
//
|
|
|
|
internal class XmlCachedStream : MemoryStream {
|
|
|
|
private const int MoveBufferSize = 4096;
|
|
|
|
|
|
|
|
private Uri uri;
|
|
|
|
|
|
|
|
internal XmlCachedStream( Uri uri, Stream stream )
|
|
|
|
: base() {
|
|
|
|
|
|
|
|
this.uri = uri;
|
|
|
|
|
|
|
|
try {
|
|
|
|
byte[] bytes = new byte[MoveBufferSize];
|
|
|
|
int read = 0;
|
|
|
|
while ( ( read = stream.Read( bytes, 0, MoveBufferSize ) ) > 0 ) {
|
|
|
|
this.Write( bytes, 0, read );
|
|
|
|
}
|
|
|
|
base.Position = 0;
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
stream.Close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|