我想讓用戶通過SoundCloud爲我的ASP.NET MVC 4項目進行身份驗證。由於沒有.NET SDK,我寫了一個自定義OAuth2Client
來處理認證。將客戶端添加到我的AuthConfig.cs
後,它適當地顯示爲登錄選項。問題是,當我點擊按鈕登錄,它總是返回如何讓用戶使用SoundCloud登錄我的網站
Login Failure.
Unsuccessful login with service.
甚至沒有要求我登錄SoundCloud。問題是什麼?我爲GitHub實現了一個非常類似的客戶端,並且沒有任何問題。
這裏是我的客戶:
public class SoundCloudOAuth2Client : OAuth2Client
{
private const string ENDUSERAUTHLINK = "https://soundcloud.com/connect";
private const string TOKENLINK = "https://api.soundcloud.com/oauth2/token";
private readonly string _clientID;
private readonly string _clientSecret;
public SoundCloudOAuth2Client(string clientID, string clientSecret) : base("SoundCloud")
{
if (string.IsNullOrWhiteSpace(clientID)) {
throw new ArgumentNullException("clientID");
}
if (string.IsNullOrWhiteSpace(clientSecret)) {
throw new ArgumentNullException("clientSecret");
}
_clientID = clientID;
_clientSecret = clientSecret;
}
protected override Uri GetServiceLoginUrl(Uri returnUrl)
{
StringBuilder serviceUrl = new StringBuilder();
serviceUrl.Append(ENDUSERAUTHLINK);
serviceUrl.AppendFormat("?client_id={0}", _clientID);
serviceUrl.AppendFormat("&response_type={0}", "code");
serviceUrl.AppendFormat("&scope={0}", "non-expiring");
serviceUrl.AppendFormat("&redirect_uri={0}", System.Uri.EscapeDataString(returnUrl.ToString()));
return new Uri(serviceUrl.ToString());
}
public override void RequestAuthentication(HttpContextBase context, Uri returnUrl)
{
base.RequestAuthentication(context, returnUrl);
}
protected override IDictionary<string, string> GetUserData(string accessToken)
{
IDictionary<String, String> extraData = new Dictionary<String, String>();
var webRequest = (HttpWebRequest)WebRequest.Create("https://api.soundcloud.com/me.json?oauth_token=" + accessToken);
webRequest.Method = "GET";
string response = "";
using (HttpWebResponse webResponse = HttpWebResponse)webRequest.GetResponse())
{
using (StreamReader reader = new StreamReader(webResponse.GetResponseStream()))
{
response = reader.ReadToEnd();
}
}
var json = JObject.Parse(response);
string id = (string)json["id"];
string username = (string)json["username"];
string permalinkUrl = (string)json["permalink_url"];
extraData = new Dictionary<String, String>
{
{"SCAccessToken", accessToken},
{"username", username},
{"permalinkUrl", permalinkUrl},
{"id", id}
};
return extraData;
}
protected override string QueryAccessToken(Uri returnUrl, string authorizationCode)
{
StringBuilder postData = new StringBuilder();
postData.AppendFormat("client_id={0}", this._clientID);
postData.AppendFormat("&redirect_uri={0}", HttpUtility.UrlEncode(returnUrl.ToString()));
postData.AppendFormat("&client_secret={0}", this._clientSecret);
postData.AppendFormat("&grant_type={0}", "authorization_code");
postData.AppendFormat("&code={0}", authorizationCode);
string response = "";
string accessToken = "";
var webRequest = (HttpWebRequest)WebRequest.Create(TOKENLINK);
webRequest.Method = "POST";
webRequest.ContentType = "application/x-www-form-urlencoded";
using (Stream s = webRequest.GetRequestStream())
{
using (StreamWriter sw = new StreamWriter(s))
sw.Write(postData.ToString());
}
using (WebResponse webResponse = webRequest.GetResponse())
{
using (StreamReader reader = new StreamReader(webResponse.GetResponseStream()))
{
response = reader.ReadToEnd();
}
}
var json = JObject.Parse(response);
accessToken = (string)json["access_token"];
return accessToken;
}
public override AuthenticationResult VerifyAuthentication(HttpContextBase context, Uri returnPageUrl)
{
string code = context.Request.QueryString["code"];
string u = context.Request.Url.ToString();
if (string.IsNullOrEmpty(code))
{
return AuthenticationResult.Failed;
}
string accessToken = this.QueryAccessToken(returnPageUrl, code);
if (accessToken == null)
{
return AuthenticationResult.Failed;
}
IDictionary<string, string> userData = this.GetUserData(accessToken);
if (userData == null)
{
return AuthenticationResult.Failed;
}
string id = userData["id"];
string name;
if (!userData.TryGetValue("username", out name) && !userData.TryGetValue("name", out name))
{
name = id;
}
return new AuthenticationResult(
isSuccessful: true, provider: "SoundCloud", providerUserId: id, userName: name, extraData: userData);
}
}
和AuthConfig.cs
:
public static void RegisterAuth()
{
OAuthWebSecurity.RegisterClient(new SoundCloudOAuth2Client(
clientID: MyValues.MyClientID,
clientSecret: MyValues.MyClientSECRET),
displayName: "SoundCloud",
extraData: null);
OAuthWebSecurity.RegisterClient(new GitHubOAuth2Client(
appId: MyValues.GITHUBAPPID,
appSecret: MyValues.GITHUBAPPSECRET), "GitHub", null);
OAuthWebSecurity.RegisterGoogleClient();
OAuthWebSecurity.RegisterYahooClient();
}
任何想法如何獲得使用Owin的SoundCloud身份驗證? – gldraphael 2014-12-12 17:09:12
@gldraphael不,對不起。我最近使用Sound Cloud auth的所有功能都在node.js中。雖然我會認爲(希望)自從我寫這篇文章以後,會有一個關於nuget或者其他東西的軟件包。 – MikeSmithDev 2014-12-12 18:25:46