using System.Threading.Tasks;
using log4net;
+
namespace Pithos.Network
{
using System;
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;
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
public string DownloadStringWithRetry(string address,int retries=0)
{
+
if (address == null)
throw new ArgumentNullException("address");
TraceStart("GET",actualAddress);
var actualRetries = (retries == 0) ? Retries : retries;
-
var task = Retry(() =>
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)
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);
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