it-swarm.com.ru

Настройка заголовка авторизации HttpClient

У меня есть HttpClient, который я использую для использования REST API. Однако у меня возникают проблемы при настройке заголовка авторизации. Мне нужно установить заголовок для токена, который я получил от выполнения моего запроса OAuth .... Я видел некоторый код для .NET, который предлагает следующее,

httpClient.DefaultRequestHeaders.Authorization = new Credential(OAuth.token);

Однако класс Credential не существует в WinRT. У кого-нибудь есть идеи, как установить заголовок авторизации?

313
Stephen Hynes

Таким образом, способ сделать это заключается в следующем,

httpClient.DefaultRequestHeaders.Authorization =
    new AuthenticationHeaderValue("Bearer", "Your Oauth token");
583
Stephen Hynes
request.DefaultRequestHeaders.Authorization = 
    new AuthenticationHeaderValue(
        "Basic", 
        Convert.ToBase64String(
            System.Text.ASCIIEncoding.ASCII.GetBytes(
                string.Format("{0}:{1}", "yourusername", "yourpwd"))));
228
TheWhiteRabbit

Я ищу хороший способ справиться с этой проблемой, и я смотрю на тот же вопрос. Надеюсь, этот ответ поможет всем, у кого такая же проблема, как я.

using (var client = new HttpClient())
{
    var url = "https://www.theidentityhub.com/{tenant}/api/identity/v1";
    client.DefaultRequestHeaders.Add("Authorization", "Bearer " + accessToken);
    var response = await client.GetStringAsync(url);
    // Parse JSON response.
    ....
}

ссылка с https://www.theidentityhub.com/hub/Documentation/CallTheIdentityHubApi

61
Willie Cheng

Я согласен с ответом TheWhiteRabbit, но если у вас много звонков с использованием HttpClient, код кажется немного повторяющимся, на мой взгляд. 

Я думаю, что есть 2 способа немного улучшить ответ.

Создайте вспомогательный класс для создания клиента:

public static class ClientHelper
{
    // Basic auth
    public static HttpClient GetClient(string username,string password)
    {
            var authValue = new AuthenticationHeaderValue("Basic", Convert.ToBase64String(Encoding.UTF8.GetBytes($"{username}:{password}")));

            var client = new HttpClient(){
                DefaultRequestHeaders = { Authorization = authValue}
                //Set some other client defaults like timeout / BaseAddress
            };
            return client;
    }

    // Auth with bearer token
    public static HttpClient GetClient(string token)
    {
            var authValue = new AuthenticationHeaderValue("Bearer", token);

            var client = new HttpClient(){
                DefaultRequestHeaders = { Authorization = authValue}
                //Set some other client defaults like timeout / BaseAddress
            };
            return client;
    }
}

Использование:

using(var client = ClientHelper.GetClient(username,password))
{
    //Perform some http call
}

using(var client = ClientHelper.GetClient(token))
{
    //Perform some http call
}

Создайте метод расширения:

Не выигрывает приз за красоту, но отлично работает :)

    public static class HttpClientExtentions
    {
        public static AuthenticationHeaderValue ToAuthHeaderValue(this string username, string password)
        {
            return new AuthenticationHeaderValue("Basic",
        Convert.ToBase64String(
            System.Text.Encoding.ASCII.GetBytes(
                $"{username}:{password}")));
        }
    }

Использование:

using (var client = new HttpClient())
{
    client.DefaultRequestHeaders.Authorization = _username.ToAuthHeaderValue(_password); 
}

Опять же, я думаю, что 2 вышеупомянутых варианта делают использование клиента чуть менее повторяющимся. Имейте в виду, что рекомендуется повторно использовать HttpClient, если вы делаете несколько http-вызовов, но я думаю, что это немного выходит за рамки этого вопроса.

33
Florian Schaal

Поскольку хорошая практика - повторно использовать экземпляр HttpClient, для проблемы производительности и исчерпания порта, и поскольку ни один из ответов не дает такого решения (и даже не ведет вас к плохим практикам :(), Я поставил здесь ссылку на ответ, который я дал на похожий вопрос:

https://stackoverflow.com/a/40707446/717372

Некоторые источники о том, как правильно использовать HttpClient:

25
Philippe

Чтобы установить базовую аутентификацию с C # HttpClient. Следующий код работает для меня. 

   using (var client = new HttpClient())
        {
            var webUrl ="http://localhost/saleapi/api/";
            var uri = "api/sales";
            client.BaseAddress = new Uri(webUrl);
            client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
            client.DefaultRequestHeaders.ConnectionClose = true;

            //Set Basic Auth
            var user = "username";
            var password = "password";
            var base64String =Convert.ToBase64String( Encoding.ASCII.GetBytes($"{user}:{password}"));
            client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic",base64String);

            var result = await client.PostAsJsonAsync(uri, model);
            return result;
        }
9
LENG UNG

Вот как я это сделал:

using (HttpClient httpClient = new HttpClient())
{
   Dictionary<string, string> tokenDetails = null;
   var messageDetails = new Message { Id = 4, Message1 = des };
   HttpClient client = new HttpClient();
   client.BaseAddress = new Uri("http://localhost:3774/");
   var login = new Dictionary<string, string>
       {
           {"grant_type", "password"},
           {"username", "[email protected]"},
           {"password", "[email protected]"},
       };
   var response = client.PostAsync("Token", new FormUrlEncodedContent(login)).Result;
   if (response.IsSuccessStatusCode)
   {
      tokenDetails = JsonConvert.DeserializeObject<Dictionary<string, string>>(response.Content.ReadAsStringAsync().Result);
      if (tokenDetails != null && tokenDetails.Any())
      {
         var tokenNo = tokenDetails.FirstOrDefault().Value;
         client.DefaultRequestHeaders.Add("Authorization", "Bearer " + tokenNo);
         client.PostAsJsonAsync("api/menu", messageDetails)
             .ContinueWith((postTask) => postTask.Result.EnsureSuccessStatusCode());
      }
   }
}

Это видео, которое вы мне помогли, очень мне помогло. Пожалуйста, проверьте это . https://www.youtube.com/watch?v=qCwnU06NV5Q

7
Dayan

Использовать базовую авторизацию и параметры Json.

using (HttpClient client = new HttpClient())
                    {
                        var request_json = "your json string";

                        var content = new StringContent(request_json, Encoding.UTF8, "application/json");

                        var authenticationBytes = Encoding.ASCII.GetBytes("YourUsername:YourPassword");

                        client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic",
                               Convert.ToBase64String(authenticationBytes));
                        client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));

                        var result = await client.PostAsync("YourURL", content);

                        var result_string = await result.Content.ReadAsStringAsync();
                    }
5
MohammadSoori

Использование класса AuthenticationHeaderValue сборки System.Net.Http

public AuthenticationHeaderValue(
    string scheme,
    string parameter
)

мы можем установить или обновить существующий заголовок Authorization для нашего httpclient следующим образом:

httpclient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", TokenResponse.AccessToken);
3
Fusion

6 лет спустя, но добавьте это на случай, если это кому-то поможет.

https://www.codeproject.com/Tips/996401/Authenticate-WebAPIs-with-Basic-and-Windows-Authen

var authenticationBytes = Encoding.ASCII.GetBytes("<username>:<password>");
using (HttpClient confClient = new HttpClient())
{
  confClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", 
         Convert.ToBase64String(authenticationBytes));
  confClient.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue(Constants.MediaType));  
  HttpResponseMessage message = confClient.GetAsync("<service URI>").Result;
  if (message.IsSuccessStatusCode)
  {
    var inter = message.Content.ReadAsStringAsync();
    List<string> result = JsonConvert.DeserializeObject<List<string>>(inter.Result);
  }
}
1
MPJ567

Я устанавливал токен канала-носителя httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue ("Bearer", токен);

Это работало в одной конечной точке, но не в другой. Проблема состояла в том, что у меня была строчная буква "b" на "предъявителя". После изменения теперь это работает для обоих API, я бью. Так легко пропустить, если вы даже не рассматриваете это как один из стогов сена, чтобы искать иглу.

Обязательно иметь «Несущий» - с большой буквы.

0
Alan Ball