- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我已经成功地集成了 Facebook、Twitter 和 Google 的预定义提供程序。但我需要允许人们通过 Paypal 登录。我创建了以下继承自 DotNetOpenAuth.AspNet.Clients.OAuth2Client 的类。
using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Net;
using System.Web;
using Validation;
using DotNetOpenAuth.AspNet;
using DotNetOpenAuth.Messaging;
using Newtonsoft.Json;
using System.Text.RegularExpressions;
namespace OauthClient
{
public class PayPalProviderClient : DotNetOpenAuth.AspNet.Clients.OAuth2Client
{
#region Oauth2
#region Constants and Fields
// https://www.x.com/developers/paypal/documentation-tools/quick-start-guides/oauth-openid-connect-integration-paypal#making-first-call-register
// 3 https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/authorize?client_id=<your_app_id>&response_type=code&scope=openid&redirect_uri=https://www.example.com/my-redirect-uri
// 5 https://your-return-url?code=auth-code&scope=space-separted-scope-list&state=state
// 6 https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/tokenservice
private const string AuthorizationEndpoint = "https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/authorize?response_type=code&scope=openid&";
private const string TokenEndpoint = "https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/tokenservice";
private const string UserDetailsEndPoint = "https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/userinfo";
private readonly string appId;
private readonly string appSecret;
#endregion
#region Constructors and Destructors
public PayPalProviderClient(string appId, string appSecret) : base("paypal")
{
Requires.NotNullOrEmpty(appId, "appId");
Requires.NotNullOrEmpty(appSecret, "appSecret");
this.appId = appId;
this.appSecret = appSecret;
}
#endregion
protected override Uri GetServiceLoginUrl(Uri returnUrl)
{
var builder = new UriBuilder(AuthorizationEndpoint);
builder.AppendQueryArgument("client_id", this.appId);
builder.AppendQueryArgument("redirect_uri", returnUrl.AbsoluteUri);
return builder.Uri;
}
protected override IDictionary<string, string> GetUserData(string accessToken)
{
var builder = new UriBuilder(UserDetailsEndPoint);
builder.AppendQueryArgument("schema", "openid");
builder.AppendQueryArgument("access_token", accessToken);
Dictionary<string, string> userData = new Dictionary<string, string>();
using (WebClient client = new WebClient())
{
string data = client.DownloadString(builder.Uri);
if (string.IsNullOrEmpty(data))
return null;
else
userData = JsonConvert.DeserializeObject<Dictionary<string, string>>(data);
}
return userData;
}
public override void RequestAuthentication(HttpContextBase context, Uri returnUrl)
{
base.RequestAuthentication(context, returnUrl);
}
private IDictionary<string, string> GetUserData(string accessCode, Uri redirectURI)
{
string token = QueryAccessToken(redirectURI, accessCode);
if (token == null || token == "")
return null;
return GetUserData(token);
}
private string QueryAccessToken(string returnUrl, string authorizationCode)
{
return QueryAccessToken(new Uri(returnUrl), authorizationCode);
}
protected override string QueryAccessToken(Uri returnUrl, string authorizationCode)
{
var builder = new UriBuilder(TokenEndpoint);
//"client_id=<your_app_id>&client_secret=<your_app_secret>&grant_type=authorization_code&code=<code>&redirect_uri=<urlencoded_return_url>"
builder.AppendQueryArgument("client_id", this.appId);
builder.AppendQueryArgument("client_secret", this.appSecret);
builder.AppendQueryArgument("grant_type", "authorization_code");
builder.AppendQueryArgument("code", authorizationCode);
builder.AppendQueryArgument("redirect_uri", returnUrl.AbsoluteUri);
using (WebClient client = new WebClient())
{
string data = client.DownloadString(builder.Uri);
if (string.IsNullOrEmpty(data))
return null;
Dictionary<string, string> userData = JsonConvert.DeserializeObject<Dictionary<string, string>>(data);
if (userData.ContainsKey("access_token"))
return userData["access_token"].ToString();
else
return string.Empty;
}
}
private string RefreshAccessToken(Uri returnUrl, string authorizationCode)
{
var builder = new UriBuilder(TokenEndpoint);
builder.AppendQueryArgument("client_id", this.appId);
builder.AppendQueryArgument("client_secret", this.appSecret);
builder.AppendQueryArgument("grant_type", "refresh_token");
builder.AppendQueryArgument("code", authorizationCode);
builder.AppendQueryArgument("redirect_uri", returnUrl.AbsoluteUri);
using (WebClient client = new WebClient())
{
string data = client.DownloadString(builder.Uri);
if (string.IsNullOrEmpty(data))
return null;
Dictionary<string, string> userData = JsonConvert.DeserializeObject<Dictionary<string, string>>(data);
if (userData.ContainsKey("access_token"))
return userData["access_token"].ToString();
else
return string.Empty;
}
}
#endregion
}
}
这是由我的 Controller 调用的
string callbackUrl = Url.Action("ExternalLoginCallback", new { ReturnUrl = returnUrl });
AuthenticationResult result = OAuthWebSecurity.VerifyAuthentication(callbackUrl);
这会导致“System.Collections.Generic.KeyNotFoundException 未被用户代码处理”——字典中不存在给定的键。
有没有人有使用 DNOA 和 PayPal 的工作示例?或者知道为什么我会收到此异常/我缺少什么 key ?
更新
我已经设法获得了一个工作版本,请参见下面的代码:
using System;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.IO;
using System.Net;
using System.Web;
using DotNetOpenAuth.AspNet;
using DotNetOpenAuth.AspNet.Clients;
using DotNetOpenAuth.Messaging;
using Newtonsoft.Json;
namespace OauthClients
{
/// <summary>
/// A DotNetOpenAuth client for logging in to paypal using OAuth2.
/// Reference: https://developers.paypal.com/accounts/docs/OAuth2
/// </summary>
public class PaypalOAuth2Client : OAuth2Client
{
#region Constants and Fields
/// <summary>
/// The authorization endpoint.
/// </summary>
private const string AuthorizationEndpoint = "https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/authorize?response_type=code&scope=openid&";
/// <summary>
/// The token endpoint.
/// </summary>
private const string TokenEndpoint = "https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/tokenservice";
/// <summary>
/// The user info endpoint.
/// </summary>
private const string UserDetailsEndPoint = "https://www.paypal.com/webapps/auth/protocol/openidconnect/v1/userinfo";
/// <summary>
/// The base uri for scopes.
/// </summary>
private const string ScopeBaseUri = "https://www.paypal.com/webapps/auth/";
/// <summary>
/// The _app id.
/// </summary>
private readonly string _clientId;
/// <summary>
/// The _app secret.
/// </summary>
private readonly string _clientSecret;
/// <summary>
/// The requested scopes.
/// </summary>
private readonly string[] _requestedScopes;
#endregion
/// <summary>
/// Creates a new paypal OAuth2 client.
/// </summary>
/// <param name="clientId">The paypal Client Id</param>
/// <param name="clientSecret">The paypal Client Secret</param>
/// <param name="requestedScopes">One or more requested scopes, passed without the base URI.</param>
public PaypalOAuth2Client(string clientId, string clientSecret): base("paypal")
{
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)
{
var builder = new UriBuilder(AuthorizationEndpoint);
builder.AppendQueryArgument("client_id", this._clientId);
builder.AppendQueryArgument("redirect_uri", returnUrl.AbsoluteUri);
return builder.Uri;
}
protected override IDictionary<string, string> GetUserData(string accessToken)
{
var request = WebRequest.Create(UserDetailsEndPoint + "?schema=openid&access_token=" + accessToken);
Dictionary<string, string> authData;
using (var response = request.GetResponse())
using (var responseStream = response.GetResponseStream())
using (var streamReader = new StreamReader(responseStream))
{
authData = JsonConvert.DeserializeObject<Dictionary<string, string>>(streamReader.ReadToEnd());
return authData;
}
}
protected override string QueryAccessToken(Uri returnUrl, string authorizationCode)
{
var builder = new UriBuilder(TokenEndpoint);
builder.AppendQueryArgument("client_id", this._clientId);
builder.AppendQueryArgument("client_secret", this._clientSecret);
builder.AppendQueryArgument("grant_type", "authorization_code");
builder.AppendQueryArgument("code", authorizationCode);
builder.AppendQueryArgument("redirect_uri", returnUrl.AbsoluteUri);
using (WebClient client = new WebClient())
{
var data = client.DownloadString(builder.Uri);
if (string.IsNullOrEmpty(data))
return null;
Dictionary<string, string> userData = JsonConvert.DeserializeObject<Dictionary<string, string>>(data);
if (userData.ContainsKey("access_token"))
return userData["access_token"].ToString();
else
return null;
}
}
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["user_id"];
string name = string.Empty ;
return new AuthenticationResult(
isSuccessful: true, provider: "PayPal", providerUserId: id, userName: name, extraData: userData);
}
}
}
最佳答案
您必须在 GetUserData 方法中提供带有 key="id"的字典条目(名称为“name”和“username”的字典键即可)。
来自 OAuth2Client.class 的代码...string id = userData["id"]; 字符串名称;
// Some oAuth providers do not return value for the 'username' attribute.
// In that case, try the 'name' attribute. If it's still unavailable, fall back to 'id'
if (!userData.TryGetValue("username", out name) && !userData.TryGetValue("name", out name)) {
name = id;
}
...
如您所见,如果其中一个键丢失,代码将失败。
关于asp.net-mvc-4 - 自定义 DotnetOpenauth Paypal 客户端( key 不存在),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15489651/
如果我的交易中的次要收款人没有 Paypal 帐户,会发生什么情况。他会收到要求他创建 Paypal 帐户的通知,还是 API 调用不会成功? 我几乎无法知道销售服务的用户在注册我的平台时是否有Pay
我们知道有这么多的支付网关服务提供商,即 Paypal ,moneybrookers,authorize.net以及许多国内和国际的支付网关提供商。现在,我的问题是,开发我们自己的支付网关服务以处理世
我们正在寻求替代我们对 Authorize.net 的使用。我花了很多时间查看 PayPal 文档,但找不到我的问题的明确答案: 使用 thousand 数十种不同的 PayPal API 可以在我们
我实际上是在为客户工作。我已经在网站上成功实现了 paypal pro,并且一切正常,我相信我遵循了他们的流程和指南,使它按照他们的意思工作。 因此,我一直在阅读 this webpage将网站投入生
我有2件商品要出售。我的运输底价为7.50美元,每增加一个项目,则为2.00美元 我已经设置了高级代码 运费= $ 7.50 运费2 = $ 2.00 一切正常,除非在购物车中添加了其他商品后,购物车
我想在不使用购物车的情况下实现一个不太复杂的解决方案。 我在自己对这个问题的回答中详细说明了解决方案,但如果有人能提出更好的方法,我会很高兴。 谢谢 最佳答案 实际上我最近不得不这样做。毫无疑问,我的
我创建了一个带有默认支付网关的shopify 小商店,即paypal express checkout。 当客户点击 Paypal 结帐按钮时,他们被要求输入他们的信用卡号并输入电子邮件并选择密码以注
一段时间以来,我一直在运行一个网站,该网站通过向 Paypal 网站发送一个 notify_url 来记录 Paypal 交易,并监听该 notify_url。我的监听页面解析从 paypal 发送的
我在网络应用程序中使用 PayPal Subscribe 按钮。我想将订阅开始日期传递给 PayPal,这不会总是今天的日期。 通常情况下,如果没有有效订阅,PayPal 会创建一个从今天开始的订阅。
大家好,我遇到了“立即购买”paypal 按钮这个令人沮丧的问题。 我希望我的客户能够通过 paypal 使用借记卡/信用卡付款,而无需创建 paypal 帐户,同时我希望能够根据产品类型将不同的定价
我想实现 paypal 授权并自动获取付款。我用过 Paypal 标准账户。我已发送带有授权参数的付款请求。
我想在我的 Paypal 沙盒账户中测试批量支付。根据 paypal 文档,它描述了将资金转移到 paypal 帐户的步骤。但我想转账到我客户的银行账户。我该怎么做? 仅供引用:PayPal 文档内容
我正在尝试登录我的沙盒帐户(使用 chrome),但由于 paypal 的新变化我不能。 Paypal 现在要求我登录开发者网站。我愿意。 然后我转到应用程序->沙盒帐户,然后按“sanbox 站点”
我有一个支持 Paypal 付款的 opencart 系统。是否有允许用户(信用卡)付款而无需强制注册 Paypal 帐户的选项。到目前为止,我使用的是 paypal 标准模块。 最佳答案 使用 Pa
我可以使用 Paypal ID 和密码使用 Paypal 详细信息进行付款而无需重定向到 Paypal 网站吗?我想在不离开我的网站的情况下完成所有付款流程。谢谢 最佳答案 您可以使用 PayFlow
当我尝试 https://fpdbs.paypal.com/dynamicimageweb?cmd=_dynamicimage 时,图像损坏了. 这是返回的错误信息: “尝试为应用程序构建 prote
我想向我的客户付款,为此我收到了他们的电子邮件,我想验证电子邮件地址是否存在于 paypal 中! 我搜索并找到了我尝试过的支付身份 API,但它只提供了我的我的数据!这是 LINK那个 API。 我
当从用户的 PayPal 账户中扣款时,PayPal 服务器在一定时间内没有响应(超时)。我们决定假设钱确实已经被拿走了,所以我们会退款。但是,如果钱没有从用户账户中扣除会怎样? PayPal返回什么
我有一个网站可以定期(比如每周一次)从客户那里收到钱。我将我的客户详细信息存储在我的数据库中,我想从我的客户帐户中提取资金,而不是每次都提示他们进行身份验证。金额会因客户而异。我怎样才能用 pay p
我有一个表格,其中一个团队签下他们的球员,然后被运送到 Paypal 进行注册付款,然后当一切都完成后我会收到一个 IPN。我正在做一个类似的表格,发现自从我创建该表格以来,Paypal 发生了很大变
我是一名优秀的程序员,十分优秀!