You've already forked linux-packaging-mono
Imported Upstream version 4.6.0.125
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
This commit is contained in:
parent
a569aebcfd
commit
e79aa3c0ed
@@ -0,0 +1,232 @@
|
||||
//------------------------------------------------------------------------------
|
||||
// <copyright file="XmlDownloadManager.cs" company="Microsoft">
|
||||
// Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
// </copyright>
|
||||
// <owner current="true" primary="true">[....]</owner>
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
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();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user