programing tip

웹 API 호출시 C # 지원되지 않는 권한 부여 유형

itbloger 2020. 12. 14. 07:56
반응형

웹 API 호출시 C # 지원되지 않는 권한 부여 유형


ac # WPF 데스크톱 앱에서 내 WebAPI에 게시를 수행하려고합니다.

내가 뭘해도 난

{ "error": "unsupported_grant_type"}

이것은 내가 시도한 것입니다 (그리고 내가 찾을 수있는 모든 것을 시도했습니다) :

또한 현재 테스트를 위해 활성화 된 dev 웹 API : http://studiodev.biz/

기본 http 클라이언트 개체 :

var client = new HttpClient()
client.BaseAddress = new Uri("http://studiodev.biz/");
client.DefaultRequestHeaders.Accept.Clear();
client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json"));
client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("text/plain"));
client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("*/*"));

다음 전송 방법으로 :

var response = await client.PostAsJsonAsync("token", "{'grant_type'='password'&'username'='username'&'password'='password'");
var response = await client.PostAsJsonAsync("token", "grant_type=password&username=username&password=password");

실패한 후 인터넷 검색을 수행하고 시도했습니다.

LoginModel data = new LoginModel(username, password);
string json = JsonConvert.SerializeObject(data);
await client.PostAsync("token", new JsonContent(json));

같은 결과이므로 시도했습니다.

req.Content = new StringContent(json, Encoding.UTF8, "application/x-www-form-urlencoded");
await client.SendAsync(req).ContinueWith(respTask =>
{
 Application.Current.Dispatcher.Invoke(new Action(() => { label.Content = respTask.Result.ToString(); }));
});

참고 : Chrome으로 성공적으로 전화를 걸 수 있습니다.

Fiddler 결과 업데이트

여기에 이미지 설명 입력

누군가가 위의 웹 API를 성공적으로 호출하도록 도와 주시겠습니까 ... 제가 명확히 도울 수 있으면 알려주십시오. 감사!!


의 기본 구현은 JSON 인코딩 ( )이 아닌 OAuthAuthorizationServerHandler양식 인코딩 (예 :) 만 허용합니다 .application/x-www-form-urlencodedapplication/JSON

귀하의 요청 ContentType은 다음 application/x-www-form-urlencoded과 같이 본문의 데이터를 전달 해야합니다 .

grant_type=password&username=Alice&password=password123

, JSON 형식이 아닙니다 .

위의 크롬 예제는 JSON으로 데이터를 전달하지 않기 때문에 작동합니다. 토큰을 얻는 데만 필요합니다. API의 다른 메소드의 경우 JSON을 사용할 수 있습니다.

이러한 종류의 문제도 여기서 설명 합니다 .


1) URL : "localhost : 55828 / token"( "localhost : 55828 / API / token"아님)

2) 요청 데이터를 기록합니다. json 형식이 아니라 큰 따옴표가없는 일반 데이터입니다. "userName=xxx@gmail.com&password=Test123$&grant_type=password"

3) 콘텐츠 유형을 확인합니다. Content-Type : 'application / x-www-form-urlencoded'(Content-Type이 아님 : 'application / json')

4) javascript를 사용하여 게시 요청을 할 때 다음을 사용할 수 있습니다.

$http.post("localhost:55828/token", 
    "userName=" + encodeURIComponent(email) +
        "&password=" + encodeURIComponent(password) +
        "&grant_type=password",
    {headers: { 'Content-Type': 'application/x-www-form-urlencoded' }}
).success(function (data) {//...

아래의 Postman 스크린 샷을 참조하십시오.

우편 배달부 요청

우편 배달부 요청 헤더


다음은 SSL을 사용하여 포트 43305에서 실행되는 로컬 웹 API 응용 프로그램의이 요청을 수행하는 데 사용한 작업 예제입니다. GitHub에도 프로젝트를 올렸습니다. https://github.com/casmer/WebAPI-getauthtoken

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Newtonsoft.Json;
using System.Net.Http;
using System.Web;

namespace GetAccessTokenSample
{
  class Program
  {
    private static string baseUrl = "https://localhost:44305";

    static void Main(string[] args)
    {

      Console.WriteLine("Enter Username: ");
      string username= Console.ReadLine();
      Console.WriteLine("Enter Password: ");
      string password = Console.ReadLine();

      LoginTokenResult accessToken = GetLoginToken(username,password);
      if (accessToken.AccessToken != null)
      {
        Console.WriteLine(accessToken);
      }
      else
      {
        Console.WriteLine("Error Occurred:{0}, {1}", accessToken.Error, accessToken.ErrorDescription);
      }

    }


    private static LoginTokenResult GetLoginToken(string username, string password)
    {

      HttpClient client = new HttpClient();
      client.BaseAddress = new Uri(baseUrl);
      //TokenRequestViewModel tokenRequest = new TokenRequestViewModel() { 
      //password=userInfo.Password, username=userInfo.UserName};
      HttpResponseMessage response =
        client.PostAsync("Token",
          new StringContent(string.Format("grant_type=password&username={0}&password={1}",
            HttpUtility.UrlEncode(username),
            HttpUtility.UrlEncode(password)), Encoding.UTF8,
            "application/x-www-form-urlencoded")).Result;

      string resultJSON = response.Content.ReadAsStringAsync().Result;
      LoginTokenResult result = JsonConvert.DeserializeObject<LoginTokenResult>(resultJSON);

      return result;
    }

    public class LoginTokenResult
    {
      public override string ToString()
      {
        return AccessToken;
      }

      [JsonProperty(PropertyName = "access_token")]
      public string AccessToken { get; set; }

      [JsonProperty(PropertyName = "error")]
      public string Error { get; set; }

      [JsonProperty(PropertyName = "error_description")]
      public string ErrorDescription { get; set; }

    }

  }
}

RestSharp를 사용하는 경우 다음과 같이 요청해야합니다.

public static U PostLogin<U>(string url, Authentication obj)
            where U : new()
        {
            RestClient client = new RestClient();
            client.BaseUrl = new Uri(host + url);
            var request = new RestRequest(Method.POST);
            string encodedBody = string.Format("grant_type=password&username={0}&password={1}",
                obj.username,obj.password);
            request.AddParameter("application/x-www-form-urlencoded", encodedBody, ParameterType.RequestBody);
            request.AddParameter("Content-Type", "application/x-www-form-urlencoded", ParameterType.HttpHeader);
            var response = client.Execute<U>(request);

             return response.Data;

        }

동일한 문제가 있었지만 토큰 URL에 대해 보안 HTTP를 통해서만 내 문제를 해결했습니다. 샘플 httpclient 코드를 참조하십시오. 일반 HTTP는 서버 유지 관리 후 작동을 중지합니다.

var apiUrl = "https://appdomain.com/token"
var client = new HttpClient();    
client.Timeout = new TimeSpan(1, 0, 0);
            var loginData = new Dictionary<string, string>
                {
                    {"UserName", model.UserName},
                    {"Password", model.Password},
                    {"grant_type", "password"}
                };
            var content = new FormUrlEncodedContent(loginData);
            var response = client.PostAsync(apiUrl, content).Result;

참고 URL : https://stackoverflow.com/questions/29246908/c-sharp-unsupported-grant-type-when-calling-web-api

반응형