Fixed frequent timeout by always closing WebResponse objects
[pithos-ms-client] / trunk / Pithos.Network / RestClient.cs
index b6b5c98..a2c5e90 100644 (file)
@@ -13,6 +13,7 @@ using System.Runtime.Serialization;
 using System.Threading.Tasks;
 using log4net;
 
+
 namespace Pithos.Network
 {
     using System;
@@ -83,10 +84,31 @@ namespace Pithos.Network
             this.Proxy = other.Proxy;
         }
 
+
+        private WebHeaderCollection _responseHeaders;
+
+        public new WebHeaderCollection ResponseHeaders
+        {
+            get
+            {
+                if (base.ResponseHeaders==null)
+                {
+                    return _responseHeaders;
+                }
+                else
+                {
+                    _responseHeaders = null;
+                    return base.ResponseHeaders;   
+                }
+                
+            }
+
+            set { _responseHeaders = value; }
+        } 
         protected override WebRequest GetWebRequest(Uri address)
         {
             TimedOut = false;
-            var webRequest = base.GetWebRequest(address);
+            var webRequest = base.GetWebRequest(address);            
             var request = (HttpWebRequest)webRequest;
             if (IfModifiedSince.HasValue)
                 request.IfModifiedSince = IfModifiedSince.Value;
@@ -106,50 +128,73 @@ namespace Pithos.Network
 
         public DateTime? IfModifiedSince { get; set; }
 
+        //Asynchronous version
         protected override WebResponse GetWebResponse(WebRequest request, IAsyncResult result)
         {
-            return ProcessResponse(()=>base.GetWebResponse(request, result));
-        }
+            Log.InfoFormat("ASYNC [{0}] {1}",request.Method, request.RequestUri);
+            HttpWebResponse response = null;
+
+            try
+            {
+                response = (HttpWebResponse)base.GetWebResponse(request, result);
+            }
+            catch (WebException exc)
+            {
+                if (!TryGetResponse(exc, out response))
+                    throw;
+            }
+
+            StatusCode = response.StatusCode;
+            LastModified = response.LastModified;
+            StatusDescription = response.StatusDescription;
+            return response;
 
-        protected override WebResponse GetWebResponse(WebRequest request)
-        {
-            return ProcessResponse(() => base.GetWebResponse(request));
         }
+      
 
-        private WebResponse ProcessResponse(Func<WebResponse> getResponse)
+        //Synchronous version
+        protected override WebResponse GetWebResponse(WebRequest request)
         {
+            HttpWebResponse response = null;
             try
-            {
-                var response = (HttpWebResponse)getResponse();
-                StatusCode = response.StatusCode;
-                LastModified = response.LastModified;
-                StatusDescription = response.StatusDescription;
-                return response;
+            {                                
+                response = (HttpWebResponse)base.GetWebResponse(request);
             }
             catch (WebException exc)
             {
-                if (exc.Response != null)
-                {
-                    var response = (exc.Response as HttpWebResponse);
-                    if (AllowedStatusCodes.Contains(response.StatusCode))
-                    {
-                        StatusCode = response.StatusCode;
-                        LastModified = response.LastModified;
-                        StatusDescription = response.StatusDescription;
+                if (!TryGetResponse(exc, out response))
+                    throw;
+            }
 
-                        return response;
-                    }
-                    if (exc.Response.ContentLength > 0)
-                    {
-                        string content = GetContent(exc.Response);
-                        Log.ErrorFormat(content);
-                    }
-                }
-                throw;
+            StatusCode = response.StatusCode;
+            LastModified = response.LastModified;
+            StatusDescription = response.StatusDescription;
+            return response;
+        }
+
+        private bool TryGetResponse(WebException exc, out HttpWebResponse response)
+        {
+            response = null;
+            //Fail on empty response
+            if (exc.Response == null)
+                return false;
+
+            response = (exc.Response as HttpWebResponse);
+            //Succeed on allowed status codes
+            if (AllowedStatusCodes.Contains(response.StatusCode))
+                return true;
+
+            //Does the response have any content to log?
+            if (exc.Response.ContentLength > 0)
+            {
+                var content = GetContent(exc.Response);
+                Log.ErrorFormat(content);
             }
+            return false;
         }
 
-        private readonly List<HttpStatusCode> _allowedStatusCodes=new List<HttpStatusCode>{HttpStatusCode.NotModified};
+        private readonly List<HttpStatusCode> _allowedStatusCodes=new List<HttpStatusCode>{HttpStatusCode.NotModified};        
+
         public List<HttpStatusCode> AllowedStatusCodes
         {
             get
@@ -177,6 +222,7 @@ namespace Pithos.Network
 
         public string DownloadStringWithRetry(string address,int retries=0)
         {
+            
             if (address == null)
                 throw new ArgumentNullException("address");
 
@@ -185,7 +231,6 @@ namespace Pithos.Network
             TraceStart("GET",actualAddress);            
             
             var actualRetries = (retries == 0) ? Retries : retries;
-            
 
             
             var task = Retry(() =>
@@ -219,17 +264,27 @@ namespace Pithos.Network
             RetryWithoutContent(address, retries, "DELETE");
         }
 
-        public string GetHeaderValue(string headerName)
+        public string GetHeaderValue(string headerName,bool optional=false)
         {
             if (this.ResponseHeaders==null)
                 throw new InvalidOperationException("ResponseHeaders are null");
             Contract.EndContractBlock();
 
             var values=this.ResponseHeaders.GetValues(headerName);
-            if (values == null)
-                throw new WebException(String.Format("The {0}  header is missing", headerName));
-            else
+            if (values != null)
                 return values[0];
+
+            if (optional)            
+                return null;            
+            //A required header was not found
+            throw new WebException(String.Format("The {0}  header is missing", headerName));
+        }
+
+        public void SetNonEmptyHeaderValue(string headerName, string value)
+        {
+            if (String.IsNullOrWhiteSpace(value))
+                return;
+            Headers.Add(headerName,value);
         }
 
         private void RetryWithoutContent(string address, int retries, string method)
@@ -252,10 +307,16 @@ namespace Pithos.Network
                 TraceStart(method, uriString);
                 if (method == "PUT")
                     request.ContentLength = 0;
+
                 var response = (HttpWebResponse)GetWebResponse(request);
-                StatusCode = response.StatusCode;
-                StatusDescription = response.StatusDescription;                
+                //var response = (HttpWebResponse)request.GetResponse();
                 
+                //ResponseHeaders= response.Headers;
+
+                LastModified = response.LastModified;
+                StatusCode = response.StatusCode;
+                StatusDescription = response.StatusDescription;
+                response.Close();
 
                 return 0;
             }, actualRetries);
@@ -439,6 +500,21 @@ namespace Pithos.Network
             var builder = new UriBuilder(String.Join("/", BaseAddress, container, objectName));
             return builder;
         }
+
+        public Dictionary<string, string> GetMeta(string metaPrefix)
+        {
+            if (String.IsNullOrWhiteSpace(metaPrefix))
+                throw new ArgumentNullException("metaPrefix");
+            Contract.EndContractBlock();
+
+            var keys = ResponseHeaders.AllKeys.AsQueryable();
+            var dict = (from key in keys
+                        where key.StartsWith(metaPrefix)
+                        let name = key.Substring(metaPrefix.Length)
+                        select new { Name = name, Value = ResponseHeaders[key] })
+                        .ToDictionary(t => t.Name, t => t.Value);
+            return dict;
+        }
     }
 
     public class RetryException:Exception