Todas as respostas anteriores descrevem o problema sem fornecer uma solução. Aqui está um método de extensão que resolve o problema, permitindo que você defina qualquer cabeçalho através do nome da string.
Uso
HttpWebRequest request = WebRequest.Create(url) as HttpWebRequest;
request.SetRawHeader("content-type", "application/json");
Classe de extensão
public static class HttpWebRequestExtensions
{
static string[] RestrictedHeaders = new string[] {
"Accept",
"Connection",
"Content-Length",
"Content-Type",
"Date",
"Expect",
"Host",
"If-Modified-Since",
"Keep-Alive",
"Proxy-Connection",
"Range",
"Referer",
"Transfer-Encoding",
"User-Agent"
};
static Dictionary<string, PropertyInfo> HeaderProperties = new Dictionary<string, PropertyInfo>(StringComparer.OrdinalIgnoreCase);
static HttpWebRequestExtensions()
{
Type type = typeof(HttpWebRequest);
foreach (string header in RestrictedHeaders)
{
string propertyName = header.Replace("-", "");
PropertyInfo headerProperty = type.GetProperty(propertyName);
HeaderProperties[header] = headerProperty;
}
}
public static void SetRawHeader(this HttpWebRequest request, string name, string value)
{
if (HeaderProperties.ContainsKey(name))
{
PropertyInfo property = HeaderProperties[name];
if (property.PropertyType == typeof(DateTime))
property.SetValue(request, DateTime.Parse(value), null);
else if (property.PropertyType == typeof(bool))
property.SetValue(request, Boolean.Parse(value), null);
else if (property.PropertyType == typeof(long))
property.SetValue(request, Int64.Parse(value), null);
else
property.SetValue(request, value, null);
}
else
{
request.Headers[name] = value;
}
}
}
Cenários
Eu escrevi um wrapper para HttpWebRequest
e não queria expor todos os 13 cabeçalhos restritos como propriedades no meu wrapper. Em vez disso, eu queria usar um simples Dictionary<string, string>
.
Outro exemplo é um proxy HTTP, no qual você precisa pegar os cabeçalhos de uma solicitação e encaminhá-los ao destinatário.
Existem muitos outros cenários em que simplesmente não é prático ou possível usar propriedades. Forçar o usuário a definir o cabeçalho por meio de uma propriedade é um design muito inflexível e é por isso que a reflexão é necessária. O lado positivo é que a reflexão é abstraída, ainda é rápida (0,001 segundo nos meus testes) e, como um método de extensão, parece natural.
Notas
Os nomes de cabeçalho não diferenciam maiúsculas de minúsculas de acordo com o RFC, http://www.w3.org/Protocols/rfc2616/rfc2616-sec4.html#sec4.2