We will work on Apr 26th (Saturday) and will be off from Apr 30th (Wednesday) until May 2nd (Friday) for public holiday in our country

RefundDataBuilder.php 9.24 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
<?php
/**
 *                       ######
 *                       ######
 * ############    ####( ######  #####. ######  ############   ############
 * #############  #####( ######  #####. ######  #############  #############
 *        ######  #####( ######  #####. ######  #####  ######  #####  ######
 * ###### ######  #####( ######  #####. ######  #####  #####   #####  ######
 * ###### ######  #####( ######  #####. ######  #####          #####  ######
 * #############  #############  #############  #############  #####  ######
 *  ############   ############  #############   ############  #####  ######
 *                                      ######
 *                               #############
 *                               ############
 *
 * Adyen Payment module (https://www.adyen.com/)
 *
 * Copyright (c) 2015 Adyen BV (https://www.adyen.com/)
 * See LICENSE.txt for license details.
 *
 * Author: Adyen <magento@adyen.com>
 */

namespace Adyen\Payment\Gateway\Request;

use Magento\Payment\Gateway\Request\BuilderInterface;

/**
 * Class CustomerDataBuilder
 */
class RefundDataBuilder implements BuilderInterface
{

    /**
     * @var \Adyen\Payment\Helper\Data
     */
    private $adyenHelper;

39
    /**
40
     * @var \Adyen\Payment\Model\ResourceModel\Order\Payment\CollectionFactory
41 42 43
     */
    private $orderPaymentCollectionFactory;

44 45 46 47 48
    /**
     * @var \Adyen\Payment\Model\ResourceModel\Invoice\CollectionFactory
     */
    protected $adyenInvoiceCollectionFactory;

49
    /**
50
     * RefundDataBuilder constructor.
51
     * @param \Adyen\Payment\Helper\Data $adyenHelper
52
     * @param \Adyen\Payment\Model\ResourceModel\Order\Payment\CollectionFactory $orderPaymentCollectionFactory
53
     */
54 55
    public function __construct(
        \Adyen\Payment\Helper\Data $adyenHelper,
56 57 58
        \Adyen\Payment\Model\ResourceModel\Order\Payment\CollectionFactory $orderPaymentCollectionFactory,
        \Adyen\Payment\Model\ResourceModel\Invoice\CollectionFactory $adyenInvoiceCollectionFactory
    ) {
59
        $this->adyenHelper = $adyenHelper;
60
        $this->orderPaymentCollectionFactory = $orderPaymentCollectionFactory;
61
        $this->adyenInvoiceCollectionFactory = $adyenInvoiceCollectionFactory;
62 63 64 65 66 67 68 69 70 71
    }

    /**
     * @param array $buildSubject
     * @return array
     */
    public function build(array $buildSubject)
    {
        /** @var \Magento\Payment\Gateway\Data\PaymentDataObject $paymentDataObject */
        $paymentDataObject = \Magento\Payment\Gateway\Helper\SubjectReader::readPayment($buildSubject);
72
        $amount = \Magento\Payment\Gateway\Helper\SubjectReader::readAmount($buildSubject);
73

74
        $order = $paymentDataObject->getOrder();
75 76 77
        $payment = $paymentDataObject->getPayment();
        $pspReference = $payment->getCcTransId();
        $currency = $payment->getOrder()->getOrderCurrencyCode();
78
        $storeId = $order->getStoreId();
79 80
        $method = $payment->getMethod();
        $merchantAccount = $this->adyenHelper->getAdyenMerchantAccount($method, $storeId);
81 82 83 84
        $grandTotal = $payment->getOrder()->getGrandTotal();


        // check if it contains a split payment
85 86 87
        $orderPaymentCollection = $this->orderPaymentCollectionFactory
            ->create()
            ->addFieldToFilter('payment_id', $payment->getId());
88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104

        // partial refund if multiple payments check refund strategy
        if ($orderPaymentCollection->getSize() > 1) {
            $refundStrategy = $this->adyenHelper->getAdyenAbstractConfigData(
                'split_payments_refund_strategy',
                $order->getStoreId()
            );
            $ratio = null;

            if ($refundStrategy == "1") {
                // Refund in ascending order
                $orderPaymentCollection->addPaymentFilterAscending($payment->getId());
            } elseif ($refundStrategy == "2") {
                // Refund in descending order
                $orderPaymentCollection->addPaymentFilterDescending($payment->getId());
            } elseif ($refundStrategy == "3") {
                // refund based on ratio
105
                $ratio = $amount / $grandTotal;
106 107 108 109
                $orderPaymentCollection->addPaymentFilterAscending($payment->getId());
            }

            // loop over payment methods and refund them all
110
            $requestBody = [];
111 112 113 114 115
            foreach ($orderPaymentCollection as $splitPayment) {
                // could be that not all the split payments need a refund
                if ($amount > 0) {
                    if ($ratio) {
                        // refund based on ratio calculate refund amount
116
                        $modificationAmount = $ratio * (
117
                                $splitPayment->getAmount() - $splitPayment->getTotalRefunded()
118
                            );
119 120
                    } else {
                        // total authorised amount of the split payment
121
                        $splitPaymentAmount = $splitPayment->getAmount() - $splitPayment->getTotalRefunded();
122

123 124 125 126 127
                        // if rest amount is zero go to next payment
                        if (!$splitPaymentAmount > 0) {
                            continue;
                        }

128
                        // if refunded amount is greater than split payment amount do a full refund
129 130 131 132 133 134 135 136 137 138 139 140 141
                        if ($amount >= $splitPaymentAmount) {
                            $modificationAmount = $splitPaymentAmount;
                        } else {
                            $modificationAmount = $amount;
                        }
                        // update amount with rest of the available amount
                        $amount = $amount - $splitPaymentAmount;
                    }

                    $modificationAmountObject = [
                        'currency' => $currency,
                        'value' => $this->adyenHelper->formatAmount($modificationAmount, $currency)
                    ];
142

143
                    $requestBody[] = [
144 145 146 147 148 149 150 151 152 153 154
                        "modificationAmount" => $modificationAmountObject,
                        "reference" => $payment->getOrder()->getIncrementId(),
                        "originalReference" => $splitPayment->getPspreference(),
                        "merchantAccount" => $merchantAccount
                    ];
                }
            }
        } else {
            //format the amount to minor units
            $amount = $this->adyenHelper->formatAmount($amount, $currency);
            $modificationAmount = ['currency' => $currency, 'value' => $amount];
155

156
            $requestBody = [
157 158 159 160 161 162 163
                [
                    "modificationAmount" => $modificationAmount,
                    "reference" => $payment->getOrder()->getIncrementId(),
                    "originalReference" => $pspReference,
                    "merchantAccount" => $merchantAccount
                ]
            ];
164 165 166 167 168 169 170 171 172

            $brandCode = $payment->getAdditionalInformation(
                \Adyen\Payment\Observer\AdyenHppDataAssignObserver::BRAND_CODE
            );

            if ($this->adyenHelper->isPaymentMethodOpenInvoiceMethod($brandCode)) {
                $openInvoiceFields = $this->getOpenInvoiceData($payment);

                //There is only one payment, so we add the fields to the first(and only) result
173
                $requestBody[0]["additionalData"] = $openInvoiceFields;
174
            }
175
        }
176

177 178 179
        $request['body'] = $requestBody;

        return $request;
180
    }
181

attilak's avatar
attilak committed
182 183 184 185
    /**
     * @param \Magento\Payment\Model\InfoInterface $payment
     * @return array|mixed
     */
186 187 188 189 190 191 192
    protected function getOpenInvoiceData($payment)
    {
        $formFields = [];
        $count = 0;
        $currency = $payment->getOrder()->getOrderCurrencyCode();

        /**
attilak's avatar
attilak committed
193
         * @var \Magento\Sales\Model\Order\Creditmemo $creditMemo
194 195 196
         */
        $creditMemo = $payment->getCreditMemo();

attilak's avatar
attilak committed
197
        foreach ($creditMemo->getItems() as $refundItem) {
198
            ++$count;
199
            $numberOfItems = (int)$refundItem->getQty();
200 201 202 203 204 205 206 207 208

            $formFields = $this->adyenHelper->createOpenInvoiceLineItem(
                $formFields,
                $count,
                $refundItem->getName(),
                $refundItem->getPrice(),
                $currency,
                $refundItem->getTaxAmount(),
                $refundItem->getPriceInclTax(),
attilak's avatar
attilak committed
209
                $refundItem->getOrderItem()->getTaxPercent(),
210
                $numberOfItems,
211 212
                $payment,
                $refundItem->getId()
213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238
            );
        }

        // Shipping cost
        if ($creditMemo->getShippingAmount() > 0) {
            ++$count;
            $formFields = $this->adyenHelper->createOpenInvoiceLineShipping(
                $formFields,
                $count,
                $payment->getOrder(),
                $creditMemo->getShippingAmount(),
                $creditMemo->getShippingTaxAmount(),
                $currency,
                $payment
            );
        }

        $formFields['openinvoicedata.numberOfLines'] = $count;

        //Retrieve acquirerReference from the adyen_invoice
        $invoiceId = $creditMemo->getInvoice()->getId();
        $invoices = $this->adyenInvoiceCollectionFactory->create()
            ->addFieldToFilter('invoice_id', $invoiceId);

        $invoice = $invoices->getFirstItem();

239
        if ($invoice) {
240 241 242 243 244
            $formFields['acquirerReference'] = $invoice->getAcquirerReference();
        }

        return $formFields;
    }
245
}