5

我正在学习如何使用VS 2015中的默认MVC模板使用OWIN外部验证。我启用了Facebook验证并添加了范围“电子邮件”,期望用户的电子邮件由Facebook返回在用户通过认证后(根据this)。不过,上下文中没有电子邮件。用户JSON对象和上下文。电子邮件也是空的。正在检索OWIN登录用户的Facebook电子邮件

这在Startup.Auth.cs

var facebookOptions = new FacebookAuthenticationOptions 
     { 
      AppId = "XXXXXX", 
      AppSecret = "XXXXXX", 
      Provider = new FacebookAuthenticationProvider 
      { 
       OnAuthenticated = context => 
       { 
        // Retrieve the OAuth access token to store for subsequent API calls 
        var accessToken = context.AccessToken; 

        // Retrieve the username 
        var facebookUserName = context.UserName; 

        // WHY IS IT EMPTY? 
        var facebookEmail = context.Email; 

        // You can even retrieve the full JSON-serialized user 
        var serializedUser = context.User; 

        return Task.FromResult(0); 
       } 
      } 
     }; 

     facebookOptions.Scope.Add("email"); 

     app.UseFacebookAuthentication(facebookOptions); 

任何想法相关的代码缺少什么?为什么Facebook电子邮件地址不被退回?

+4

可能有用的信息在这里:HTTP: //stackoverflow.com/a/32636149/3130094 –

+1

谢谢@JamieDunstan - 这是有帮助的! – Andrey

回答

4

原因是在Facebook API v 2.4中出现了重大变化,您必须指定要检索的字段。所使用的图形要求是:

https://graph.facebook.com/v2.3/me?access_token=XXXXX 

,但由于性能原因为FB API 2.4版本的,你还必须指定的Fileds你想的范围之内检索:

https://graph.facebook.com/v2.4/me?fields=id,name,email&access_token=XXXXX 

微软FB客户端实现默认情况下,重视对的access_token查询字符串作为导致断请求(额外的问号)“的access_token?”:

https://graph.facebook.com/v2.4/me?fields=id,name,email?access_token=XXXXX

所以,为了解决这个问题,我们需要使用自定义的BackchannelHttpHandler。首先,我们创建了端点类:

public class FacebookBackChannelHandler : HttpClientHandler 
    { 
     protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken) 
     { 
      if (!request.RequestUri.AbsolutePath.Contains("/oauth")) 
      { 
       request.RequestUri = new Uri(request.RequestUri.AbsoluteUri.Replace("?access_token", "&access_token")); 
      } 

      return await base.SendAsync(request, cancellationToken); 
     } 
    } 

然后我们提供它在Facebook的身份验证选项有明确指定UserInformationEndpoint沿:

var facebookAuthOptions = new FacebookAuthenticationOptions 
     { 
      AppId = ConfigurationManager.AppSettings["FacebookAppId"], 
      AppSecret = ConfigurationManager.AppSettings["FacebookAppSecret"], 
      BackchannelHttpHandler = new FacebookBackChannelHandler(), 
      UserInformationEndpoint = "https://graph.facebook.com/v2.4/me?fields=id,name,email", 
      Scope = { "email" } 
      <.....> 
     }; 

来源:https://stackoverflow.com/a/32636149/3130094

相关问题