- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
创建.NET的最简单方法是什么? ASP.NET MVC Facebook 申请OAuth有明确的范围吗?
我尝试了很多例子。 OAuthWebSecurity.RegisterClient 不支持添加更多 Facebook 范围。 Facebook application template创建一个 Canvas ,我需要非 Canvas 应用程序。 FacebookScopedClient 不完整,无法使用此修复程序。
你有什么建议?
我也开放 JavaScript/jQuery 解决方案。
最佳答案
可能我来得太晚了,但几天前我遇到了这个问题,我在网上发现了一些糟糕的解决方案。所以我写了自己的解决方案,现在只能发布。
我利用了 DotNetOpenAuth.AspNet.Clients.OAuth2Client 类来完成大部分工作。我只是将其扩展为包括范围和额外的用户数据。
public class FacebookExtendedClient : DotNetOpenAuth.AspNet.Clients.OAuth2Client
{
protected FacebookClient facebookClient;
protected string fields;
protected string scope;
protected Func<string, object, string> fieldTransformer;
protected bool emailAsUsername;
protected IDictionary<string, string> userData;
private string[] splittedFields;
private string[] splittedScope;
protected const string serviceLoginBaseUrl = "https://www.facebook.com/dialog/oauth";
protected const string serviceMeBaseUrl = "https://graph.facebook.com/me";
protected const string serviceAccessTokenBaseUrl = "https://graph.facebook.com/oauth/access_token";
/// <summary>
/// Create an instrance of the class.
/// </summary>
/// <param name="appId">The App ID of the application used to connect to Facebook service.</param>
/// <param name="appSecret">The App Secret of the application used to connect to Facebook service.</param>
/// <param name="fields">
/// String containing comma separated fields to add to the request.
/// If empty the request will retrieve the default fields based of the specified scope.
/// </param>
/// <param name="fieldTransformer">
/// Function to be applied to the values retrived from facebook.
/// If null provided the method will try to cast values from object to string explicitly,
/// an InvalidCastException will be thrown if the cast will not be possible.
/// </param>
/// <param name="scope">
/// String containing comma separated permissions to add to the request.
/// If empty the request will have the basic scope.
/// </param>
/// <param name="emailAsUsername">Makes the email of the facebook user used as authentication username.</param>
public FacebookExtendedClient(string appId, string appSecret, string fields = "", Func<string, object, string> fieldTransformer = null, string scope = "", bool emailAsUsername = false)
: base("facebook")
{
if (string.IsNullOrEmpty(appId))
throw new ArgumentException("The appId argument can not be null or empty.", "appId");
if (string.IsNullOrEmpty(appSecret))
throw new ArgumentException("The appSecret argument can not be null or empty.", "appSecret");
fields = fields.Replace(" ", "");
scope = scope.Replace(" ", "");
this.splittedFields = fields.Split(',');
this.splittedScope = scope.Split(',');
if (emailAsUsername == true && !this.splittedFields.Contains("email") && !this.splittedScope.Contains("email"))
throw new ArgumentException("The scope argument must contain the 'email' permission and the 'email' field to allow emailAsUsername to true.", "scope");
this.facebookClient = new FacebookClient();
this.facebookClient.AppId = appId;
this.facebookClient.AppSecret = appSecret;
this.fields = fields;
this.fieldTransformer = fieldTransformer;
this.scope = scope;
this.emailAsUsername = emailAsUsername;
}
public FacebookClient FacebookClient
{
get
{
return this.facebookClient;
}
}
public IDictionary<string, string> UserData
{
get
{
return this.userData;
}
}
protected override Uri GetServiceLoginUrl(Uri returnUrl)
{
Dictionary<string, object> parameters = new Dictionary<string, object>();
parameters.Add("redirect_uri", returnUrl.AbsoluteUri);
if (!string.IsNullOrEmpty(this.scope))
parameters.Add("scope", this.scope);
return this.facebookClient.GetLoginUrl(parameters);
}
protected override IDictionary<string, string> GetUserData(string accessToken)
{
// This method makes the AuthenticationResult's UserName property be the facebook username of the logged user,
// but if the facebook username is missing the facebook id will be used.
// If emailAsUsername is true then AuthenticationResult's UserName property is the email retrieved from facebook
// and the facebook username can be retrieved by the key "fb_username" in this.userData
FacebookClient facebookClient = new FacebookClient(accessToken);
var getResult = facebookClient.Get<IDictionary<string, object>>("me", new { fields = this.fields });
Dictionary<string, string> result = new Dictionary<string, string>();
if (this.fieldTransformer != null)
{
foreach (var pair in getResult)
result.Add(pair.Key, this.fieldTransformer(pair.Key, pair.Value));
}
else
{
foreach (var pair in getResult)
{
string value = pair.Value.ToString();
if (value == null)
throw new InvalidCastException("Cast not possible for the object associate to the key '" + pair.Key + "'.");
result.Add(pair.Key, value);
}
}
if (this.splittedFields.Contains("username"))
result["fb_username"] = result["username"];
if (this.emailAsUsername)
result["username"] = result["email"];
this.userData = result;
return result;
}
protected override string QueryAccessToken(Uri returnUrl, string authorizationCode)
{
UriBuilder builder = new UriBuilder(serviceAccessTokenBaseUrl);
builder.Query = string.Format("client_id={0}&client_secret={1}&redirect_uri={2}&code={3}",
this.facebookClient.AppId, this.facebookClient.AppSecret, HttpUtility.UrlEncode(Encoding.ASCII.GetBytes(returnUrl.AbsoluteUri)), authorizationCode);
using (WebClient client = new WebClient())
{
string str = client.DownloadString(builder.Uri);
if (string.IsNullOrEmpty(str))
return null;
return HttpUtility.ParseQueryString(str)["access_token"];
}
}
}
您也可以像这样在 OAuthWebSecurity 中注册它(将 RegisterAuth 方法放在 Application_Start 中,就像在 InternetApplication 模板中一样):
public static class AuthConfig
{
public static void RegisterAuth()
{
configuration.LoadFromAppSettings();
OAuthWebSecurity.RegisterClient(new FacebookExtendedClient(
"##YOUR_APP_ID##",
"##YOUR_APP_SECRET##",
"id,first_name,last_name,link,username,gender,email,age_range,picture.height(200)",
new Func<string, object, string>(fieldsTransformer),
"email"));
}
private static string fieldsTransformer(string key, object value)
{
switch (key)
{
case "picture":
var data = (value as IDictionary<string, object>)["data"] as IDictionary<string, object>;
return data["url"].ToString();
case "age_range":
var min = (value as IDictionary<string, object>)["min"];
return min.ToString();
default:
return value.ToString();
}
}
}
正如您在上面的示例中看到的,fieldsTransformer 方法将获取所选每个字段的键和值,在本例中,它将把 facebook 检索到的图片对象转换为图片的 url。这是一个方便的方法,如果为 Func 参数提供 null,则将保存值对象的 JSON 表示形式。
稍后登录后检索客户端信息,可以这样做:
[Authorize]
public class HomeController : Controller
{
public ActionResult Index()
{
IDictionary<string, string> userData = (OAuthWebSecurity.GetOAuthClientData("facebook").AuthenticationClient as FacebookExtendedClient).UserData;
string email = userData["email"];
// If leave null the fieldTransform of the client you can access to complex properties like this:
JObject picture = JObject.Parse(userData["picture"]);
string url = (picture["data"] as JObject)["url"].ToString();
ViewBag.Email = userData["email"];
ViewBag.PictureUrl = url;
return View();
}
}
希望您喜欢这段代码,即使有点晚了! :)
关于asp.net-mvc - 具有定义范围的 .NET ASP.NET MVC 应用程序 Facebook OAuth,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14987868/
我试图对 ASP.Net MVC 有一个高层次的理解,我开始意识到它看起来很像原始的 ASP 脚本。过去,我们将“模型”/业务逻辑代码组织到 VBScript 类或 VB COM 组件中。 当然,现在
我已经搜索了一段时间,但似乎找不到答案。 我想在我的旋转木马中显示一个计数器,左边是当前项目(工作),左边是项目总数。 我的代码:
. 最佳答案 Scott Gu 称这些为代码块。这就是我的看法。 http://weblogs.asp.net/scottgu/archive/2010/04/06/new-lt-gt-syntax
我有一个使用 Visual Studio 2010/.net 4/VB 制作的网站。 我真的很喜欢我发现的 FAQ 系统的布局,因为它很简单,但它是经典的 asp。所以,显然,我不能包括我的母版页布局
好吧,对于你们许多人来说,这个问题可能有一个非常明显的答案,但它让我难住了。 我有一个 asp.net Web 表单,上面有两个控件(嗯,不止这两个,但我们将重点关注这些) - 第一个是 asp:dr
当我将 ASP.NET 复选框控件设置为 asp.net 更新面板的异步回发触发器时,EventName 属性是什么? 最佳答案 我相信它是 CheckedChanged。 关于asp.net - a
我有一个用经典 asp 编写的(巨大的)网站。现在我必须切换到 vb.net (razor)。有没有办法将这两个结合起来直到切换完成? 有没有办法让应用程序与经典的 asp 和 vb.net 一起工作
I am creating a products page, where the user selects an option in a radiobuttonlist for example, an
我最近将一个经典的 ASP 应用程序转换为 ASP.NET 3.5,但我觉得我的经典 ASP 版本要快一些(我不知道可能买家会后悔)。 所以你们能帮我解决这个问题吗,让我知道哪个更快,asp、asp.
从本周开始,我被要求开始学习如何使用 ASP 开发网站。我通过 XNA 对 C# 有一定的经验,所以这部分对我来说并不是什么麻烦。 我一直在关注Music Store Tutorial这需要我设置一个
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 8 年前。 Improve this ques
我想将一些表单变量发布到经典 ASP 页面中。我不想改变经典的 ASP 页面,因为需要完成大量的工作,以及消耗它们的页面数量。 经典的 ASP 页面需要将表单变量 Username 和 Userpas
已结束。此问题正在寻求书籍、工具、软件库等的推荐。它不满足Stack Overflow guidelines 。目前不接受答案。 我们不允许提出寻求书籍、工具、软件库等推荐的问题。您可以编辑问题,以便
在某种程度上,这可能是一个异端问题。我们有一个大型站点,其中许多页面仍在ASP中。通常,并没有真正动态的,而是包括(通过SSI或Server.Execute)定期重新生成的HTML块。看起来好像是一个
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 9 年前。 Improve this ques
我有一个遗留的 ASP 应用程序——在不久的某个时候——需要迁移到 ASP.Net 2.0(以与也在 2.0 中的其他应用程序兼容)。 对于这类事情是否有最佳实践,即作为第一步将当前 html、vbs
我目前在一家公司工作,该公司使用 ASP.NET Webforms 和旧 ASP 页面的组合进行 Web 开发。这对于他们当前的项目来说效果很好,但我想说服/建议他们切换到 ASP.NET MVC,因
我有一个经典的 asp 应用程序。我想将该页面的竞赛表格发布到 Asp.Net 表格。原因是我想在进入数据库之前使用我在 Asp.Net 页面中内置的大量逻辑进行验证,而我对 asp 不太了解。更不用
我知道在 ASP.NET MVC 中,您可以拥有移动 View 并执行类似 Index.mobile.cshtml 的操作。和 _Layout.mobile.cshtml并且服务器知道将这些 View
我需要从一些服务器端 c#.net 代码中调用经典 asp 页面上的 VBscript 函数 - 有谁知道一种干净的方法来做到这一点?在 .net 中重写函数不是一种选择。 我会再解释一下这个问题..
我是一名优秀的程序员,十分优秀!