我无法让PayPal从我的应用程序接受实时付款。我正在使用Azure服务器,使用基于Xamarin的Android客户端应用程序在C#中编码。我可以使用PayPal沙箱进行所有操作,但是当我切换到实时凭据时它会失败。这是我的付款流程:
1)在我的Android应用程序中,启动PayPal应用程序以授权付款。这将返回授权码。此代码类似于C21AAHHVxYjktRk82SomDGL6ssqd85LLmso5uxADtMW4ABEyieWdC_6mUxWK84hfeteotgeorgiengeoiugHUWRiu-Q
2)将代码传递给服务器;使用具有client_credentials的grant_type的代码来获取承载令牌。返回的令牌是这样的:
"scope": "https://uri.paypal.com/services/subscriptions https://api.paypal.com/v1/payments/.* https://api.paypal.com/v1/vault/credit-card https://uri.paypal.com/services/applications/webhooks https://uri.paypal.com/services/payments/futurepayments openid https://api.paypal.com/v1/vault/credit-card/.*",
"nonce": "2017-12-31T12:07:46ZPYBveV0qcOc2apceqHbTBrXXVsuGCryP05-xxxxxxxxxxx",
"access_token": "A23AAHweO7uu8vuXs45CVOafwqFHGj03riRShISJRnHn4Pr1bzwdDfaG5JkqCT5oZ_pFWGBK-wScOq2JeDERe2USxxxxxxxxxxxxx",
"token_type": "Bearer",
"expires_in": 28800,
"refresh_token": "R23AAFR7MtimwU1AeQ7V6pTw0ugxN4AzOsRr-tBfmChMBAK8IMrJO_WQ6WgLUxbkK8ssvHoPVFFdVVAorNZriMX8gUA1tb-iRzwj7aiQ3_KCv5SDnqjf5j3vxKeZG052Q7WpSbW2zxxxxxxxxxxxxx"
3)提交REST请求以及持票人令牌以授权付款。 REST请求将如下所示:
{
"intent": "sale",
"payer": {
"payment_method": "paypal"
},
"transactions": [{
"amount": {
"currency": "GBP",
"total": 1.0
},
"payee": {
"email": "xxxxxxxxxxxxxxxxxx"
},
"description": "Test item",
"payment_options": {
"allowed_payment_method": "INSTANT_FUNDING_SOURCE"
}
}
请注意,我们必须在我们的请求中设置收款人,因为我们自己没有收到付款。
我得到的回复是这样的:
"id":"PAY-258208691145776xxxxxxxxx",
"intent":"sale",
"state":"created",
"cart":"1JF69642C9xxxxxxx",
"transactions":[
{
"amount":{
"total":"1.00",
"currency":"GBP"
},
"payee":{
"merchant_id":"xxxxxxxxxxxxxx",
"email":"xxxxxxxxxxxxxxx"
},
"description":"Test item",
"related_resources":[]
}
],
"redirect_urls":{
"return_url":"xxxxxxxxxxxxxxxxx/return?paymentId=PAY-xxxxxxxxxxxxxxxxxxxxxxx",
"cancel_url":"xxxxxxxxxxxxxxxx/cancel"
},
"create_time":"2018-08-20T10:22:19Z",
"update_time":"2018-08-20T10:22:20Z",
"links":
[
{
"href":"https://api.paypal.com/v1/payments/payment/PAY-xxxxxxxxxxxxxxxxxxxxxxx",
"rel":"self","
"method":"GET"
},
{
"href":"https://api.paypal.com/v1/payments/payment/PAY-xxxxxxxxxxxxxxxxxxxxxxx/execute",
"rel":"execute",
"method":"POST"
},
{
"href":"https://www.paypal.com/cgi-bin/webscr?cmd=_express-checkout&token=EC-xxxxxxxxxxxxxxxxxxxx",
"rel":"approval_url","method":"REDIRECT"
}
]
到目前为止都很好。
4)执行付款并将响应返回给客户端应用程序。这是执行付款的请求,我使用步骤(1)中的access_token和https://api.paypal.com/v1/payments/payment/PAY-xxxxxxxxxxxxxxxxxxxxxxx/execute的付款URL:
using (HttpClient client = new HttpClient())
{
try
{
client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("*/*"));
client.DefaultRequestHeaders.AcceptLanguage.Add(new StringWithQualityHeaderValue("en-GB"));
client.DefaultRequestHeaders.Add("Authorization", "Bearer " + token);
var responseMessage = await client.GetAsync(paymentURL);
ret1 = await responseMessage.Content.ReadAsStringAsync();
}
catch (Exception ex)
{
return ex.ToString();
}
}
这一切都适用于PayPal的沙箱。当我尝试运行它时,它在最后一步失败,出现MALFORMED_REQUEST错误。
所以我使用cURL(即完全手动,复制和粘贴命令)运行测试事务,使用相同的付款细节。付款流程略有不同,但有效。但是,我从PayPal返回了一个client_id,这在我从步骤(3)得到的响应中没有,所以我无法使用cURL运行我的实际事务,但我怀疑这是因为cURL方法是一个Web支付流程(即我最终在PayPal的网站上授权付款)而我的系统不使用网络,只使用PayPal的移动应用程序。顺便提一下,该应用程序返回的长代码在cURL请求中不起作用。
这里有些东西似乎非常错误,但它在沙箱中有效,即使我在步骤1中使用grant_type的grant_code(但是直播,grant_type在步骤2中失败)。我正在使用正确的PayPal现场ID和密码。有没有人对我在这里做错了什么有任何建议?我已经尝试过PayPal支持,但他们只建议使用cURL,如果没有client_id,我就无法进行交易。
非常感谢。
我终于得到了PayPal的答案,显然他们必须批准你的帐户才能启用未来的付款,这是我最初用来获得授权类型的authorization_code然后允许我在REST响应中设置收款人。我在上面列出的步骤中所做的只是按照PayPal技术支持的说明进行操作,而这些支持无论如何都不起作用。希望这可以解决问题,但如果PayPal将这些信息放在开发者网站的某个地方(也许它被埋在某处,但我还没有找到它),那将会非常有用。