using System;
using System.Collections.Generic;
using System.Drawing;
using System.IO;
using System.Net;
using System.Text;
using System.Threading;
namespace ClassicalSharp.Network {
	
	///  Specialised producer and consumer queue for downloading data asynchronously. 
	public class AsyncDownloader : IDisposable {
		
		EventWaitHandle handle = new EventWaitHandle( false, EventResetMode.AutoReset );
		Thread worker;
		readonly object requestLocker = new object();
		List requests = new List();
		readonly object downloadedLocker = new object();
		Dictionary downloaded = new Dictionary();
		string skinServer = null;
		
		public AsyncDownloader( string skinServer ) {
			this.skinServer = skinServer;
			WebRequest.DefaultWebProxy = null;
			
			worker = new Thread( DownloadThreadWorker, 256 * 1024 );
			worker.Name = "ClassicalSharp.ImageDownloader";
			worker.IsBackground = true;
			worker.Start();
		}
		
		///  Asynchronously downloads a skin. If 'skinName' points to the url then the skin is
		/// downloaded from that url, otherwise it is downloaded from the url 'defaultSkinServer'/'skinName'.png 
		///  Identifier is skin_'skinName'.
		public void DownloadSkin( string identifier, string skinName ) {
			string strippedSkinName = Utils.StripColours( skinName );
			string url = Utils.IsUrlPrefix( skinName ) ? skinName :
				skinServer + strippedSkinName + ".png";
			AddRequest( url, true, identifier, RequestType.Bitmap, DateTime.MinValue );
		}
		
		///  Asynchronously downloads a bitmap image from the specified url.  
		public void DownloadImage( string url, bool priority, string identifier ) {
			AddRequest( url, priority, identifier, RequestType.Bitmap, DateTime.MinValue );
		}
		
		///  Asynchronously downloads a string from the specified url.  
		public void DownloadPage( string url, bool priority, string identifier ) {
			AddRequest( url, priority, identifier, RequestType.String, DateTime.MinValue );
		}
		
		///  Asynchronously downloads a byte array from the specified url.  
		public void DownloadData( string url, bool priority, string identifier ) {
			AddRequest( url, priority, identifier, RequestType.ByteArray, DateTime.MinValue );
		}
		
		///  Asynchronously downloads a bitmap image from the specified url.  
		public void DownloadImage( string url, bool priority, string identifier, DateTime lastModified ) {
			AddRequest( url, priority, identifier, RequestType.Bitmap, lastModified );
		}
		
		///  Asynchronously downloads a byte array from the specified url.  
		public void DownloadData( string url, bool priority, string identifier, DateTime lastModified ) {
			AddRequest( url, priority, identifier, RequestType.ByteArray, lastModified );
		}
		
		///  Asynchronously retrieves the content length of the body response from specified url.  
		public void RetrieveContentLength( string url, bool priority, string identifier ) {
			AddRequest( url, priority, identifier, RequestType.ContentLength, DateTime.MinValue );
		}
		
		void AddRequest( string url, bool priority, string identifier, RequestType type, DateTime lastModified ) {
			lock( requestLocker )  {
				Request request = new Request( url, identifier, type, lastModified );
				if( priority ) {
					requests.Insert( 0, request );
				} else {
					requests.Add( request );
				}
			}
			handle.Set();
		}
		
		///  Informs the asynchronous thread that it should stop processing further requests
		/// and can consequentially exit the for loop.
		/// Note that this will *block** the calling thread as the method waits until the asynchronous
		/// thread has exited the for loop. 
		public void Dispose() {
			lock( requestLocker )  {
				requests.Insert( 0, null );
			}
			
			handle.Set();
			worker.Join();
			((IDisposable)handle).Dispose();
		}
		
		///  Removes older entries that were downloaded a certain time ago
		/// but were never removed from the downloaded queue. 
		public void PurgeOldEntries( int seconds ) {
			lock( downloadedLocker ) {
				DateTime now = DateTime.UtcNow;
				List itemsToRemove = new List( downloaded.Count );
				
				foreach( var item in downloaded ) {
					DateTime timestamp = item.Value.TimeDownloaded;
					if( (now - timestamp).TotalSeconds > seconds ) {
						itemsToRemove.Add( item.Key );
					}
				}
				
				for( int i = 0; i < itemsToRemove.Count; i++ ) {
					string key = itemsToRemove[i];
					DownloadedItem item;
					downloaded.TryGetValue( key, out item );
					downloaded.Remove( key );
					Bitmap bmp = item.Data as Bitmap;
					if( bmp != null )
						bmp.Dispose();
				}
			}
		}
		
		///  Returns whether the requested item exists in the downloaded queue.
		/// If it does, it removes the item from the queue and outputs it. 
		///  If the asynchronous thread failed to download the item, this method
		/// will return 'true' and 'item' will be set. However, the contents of the 'item' object will be null.
		public bool TryGetItem( string identifier, out DownloadedItem item ) {
			bool success = false;
			lock( downloadedLocker ) {
				success = downloaded.TryGetValue( identifier, out item );
				if( success ) {
					downloaded.Remove( identifier );
				}
			}
			return success;
		}
		void DownloadThreadWorker() {
			while( true ) {
				Request request = null;
				lock( requestLocker ) {
					if( requests.Count > 0 ) {
						request = requests[0];
						requests.RemoveAt( 0 );
						if( request == null )
							return;
					}
				}
				if( request != null ) {
					ProcessRequest( request );
				} else {
					handle.WaitOne();
				}
			}
		}
		
		void ProcessRequest( Request request ) {
			string url = request.Url;
			Utils.LogDebug( "Downloading {0} from: {1}", request.Type, url );
			object value = null;
			WebException webEx = null;
			
			try {
				HttpWebRequest req = MakeRequest( request );
				using( HttpWebResponse response = (HttpWebResponse)req.GetResponse() )
					value = DownloadContent( request, response );
			} catch( Exception ex ) {
				if( !( ex is WebException || ex is ArgumentException || ex is UriFormatException ) ) throw;
				Utils.LogDebug( "Failed to download from: " + url );
				
				if( ex is WebException )
					webEx = (WebException)ex;
			}
			value = CheckIsValidImage( value, url );
			lock( downloadedLocker ) {
				DownloadedItem oldItem;
				DownloadedItem newItem = new DownloadedItem( value, request.TimeAdded, url, webEx );
				
				if( downloaded.TryGetValue( request.Identifier, out oldItem ) ) {
					if( oldItem.TimeAdded > newItem.TimeAdded ) {
						DownloadedItem old = oldItem;
						oldItem = newItem;
						newItem = old;
					}
					Bitmap oldBmp = oldItem.Data as Bitmap;
					if( oldBmp != null )
						oldBmp.Dispose();
				}
				downloaded[request.Identifier] = newItem;
			}
		}
		
		object DownloadContent( Request request, HttpWebResponse response ) {
			if( request.Type == RequestType.Bitmap ) {
				MemoryStream data = DownloadBytes( response );
				return new Bitmap( data );
			} else if( request.Type == RequestType.String ) {
				MemoryStream data = DownloadBytes( response );
				byte[] rawBuffer = data.GetBuffer();
				return Encoding.UTF8.GetString( rawBuffer, 0, (int)data.Length );
			} else if( request.Type == RequestType.ByteArray ) {
				MemoryStream data = DownloadBytes( response );
				return data.ToArray();
			} else if( request.Type == RequestType.ContentLength ) {
				return response.ContentLength;
			}
			return null;
		}
		
		object CheckIsValidImage( object value, string url ) {
			// Mono seems to be returning a bitmap with a native pointer of zero in some weird cases.
			// We can detect this as every single property access raises an ArgumentException.
			try {
				Bitmap bmp = value as Bitmap;
				if( bmp != null ) {
					int height = bmp.Height;
				}
				return value;
			} catch( ArgumentException ) {
				Utils.LogDebug( "Failed to download from: " + url );
				return null;
			}
		}
		
		HttpWebRequest MakeRequest( Request request ) {
			HttpWebRequest req = (HttpWebRequest)WebRequest.Create( request.Url );
			req.AutomaticDecompression = DecompressionMethods.GZip;
			req.ReadWriteTimeout = 15 * 1000;
			req.Timeout = 15 * 1000;
			req.Proxy = null;
			req.UserAgent = Program.AppName;
			
			if( request.LastModified != DateTime.MinValue )
				req.IfModifiedSince = request.LastModified;
			return req;
		}
		
		static byte[] buffer = new byte[4096 * 8];
		MemoryStream DownloadBytes( HttpWebResponse response ) {
			int length = (int)response.ContentLength;
			MemoryStream dst = length >= 0 ?
				new MemoryStream( length ) : new MemoryStream();
			
			using( Stream src = response.GetResponseStream() ) {
				int read = 0;
				while( (read = src.Read( buffer, 0, buffer.Length )) > 0 ) {
					dst.Write( buffer, 0, read );
				}
			}
			return dst;
		}
		
		sealed class Request {
			
			public string Url;
			public string Identifier;
			public RequestType Type;
			public DateTime TimeAdded;
			public DateTime LastModified;
			
			public Request( string url, string identifier, RequestType type, DateTime lastModified ) {
				Url = url;
				Identifier = identifier;
				Type = type;
				TimeAdded = DateTime.UtcNow;
				LastModified = lastModified;
			}
		}
	}
	
	public enum RequestType {
		Bitmap,
		String,
		ByteArray,
		ContentLength,
	}
	
	///  Represents an item that was asynchronously downloaded. 
	public class DownloadedItem {
		
		///  Contents that were downloaded. 
		public object Data;
		
		///  Instant in time the item was originally added to the download queue. 
		public DateTime TimeAdded;
		
		///  Instant in time the item was fully downloaded. 
		public DateTime TimeDownloaded;
		
		///  Full URL this item was downloaded from. 
		public string Url;
		
		///  Exception that occurred if this request failed, can be null. 
		public WebException WebEx;
		
		public DownloadedItem( object data, DateTime timeAdded, string url, WebException webEx ) {
			Data = data;
			TimeAdded = timeAdded;
			TimeDownloaded = DateTime.UtcNow;
			Url = url;
			WebEx = webEx;
		}
	}
}