验证 POST 数据来自 PayPal

Posted

技术标签:

【中文标题】验证 POST 数据来自 PayPal【英文标题】:Verify POST data comes from PayPal 【发布时间】:2014-06-28 19:58:25 【问题描述】:

如何 ?

我正在一家销售一些产品的商店工作。支付网关是 PayPal。 最初,我设置了一个自定义 PayPal 表单并使用 IPN 响应来验证从 PayPal 发送给我的数据。

现在我的客户购买了使用 PayPal PayFlow 的 PayPal 预付款。响应不再通过 IPN 发送(或者它们是?),而是由 SILENT POST 返回,基本上,当事务在其端执行时,它被发送到我选择的链接,我通过该链接处理数据。

如何验证 POST 数据的来源,以便我知道它来自 PayPal 而不是恶意用户。我找不到任何关于此的文档。当用户单击贝宝页面上的“取消”按钮并将其重定向到我网站上的取消页面时,我也想要同样的想法。我希望该 POST 数据源也得到验证。

对此有什么想法吗?

【问题讨论】:

你是说IPN不再起作用了吗?我仍然在我的项目中使用它... paypal 不提供示例脚本吗?我找到了开发者指南:developer.paypal.com/docs/classic/payflow/integration-guide @esqew PayPal IPN 有效,但不适用于 PayPal 高级付款。我从PayPal PayFlowlink的集成布局C以及通过借记卡/信用卡支付时,它也通过IPN发送验证,而且选择使用PayPal结帐时,返回的数据仅通过静音帖子发送。 span> @user4035 我已经集成了它,它可以工作,我也可以根据 SILENT POST 验证我的数据库,但它没有指定如何检查 POST 数据是否来自 PayPal。跨度> IPN 本身就是一个无声的 POST,所以我对你在这里所说的话感到困惑。据我所知,Payments Advanced 应该触发 IPN,就像任何其他 PayPal 付款一样。几乎听起来您之前将 PDT 和 IPN 组合在一起,而 PDT 不适用于 Advanced。这听起来像是发生了什么吗? 【参考方案1】:

比我找到的第一个解决方案,还有一些 PayPal 支持人员提到了类似的东西,但他不能提供细节,因为他说他不是专家。

基本上,您必须使用从 SILENT POST 收到的 PNREF 运行 INQUIRY 类型的 TRANSACTION,如果响应返回 ORIGRESULT 等于 0,则该交易存在于您帐户下的 PayPal 数据库中。

我还发送了带有查询的 SECURE TOKEN ID 和 SECURE TOKEN,我不知道它是否有帮助,我认为这是一个解决方案,我尝试只发送不带 ORIGID 的 TOKEN 和 TOKEN ID,有时它有时会起作用不是。

在来自 PayPal 的开发人员参考中指定了在 TRXTYPE=I(查询交易)上必须指定 ORIGID。

代码如下:

//GET POST DATA FROM SILENT POST
$data = $_POST;
$result = $data['RESULT'];
$pnref = $data['PNREF'];
$secure_token = $data['SECURETOKEN'];
$secure_token_id = $data['SECURETOKENID'];
$fee_amount = $data['AMT'];

if(!isset($result) || $result != '0')
  //DO TRANSACTION FAILED OPERATIONS
  exit;


//CHECK IF PNREF ID EXISTS ON PAYPAL
$post_data = "PARTNER=yourpartnerid"
            ."&VENDOR=your_vendor"
            ."&USER=your_user"
            ."&PWD=your_password"
            ."&SECURETOKENID=" . $secure_token_id
            ."&SECURETOKEN=" . $secure_token
            ."&ORIGID=" . $pnref
            ."&TRXTYPE=I"
            ."&VERBOSITY=HIGH";


$ch = curl_init('https://payflowpro.paypal.com');
curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
curl_setopt($ch, CURLOPT_POST, TRUE);
curl_setopt($ch, CURLOPT_POSTFIELDS, $post_data);       
$resp = curl_exec($ch);

$inquiry = array();
parse_str($resp, $inquiry);

//IF ORIGRESULT is 0 then PNREF/transaction exists.
if($inquiry['ORIGRESULT'] == '0' && $inquiry['RESPMSG'] == 'Approved') $validated = true; 
else $validated = false; 

if($result != 0 || $amount != 'your_correct_fee' || $validated == false)
  // DO TRANSACTION NOT VALID OR HAS FAILED OPERATIONS
  exit;


//DO TRANSACTION SUCCESSFULL OPERATIONS

INQUIRY 的响应如下所示:

RESULT=0&PNREF=ETHPC0BBF5FB&TRANSSTATE=8&ORIGRESULT=0&ORIGPNREF=ELFPB0E766F5&RESPMSG=Approved&AVSADDR=N&AVSZIP=N&ORIGPPREF=8GT035513B296200N&CORRELATIONID=97306f6456378&SETTLE_DATE=2014-07-09 14:11:36&TRANSTIME=2014-07-09 14:11:36&FIRSTNAME=John&LASTNAME=doe&AMT=0.0

另一种方法是检查 SILENT POST 来自的 IP。 我注意到所有 SILENT POST 数据都来自 173.0.81.65

$ip_address = $_SERVER['REMOTE_ADDR'];
if($ip_address != '173.0.81.65') exit; 

@安德鲁天使 SILENT POST 和 NotifyURL 不是一回事。 在 NotifyURL 中,您可以使用 IPN 和数据验证,因为您收到不同类型的回帖数据字符串,所以在此处完成的方式非常不同。

我与 PayPal 支持人员进行了交谈,他们说 NotifyURL 和 IPN 不适用于 PayPal Advanced Payments,仅适用于 PayPal Payments Pro。

确实,它们都执行相同的功能,但它们是不同的东西,源代码验证的方式也不同。

希望这对某人有所帮助,等待意见。

【讨论】:

以上是关于验证 POST 数据来自 PayPal的主要内容,如果未能解决你的问题,请参考以下文章

IPN 验证返回 INVALID,因为 PayPal 在 POST 中向 IPN 侦听器发送了不正确的字符集

带有 paypal api 的客户端无效,使用 HTTPoison.post 的客户端身份验证失败!/3

PayPal 支付标准从移动设备返回 GET 而不是 POST,因此无法验证记录支付

Paypal ipn 未验证:无效

Paypal 仅在订阅完成后返回验证码?

Paypal 移动图书馆 IPN 验证