使用 paypal/rest-api-sdk-php 的 laravel paypal 集成错误

Posted

技术标签:

【中文标题】使用 paypal/rest-api-sdk-php 的 laravel paypal 集成错误【英文标题】:laravel paypal intergration error using paypal/rest-api-sdk-php 【发布时间】:2021-07-05 06:20:56 【问题描述】:

在我单击 myweb 应用程序中的付款按钮时出现的错误下方。我查看过类似问题的解决方案可能是什么,但似乎没有解决我的解决方案。 访问https://api.sandbox.paypal.com/v1/payments/payment 时得到 Http 响应代码 400。 PayPal\Core\PayPalHttpConnection:错误:访问https://api.sandbox.paypal.com/v1/payments/payment 时得到 Http 响应代码 400。 "name":"MALFORMED_REQUEST","message":"传入的 JSON 请求未映射到 API 请求"​​,"information_link":"https://developer.paypal.com/webapps/developer/docs/api/#MALFORMED_REQUEST ","debug_id":"7abde5aa538aa"

下面是我的控制器 PaypalController.php

<?php

namespace App\Http\Controllers;

use Gloudemans\Shoppingcart\Facades\Cart;
use Illuminate\Support\Facades\Input;
use Illuminate\Http\Request;
use PayPal\Api\Details;
use PayPal\Api\Amount;
use PayPal\Api\Item;
use PayPal\Api\WebProfile;
use PayPal\Api\ItemList;
use PayPal\Api\InputFields;
use PayPal\Api\Payer;
use PayPal\Api\Payment;
use PayPal\Api\PaymentExecution;
use PayPal\Api\RedirectUrls;
use PayPal\Api\Transaction;
use PayPal\Auth\OAuthTokenCredential;
use PayPal\Rest\ApiContext;
use Redirect;
use Session;
use URL;
use App\Invoice;
use App\Project;
use App\Suborder;
use Str;
//Use Request;
use Auth;
use Config;
use DB;

class PaypalController extends Controller

    private $apiContext;

    public function __construct()
    
        # Main configuration in constructor
        $paypalConfig = Config::get('paypal');

        $this->apiContext = new ApiContext(new OAuthTokenCredential(
                $paypalConfig['client_id'],
                $paypalConfig['secret'])
        );

        $this->apiContext->setConfig($paypalConfig['settings']);
    

    public function pay_amount(Request $request, $id)
    
       $id =DB::table('projects')->where('id', $id)->get();
        return view('projects.pay')
        ->with('id', $id);
    

    public function payWithpaypal(Request $request, $id)
    
        # We initialize the payer object and set the payment method to PayPal
        $payer = new Payer();
        $payer->setPaymentMethod('paypal');

        # We insert a new order in the order table with the 'initialised' status
        // $project = new Project();
        // $project->user_name = Auth::user()->name;
        // $project->invoice_id = null;
        // $project->status = 'initialised';
        // $project->save();

        $project = Project::findOrFail($request->id);
        //$project = new Project();
        $project->update(['user_name' => (Auth::user()->name)]);
        $project->update(['invoice_id' => (null)]);
        $project->update(['status' => ('initialised')]);
        $project->save();

        # We need to update the order if the payment is complete, so we save it to the session
        Session::put('projectId', $project->getKey());

        # We get all the items from the cart and parse the array into the Item object

        //dd($project->project_id);

        $item = new Item();
        $items[] = (new Item())
            ->setCurrency('USD')
            ->setQuantity(1)
            ->setPrice($project->amount);
            //->setName($project->project_id)
            //->setName($item->name)
            //->setQuantity($item->qty)
            //->setPrice($item->price);
            //->setPrice($project->total_amount);

            //->setPrice($request->get('amount')); 

        # We create a new item list and assign the items to it

        $itemList = new ItemList();
        $itemList->setItems(array($items));

        # Disable all irrelevant PayPal aspects in payment
        // $inputFields = new InputFields();
        // $inputFields->setAllowNote(true)
        //     ->setNoShipping(1)
        //     ->setAddressOverride(0);

        // $webProfile = new WebProfile();
        // $webProfile->setName(uniqid())
        //     ->setInputFields($inputFields)
        //     ->setTemporary(true);

        // $createProfile = $webProfile->create($this->apiContext);

        # We get the total price of the cart

        $amount = new Amount();
        $amount->setCurrency('USD')
                ->setTotal($project->amount);
            //->setTotal(Cart::subtotal());


        $transaction = new Transaction();
        $transaction->setAmount($amount);
        $transaction->setItemList($itemList)
            ->setDescription('Your transaction description');    

        $redirectURLs = new RedirectUrls();
        $redirectURLs->setReturnUrl(URL::to('status'))
            ->setCancelUrl(URL::to('status'));

        $payment = new Payment();
        $payment->setIntent('Sale')
        //$payment->setIntent('authorize')
            ->setPayer($payer)
            ->setRedirectUrls($redirectURLs)
            ->setTransactions(array($transaction));

        //$payment->setExperienceProfileId($createProfile->getId());
        //$payment->create($this->apiContext);


        try 

            $payment->create($this->apiContext);
            $execution = new PaymentExecution(); $result = $payment->execute($execution, $apiContext);


         catch (\PayPal\Exception\PPConnectionException $ex) 

            if (\Config::get('app.debug')) 

                \Session::put('error', 'Connection timeout');
                return Redirect::to('/');

             else 

                \Session::put('error', 'Some error occur, sorry for inconvenient');
                return Redirect::to('/');

            

        

        foreach ($payment->getLinks() as $link) 

            if ($link->getRel() == 'approval_url') 

                $redirect_url = $link->getHref();
                break;

            

        

        # We store the payment ID into the session
        Session::put('paypalPaymentId', $payment->getId());

        if (isset($redirectURL)) 
            return Redirect::away($redirectURL);
        

        Session::put('error', 'There was a problem processing your payment. Please contact support.');

        return Redirect::to('/client/projects');
    


    public function getPaymentStatus()
    
        /** Get the payment ID before session clear **/
        $payment_id = Session::get('paypal_payment_id');

        /** clear the session payment ID **/
        Session::forget('paypal_payment_id');
        if (empty(Input::get('PayerID')) || empty(Input::get('token'))) 

            \Session::put('error', 'Payment failed');
            return Redirect::to('/');

        

        $payment = Payment::get($payment_id, $this->_api_context);
        $execution = new PaymentExecution();
        $execution->setPayerId(Input::get('PayerID'));

        /**Execute the payment **/
        $result = $payment->execute($execution, $this->_api_context);

        if ($result->getState() == 'approved') 

            \Session::put('success', 'Payment success');
            return Redirect::to('/');

        

        \Session::put('error', 'Payment failed');
        return Redirect::to('/');

    

 
```

【问题讨论】:

【参考方案1】:

PayPal-PHP-SDK 和 v1/payments API 已弃用。

改为使用 Checkout-PHP-SDK 作为 v2/checkout/orders API,此处记录:https://developer.paypal.com/docs/business/checkout/server-side-api-calls/#server-side-api-calls

您应该创建两条路线,一条用于“创建订单”,一条用于“捕获订单”。这些路由应该只返回/输出 JSON 数据(没有 html 或文本)

将这两条路线与以下批准流程配对:https://developer.paypal.com/demo/checkout/#/pattern/server

【讨论】:

以上是关于使用 paypal/rest-api-sdk-php 的 laravel paypal 集成错误的主要内容,如果未能解决你的问题,请参考以下文章

在使用加载数据流步骤的猪中,使用(使用 PigStorage)和不使用它有啥区别?

今目标使用教程 今目标任务使用篇

Qt静态编译时使用OpenSSL有三种方式(不使用,动态使用,静态使用,默认是动态使用)

MySQL db 在按日期排序时使用“使用位置;使用临时;使用文件排序”

使用“使用严格”作为“使用强”的备份

Kettle java脚本组件的使用说明(简单使用升级使用)