Symfony2 - ошибка Payum/TargetPay при получении платежа

У меня есть приложение в Symfony 2.8, в котором я получаю сообщение об ошибке при получении платежа через payum - omnipay-bridge и omnipay/targetpay.

Когда я звоню в мой prepareAction, меня перенаправляют на внешнюю платежную форму на сайте targetpay. Когда я отменяю (отмена = оплата в тестовом режиме), я перенаправляюсь обратно на мой сайт со следующим исключением:

Invalid input given. Should be an array or instance of \Traversable (500 Internal Server Error)

Когда я обновляю страницу, я оказываюсь на последнем маршруте "bsdb_payment_membership_status" (см. PaymentController), и статус моего платежа "новый". Что не так с моим процессом оплаты?

Мой composer.json выглядит так:

 "payum/payum-bundle": "~2.0",
  "payum/omnipay-bridge": "~1.2",
  "omnipay/targetpay": "~2.1",

Мой config.yml выглядит так:

payum:
    security:
        token_storage:
            Bsdb\BsdbPaymentBundle\Entity\PaymentToken: { doctrine: orm }

    storages:
        Bsdb\BsdbPaymentBundle\Entity\PaymentDetails: { doctrine: orm }

    gateways:
        bsdb_membership_payment:
            factory: omnipay
            type: "TargetPay_Mrcash"
            options:
                subAccountId: 123456
                testMode: true

Я создал собственный PaymentBundle с: - Entity PaymentDetails - Entity PaymentToken - Контроллер PaymentController с prepareAction и captureDoneAction

Entity PaymentToken

namespace Bsdb\BsdbPaymentBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Payum\Core\Model\Token;

/**
 * @ORM\Table
 * @ORM\Entity
 */
class PaymentToken extends Token
{
}

Entity PaymentDetails

namespace Bsdb\BsdbPaymentBundle\Entity;

use Doctrine\ORM\Mapping as ORM;
use Payum\Core\Model\ArrayObject;
use Application\Sonata\UserBundle\Entity\User;

/**
 * @ORM\Table(name="PaymentDetails")
 * @ORM\Entity
 */
class PaymentDetails extends ArrayObject
{
    /**
     * @ORM\Column(name="id", type="integer")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="IDENTITY")
     *
     * @var integer $id
     */
    protected $id;

    /**
     * @var User $user
     *
     * @ORM\ManyToOne(targetEntity="Application\Sonata\UserBundle\Entity\User")
     * @ORM\JoinColumns({
     *   @ORM\JoinColumn(name="user_id", referencedColumnName="id", nullable=false)
     * })
     */
    protected $user;

    /**
     * @var \DateTime
     *
     * @ORM\Column(name="started_at", type="datetime", nullable=true)
     */
    protected $startedAt;

    /**
     * @var \DateTime
     *
     * @ORM\Column(name="completed_at", type="datetime", nullable=true)
     */
    protected $completedAt;

    /**
     * Set start date:
     */
    public function __construct()
    {
        $this->setStartedAt(new \DateTime());
    }

    /**
     * @return int
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * @return User
     */
    public function getUser()
    {
        return $this->user;
    }

    /**
     * @param User $user
     */
    public function setUser($user)
    {
        $this->user = $user;
    }

    /**
     * @return \DateTime
     */
    public function getStartedAt()
    {
        return $this->startedAt;
    }

    /**
     * @param \DateTime $startedAt
     */
    public function setStartedAt($startedAt)
    {
        $this->startedAt = $startedAt;
    }

    /**
     * @return \DateTime
     */
    public function getCompletedAt()
    {
        return $this->completedAt;
    }

    /**
     * @param \DateTime $completedAt
     */
    public function setCompletedAt($completedAt)
    {
        $this->completedAt = $completedAt;
    }

    public function getDetailsString()
    {
        return json_encode($this->details);
    }
}

Контроллер PaymentController

namespace Bsdb\BsdbPaymentBundle\Controller;

use Application\Sonata\UserBundle\Entity\User;
use Application\Sonata\UserBundle\Event\UserEvent;
use Application\Sonata\UserBundle\Service\UserGroupAttacher;
use Application\Sonata\UserBundle\UserEvents;
use Symfony\Bundle\FrameworkBundle\Controller\Controller;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Route;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Template;
use Symfony\Component\HttpFoundation\Request;
use Payum\Core\Request\GetHumanStatus;
use Symfony\Component\HttpFoundation\JsonResponse;

class PaymentController extends Controller
{
    /**
     * @Route(name="bsdb_payment_membership_prepare", path="/membership/prepare")
     */
    public function prepareAction()
    {
        $paymentName = 'bsdb_membership_payment';
        $storage = $this->get('payum')->getStorage('Bsdb\BsdbPaymentBundle\Entity\PaymentDetails');
        $currentUser = $this->get('security.context')->getToken()->getUser();

        /** @var \Bsdb\BsdbPaymentBundle\Entity\PaymentDetails $paymentDetails */
        $paymentDetails = $storage->create();

        $paymentDetails['amount'] = $this->container->getParameter('premium.amount');
        $paymentDetails['description'] = $this->container->getParameter('premium.description');
        $paymentDetails['clientIp'] = $this->get('request')->getClientIp();
        $paymentDetails->setUser($currentUser);

        $storage->update($paymentDetails);

        $captureToken = $this->get('payum')->getTokenFactory()->createCaptureToken(
            $paymentName,
            $paymentDetails,
            'bsdb_payment_membership_captured' // the route to redirect after capture;
        );

        return $this->redirect($captureToken->getTargetUrl());
    }

    /**
     * @Route(name="bsdb_payment_membership_captured", path="/membership/captured")
     */
    public function captureDoneAction(Request $request)
    {
        $token = $this->get('payum')->getHttpRequestVerifier()->verify($request);
        $gateway = $this->get('payum')->getGateway($token->getGatewayName());

        $gateway->execute($status = new GetHumanStatus($token));
        $payment = $status->getFirstModel();

        if ($status->isCaptured()) {
            /** @var User $user */
            $user = $this->getUser();
            $user->setLastPremiumUpgrade(new \DateTime());
            $user->setPremiumExpiresAt(new \DateTime(sprintf('+%s days', $this->container->getParameter('premium.expiration.period'))));

            // Set group
            $entityManager = $this->getDoctrine()->getManager();
            $groupAttachter = new UserGroupAttacher($entityManager);
            $groupAttachter->attachUserGroup($user, $this->container->getParameter('premium.group'));

            // Update payment data:
            $payment->setCompletedAt(new \DateTime());

            // Save:
            $entityManager->persist($user);
            $entityManager->flush();

            // Trigger events for emails and stuff:
            $event = new UserEvent($user);
            $this->get('event_dispatcher')->dispatch(UserEvents::USER_PREMIUM_PURCHASED, $event);
        }

        return $this->redirect($this->generateUrl('bsdb_payment_membership_status', array('status' => $status->getValue())));
    }

    /**
     * @Route(name="bsdb_payment_membership_status", path="/membership/status/{status}")
     * @Template()
     */
    public function paidAction($status)
    {
        return array('status' => $status);
    }

} 

Стек трассировки исключений

Спасибо за ваш опыт!

0 ответов

Другие вопросы по тегам