搜尋
首頁後端開發php教程Paypal實現循環扣款(訂閱)功能的範例程式碼分享

本文主要介紹了Paypal實現循環扣款(訂閱)的想法與方法;並對如何使用Paypal的支付介面做下總結,具有很好的參考價值。下面跟著小編一起來看吧

起因

業務需求要整合Paypal,實現循環扣款功能,然而百度和GOOGLE了一圈,除官網外面,沒找到相關開發教程,只好在Paypal上看,花了兩天後整合成功,這裡對如何使用Paypal的支付介面做下總結。

Paypal現在有多套介面:

  • 透過Braintree(後面會談Braintree)實作Express Checkout;

  • #建立App,透過REST Api的介面方式(現在的主流介面方式);

  • NVP/SOAP API apps的介面(舊介面);

Braintree的介面

Braintree是Paypal收購的公司,它除了支援Paypal的支付外,還提供了升級計劃,信用卡,客戶資訊等一系列全套的管理,使用上更方便;這些功能Paypal第二套REST介面其實也整合了大部分,但是Paypal的Dashboard不能直接管理這些資訊而Braintree可以,所以我其實我更願意用Braintree。關鍵是我使用的後端框架是Laravel,它的cashier解決方案預設可以支援Braintee,所以這套介面是我的首選。但是當我把它的功能都實現後發現一個蛋痛的問題:Braintree在國內不支持。 。 。 。 。 。卒。 。 。

REST API

這是順應時代發展的產物,如果你之前有使用OAuth 2.0與REST API,那麼看這些介面應該不會有什麼困惑。

舊介面

除非REST API介面有不能滿足的,例如政策限制,否則不建議使用。全世界都在往OAuth 2.0的認證方式和REST API的API使用方式遷移,幹嘛逆勢而行。因此在REST API能解決問題情況下,我也沒對這套介面做深入比較。

REST API的介紹

如果自己直接調這些API還是很繁瑣的,同時我們只想盡快完成業務要求而不是陷入對API的深入了解。

那麼如何開始呢,建議直接安裝官方提供的PayPal-PHP-SDK,透過其Wiki作為起點。

在完成首個範例之前,請確保你有Sandbox帳號,並且正確配置了:

  • Client ID

  • Client Secret

  • Webhook API(必須是https開頭且是443端口,本地調試建議結合ngrok反向代理生成地址)

  • #Returnurl(注意項目同上)

  • #Payments 一次性支付接口,不支援循環捐款。主要支付內容有支援Paypal支付,信用卡支付,透過已保存的信用卡支援(需要使用Vault接口,會有這樣的接口主要是PCI的要求,不允許一般的網站採集信用卡的敏感資訊),支援付給第三方收款人。

  • Payouts 沒用到,忽略;

  • Authorization and Capture 支援直接透過Paypal的帳號登陸你的網站,並取得相關資訊;

  • Sale 跟商城有關,沒用到,忽略;

  • Order 跟商城有關,沒用到,忽略;

  • Billing Plan & Agreements 升級計畫與簽約,也就是訂閱功能,實現循環扣款必須使用這裡的功能,這是本文的重點;

  • Vault 儲存信用卡資訊

  • Payment Experience 沒用到,忽略;

  • Notifications 處理Webhook的訊息,重要,但不是本文關注內容;

  • Invoice 票據處理;

  • Identity 認證處理,實現OAuth 2.0的登陸,取得對應token以便請求其他API,這塊Paypal-PHP-SDK已經做進去,本文也不談。

    如何實現循環扣款

    #分成四個步驟:

    1. 建立昇級計劃,並啟動;

    2. 建立訂閱(創建Agreement),然後將跳到Paypal的網站等待使用者同意;

    3. ##使用者同意後,執行訂閱

    4. 取得扣款帳單

    1.建立昇級計畫

    升級計畫對應Plan這個類別。這一步有幾個注意點:

    • 升級計畫建立後,處於CREATED狀態,必須將狀態修改為ACTIVE才能正常使用。

    • Plan有PaymentDefinition和MerchantPreferences兩個對象,這兩個物件都不能為空;

    • 如果想建立TRIAL類型的計劃,該計劃還必須有配套的REGULAR的付款定義,否則會報錯;

    • #看程式碼有呼叫一個setSetupFee(非常,非常,非常重要)方法,該方法設定了完成訂閱後首次扣款的費用,而Agreement物件的循環扣款方法設定的是第2次開始時的費用。

    以建立一個Standard的計畫為例,其參數如下:

    $param = [
     "name" => "standard_monthly",
     "display_name" => "Standard Plan",
     "desc" => "standard Plan for one month",
     "type" => "REGULAR",
     "frequency" => "MONTH",
     "frequency_interval" => 1,
     "cycles" => 0,
     "amount" => 20,
     "currency" => "USD"
     ];

    建立並啟動計畫程式碼如下:

     //上面的$param例子是个数组,我的实际应用传入的实际是个对象,用户理解下就好。
     public function createPlan($param)
     {
     $apiContext = $this->getApiContext();
     $plan = new Plan();
     // # Basic Information
     // Fill up the basic information that is required for the plan
     $plan->setName($param->name)
     ->setDescription($param->desc)
     ->setType('INFINITE');//例子总是设置为无限循环
     // # Payment definitions for this billing plan.
     $paymentDefinition = new PaymentDefinition();
     // The possible values for such setters are mentioned in the setter method documentation.
     // Just open the class file. e.g. lib/PayPal/Api/PaymentDefinition.php and look for setFrequency method.
     // You should be able to see the acceptable values in the comments.
     $paymentDefinition->setName($param->name)
     ->setType($param->type)
     ->setFrequency($param->frequency)
     ->setFrequencyInterval((string)$param->frequency_interval)
     ->setCycles((string)$param->cycles)
     ->setAmount(new Currency(array('value' => $param->amount, 'currency' => $param->currency)));
     // Charge Models
     $chargeModel = new ChargeModel();
     $chargeModel->setType('TAX')
     ->setAmount(new Currency(array('value' => 0, 'currency' => $param->currency)));
     $returnUrl = config('payment.returnurl');
     $merchantPreferences = new MerchantPreferences();
     $merchantPreferences->setReturnUrl("$returnUrl?success=true")
     ->setCancelUrl("$returnUrl?success=false")
     ->setAutoBillAmount("yes")
     ->setInitialFailAmountAction("CONTINUE")
     ->setMaxFailAttempts("0")
     ->setSetupFee(new Currency(array('value' => $param->amount, 'currency' => 'USD')));
     $plan->setPaymentDefinitions(array($paymentDefinition));
     $plan->setMerchantPreferences($merchantPreferences);
     // For Sample Purposes Only.
     $request = clone $plan;
     // ### Create Plan
     try {
     $output = $plan->create($apiContext);
     } catch (Exception $ex) {
     return false;
     }
     $patch = new Patch();
     $value = new PayPalModel('{"state":"ACTIVE"}');
     $patch->setOp('replace')
     ->setPath('/')
     ->setValue($value);
     $patchRequest = new PatchRequest();
     $patchRequest->addPatch($patch);
     $output->update($patchRequest, $apiContext);
     return $output;
     }

    2.創建訂閱(創建Agreement),然後將跳到Paypal的網站等待用戶同意

    #Plan創建後,要怎麼讓用戶訂閱呢,其實就是創建Agreement,關於Agreement,有以下注意點:

    • 如前面所述,Plan物件的setSetupFee方法,設定了完成訂閱後首次扣款的費用,而Agreement物件的循環扣款方法設定的是第2次開始時的費用。

    • setStartDate方法設定的是第2次扣款時的時間,因此如果你按月循環,應該是當前時間加一個月,同時該方法要求時間格式是ISO8601格式,使用Carbon庫可輕鬆解決;

    • 在創建Agreement的時候,此時還沒有產生唯一ID,於是我碰到了一點小困難:那就是當用戶完成訂閱的時候,我怎麼知道這個訂閱是哪個用戶的?透過Agreement的getApprovalLink方法得到的URL,裡面的token是唯一的,我透過提取該token作為識別方式,在用戶完成訂閱後替換成真正的ID。

    範例參數如下:

    $param = [
     'id' => 'P-26T36113JT475352643KGIHY',//上一步创建Plan时生成的ID
     'name' => 'Standard', 
     'desc' => 'Standard Plan for one month'
    ];

    程式碼如下:

     public function createPayment($param)
     {
     $apiContext = $this->getApiContext();
     $agreement = new Agreement();
     $agreement->setName($param['name'])
     ->setDescription($param['desc'])
     ->setStartDate(Carbon::now()->addMonths(1)->toIso8601String());
     // Add Plan ID
     // Please note that the plan Id should be only set in this case.
     $plan = new Plan();
     $plan->setId($param['id']);
     $agreement->setPlan($plan);
     // Add Payer
     $payer = new Payer();
     $payer->setPaymentMethod('paypal');
     $agreement->setPayer($payer);
     // For Sample Purposes Only.
     $request = clone $agreement;
     // ### Create Agreement
     try {
     // Please note that as the agreement has not yet activated, we wont be receiving the ID just yet.
     $agreement = $agreement->create($apiContext);
     // ### Get redirect url
     // The API response provides the url that you must redirect
     // the buyer to. Retrieve the url from the $agreement->getApprovalLink()
     // method
     $approvalUrl = $agreement->getApprovalLink();
     } catch (Exception $ex) {
     return "create payment failed, please retry or contact the merchant.";
     }
     return $approvalUrl;//跳转到$approvalUrl,等待用户同意
     }

    函數執行後傳回$approvalUrl,記得透過redirect( $approvalUrl)跳到Paypal的網站等待用戶付款。

    用戶同意後,執行訂閱

    用戶同意後,訂閱還未完成,必須執行Agreement的execute方法才算完成真正的訂閱。這一步驟的注意點在於

    • 完成訂閱後,並不等於扣款,可能會延遲幾分鐘;

    • ##如果第一步的setSetupFee費用設定為0,則必須等到循環扣款的時間到了才會產生訂單;

    程式碼片段如下:

     public function onPay($request)
     {
     $apiContext = $this->getApiContext();
     if ($request->has('success') && $request->success == 'true') {
     $token = $request->token;
     $agreement = new \PayPal\Api\Agreement();
     try {
     $agreement->execute($token, $apiContext);
     } catch(\Exception $e) {
     return ull;
     return $agreement;
     }
     return null;
     }

    • #在取得交易記錄

    • 訂閱後,可能不會立刻產生交易扣費的交易記錄,如果為空則過幾分鐘再嘗試。本步驟注意點:

    • start_date與end_date不能為空

    • #實際測試時,該函數傳回的物件不能總是傳回空的JSON對象,因此如果有需要輸出JSON,請依照AgreementTransactions的API說明,手動取出對應參數。

       /** 获取交易记录
       * @param $id subscription payment_id
       * @warning 总是获取该subscription的所有记录
       */
       public function transactions($id)
       {
       $apiContext = $this->getApiContext();
       $params = ['start_date' => date('Y-m-d', strtotime('-15 years')), 'end_date' => date('Y-m-d', strtotime('+5 days'))];
       try {
       $result = Agreement::searchTransactions($id, $params, $apiContext);
       } catch(\Exception $e) {
       Log::error("get transactions failed" . $e->getMessage());
       return null;
       }
       return $result->getAgreementTransactionList() ;
       }
    • 最後,Paypal官方當然也有對應的教程,不過是呼叫原生介面的,跟我上面流程不一樣點在於只說了前3步
    ### ###需要考慮的問題#########功能是實現了,但是也發現不少注意點:############國內使用Sandbox測試時連接特別慢,經常提示逾時或出錯,因此需要特別考慮執行中途用戶關閉頁面的情況;#############一定要實現webhook,否則當用戶進Paypal取消訂閱時,你的網站將得不到通知;############訂閱(Agreement)一旦產生,除非主動取消,否則將一直生效。因此如果你的網站設計了多個升級方案(例如Basic,Standard,Advanced),當使用者已經訂閱某個方案後,去切換升級方案時,開發上必須取消前一個升級方案;###### ######用戶同意訂閱-(取消舊訂閱-完成新訂閱的簽約-###修改用戶資訊###為新的訂閱),括號整個過程應該是原子操作,同時耗時又長,因此應該將其放到###隊列###中執行直到成功體驗會更好。 ##########

    以上是Paypal實現循環扣款(訂閱)功能的範例程式碼分享的詳細內容。更多資訊請關注PHP中文網其他相關文章!

    陳述
    本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
    超越炒作:評估當今PHP的角色超越炒作:評估當今PHP的角色Apr 12, 2025 am 12:17 AM

    PHP在現代編程中仍然是一個強大且廣泛使用的工具,尤其在web開發領域。 1)PHP易用且與數據庫集成無縫,是許多開發者的首選。 2)它支持動態內容生成和麵向對象編程,適合快速創建和維護網站。 3)PHP的性能可以通過緩存和優化數據庫查詢來提升,其廣泛的社區和豐富生態系統使其在當今技術棧中仍具重要地位。

    PHP中的弱參考是什麼?什麼時候有用?PHP中的弱參考是什麼?什麼時候有用?Apr 12, 2025 am 12:13 AM

    在PHP中,弱引用是通過WeakReference類實現的,不會阻止垃圾回收器回收對象。弱引用適用於緩存系統和事件監聽器等場景,需注意其不能保證對象存活,且垃圾回收可能延遲。

    解釋PHP中的__ Invoke Magic方法。解釋PHP中的__ Invoke Magic方法。Apr 12, 2025 am 12:07 AM

    \_\_invoke方法允許對象像函數一樣被調用。 1.定義\_\_invoke方法使對象可被調用。 2.使用$obj(...)語法時,PHP會執行\_\_invoke方法。 3.適用於日誌記錄和計算器等場景,提高代碼靈活性和可讀性。

    解釋PHP 8.1中的纖維以進行並發。解釋PHP 8.1中的纖維以進行並發。Apr 12, 2025 am 12:05 AM

    Fibers在PHP8.1中引入,提升了並發處理能力。 1)Fibers是一種輕量級的並發模型,類似於協程。 2)它們允許開發者手動控制任務的執行流,適合處理I/O密集型任務。 3)使用Fibers可以編寫更高效、響應性更強的代碼。

    PHP社區:資源,支持和發展PHP社區:資源,支持和發展Apr 12, 2025 am 12:04 AM

    PHP社區提供了豐富的資源和支持,幫助開發者成長。 1)資源包括官方文檔、教程、博客和開源項目如Laravel和Symfony。 2)支持可以通過StackOverflow、Reddit和Slack頻道獲得。 3)開發動態可以通過關注RFC了解。 4)融入社區可以通過積極參與、貢獻代碼和學習分享來實現。

    PHP與Python:了解差異PHP與Python:了解差異Apr 11, 2025 am 12:15 AM

    PHP和Python各有優勢,選擇應基於項目需求。 1.PHP適合web開發,語法簡單,執行效率高。 2.Python適用於數據科學和機器學習,語法簡潔,庫豐富。

    php:死亡還是簡單地適應?php:死亡還是簡單地適應?Apr 11, 2025 am 12:13 AM

    PHP不是在消亡,而是在不斷適應和進化。 1)PHP從1994年起經歷多次版本迭代,適應新技術趨勢。 2)目前廣泛應用於電子商務、內容管理系統等領域。 3)PHP8引入JIT編譯器等功能,提升性能和現代化。 4)使用OPcache和遵循PSR-12標準可優化性能和代碼質量。

    PHP的未來:改編和創新PHP的未來:改編和創新Apr 11, 2025 am 12:01 AM

    PHP的未來將通過適應新技術趨勢和引入創新特性來實現:1)適應云計算、容器化和微服務架構,支持Docker和Kubernetes;2)引入JIT編譯器和枚舉類型,提升性能和數據處理效率;3)持續優化性能和推廣最佳實踐。

    See all articles

    熱AI工具

    Undresser.AI Undress

    Undresser.AI Undress

    人工智慧驅動的應用程序,用於創建逼真的裸體照片

    AI Clothes Remover

    AI Clothes Remover

    用於從照片中去除衣服的線上人工智慧工具。

    Undress AI Tool

    Undress AI Tool

    免費脫衣圖片

    Clothoff.io

    Clothoff.io

    AI脫衣器

    AI Hentai Generator

    AI Hentai Generator

    免費產生 AI 無盡。

    熱門文章

    R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
    3 週前By尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O.最佳圖形設置
    3 週前By尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O.如果您聽不到任何人,如何修復音頻
    3 週前By尊渡假赌尊渡假赌尊渡假赌
    WWE 2K25:如何解鎖Myrise中的所有內容
    4 週前By尊渡假赌尊渡假赌尊渡假赌

    熱工具

    VSCode Windows 64位元 下載

    VSCode Windows 64位元 下載

    微軟推出的免費、功能強大的一款IDE編輯器

    mPDF

    mPDF

    mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

    MantisBT

    MantisBT

    Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

    WebStorm Mac版

    WebStorm Mac版

    好用的JavaScript開發工具

    ZendStudio 13.5.1 Mac

    ZendStudio 13.5.1 Mac

    強大的PHP整合開發環境