123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397 |
- <?php
- /**
- * Archivo SDK de Pagopar
- * @author "Pagopar" <desarrollo@pagopar.com>
- * @version 1 27/4/2017
- */
- require_once 'lib/DBPagopar.php';
- require_once 'classes/OrderPagopar.php';
- require_once 'classes/ConsultPagopar.php';
- class Pagopar{
- //URLs de configuración
- const URL_BASE = 'https://api.pagopar.com/api/';
- const URL_COMERCIOS = self::URL_BASE.'comercios/1.1/iniciar-transaccion';
- const URL_PEDIDOS = self::URL_BASE.'pedidos/1.1/traer';
- const URL_FLETE = self::URL_BASE.'calcular-flete/1.1/traer';
- const URL_CATEGORIAS = self::URL_BASE.'categorias/1.1/traer';
- const URL_CIUDADES = self::URL_BASE.'ciudades/1.1/traer';
- const URL_REDIRECT = 'https://pagopar.com/pagos/%s';
- //Tipos de Tokens generados
- const TOKEN_TIPO_CONSULTA = 'CONSULTA';
- const TOKEN_TIPO_CIUDAD = 'CIUDADES';
- const TOKEN_TIPO_CATEGORIA = 'CATEGORIAS';
- const TOKEN_TIPO_FLETE = 'CALCULAR-FLETE';
- //Base de datos
- protected $db;
- //Datos del pedido del comercio
- private $idOrder;
- private $hashOrder;
- public $order;
- private $preOrder;
- private $itemsIdMethods;
- public $privateKey = null;
- public $publicKey = null;
- /**
- * Constructor de la clase
- * @param int $id Id del pedido
- * @param $db
- * @internal param Database $PDO $db Base de Datos (Basada en PDO)
- */
- public function __construct($id = null,$db) {
- $this->db = $db;
- $this->idOrder = $id;
- $this->order = new OrderPagopar($id);
- }
- /**
- * Invoca a la URL de acuerdo a los parámetros
- * @param array $args Parámetros
- * @param string $url Url a invocar
- * @return string Respuesta en formato JSON
- */
- private function runCurl($args, $url){
- $args = json_encode($args);
- $ch = curl_init();
- $headers= array('Accept: application/json','Content-Type: application/json');
- curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
- curl_setopt($ch, CURLOPT_URL, $url);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);
- curl_setopt($ch, CURLOPT_POST, true);
- curl_setopt($ch, CURLOPT_POSTFIELDS, $args);
- $response = curl_exec($ch);
- $error = curl_error($ch);
- $info = curl_getinfo($ch);
- curl_close($ch);
- return $response;
- }
- /**
- * Guarda en un array ($itemsIdMethods) los id_producto de los ítems y sus índices en el array
- * compras_items ([id_producto => índice]) para que tengamos un acceso más rápido a los ítems.
- * Además, para los ítems con más de un método de envío, guardamos los tipos de envío disponibles
- * en un array ($multiple_methods) con sus valores para retornar después al usuario.
- * @param array $arrayResponse Array que contiene las distintas opciones de envío de los ítems.
- * @return string Respuesta en formato JSON que contiene las distintas opciones de envío de los ítems.
- * Si se retorna false, se asume que no existen opciones de envío o que sólo existe una opción
- * de envío para cada ítem.
- */
- private function saveMethodsOfShipping($arrayResponse){
- $multiple_methods = [];
- $this->itemsIdMethods = [];
- //Si alguno de los ítems devueltos tiene los dos métodos de pago,
- foreach ($arrayResponse->compras_items as $index => $item) {
- //En un array guardamos los ids de los productos como índices y los índices como valor
- //para que sean más fáciles de encontrar después.
- $this->itemsIdMethods[$item->id_producto] = $index;
- if(isset($item->opciones_envio) && isset($item->opciones_envio->metodo_aex) && isset($item->opciones_envio->propio)){
- $multiple_methods[$item->id_producto] = [
- "metodo_aex"=>$item->opciones_envio['metodo_aex'],
- "propio"=>$item->opciones_envio['propio']
- ];
- }
- }
- //Guardamos la respuesta en preOrder
- $this->preOrder = $arrayResponse;
- return (!empty($multiple_methods))?json_encode($multiple_methods):false;
- }
- /**
- * Inicia la conexión con Pagopar y obtiene los métodos de envío para cada ítem
- * @return string|boolean Respuesta en formato JSON que contiene las distintas opciones de envío de los ítems.
- * Si se retorna false, se asume que no existen opciones de envío o que sólo existe una opción
- * de envío para cada ítem.
- */
- public function getMethodsOfShipping(){
- $orderPagopar = $this->order->makeOrder();
- $token = $this->generateToken(self::TOKEN_TIPO_FLETE);
- $args = ['token'=>$token,'token_publico'=>$this->order->publicKey, 'dato'=>json_encode($orderPagopar)];
- $response = $this->runCurl($args, self::URL_FLETE);
- $arrayResponse = json_decode($response);
- if(isset($arrayResponse->respuesta)&&!$arrayResponse->respuesta){
- throw new Exception($arrayResponse->resultado);
- }elseif(isset($arrayResponse->resultado) && $arrayResponse->resultado === "Sin datos"){
- throw new Exception($arrayResponse->resultado);
- }
- return $this->saveMethodsOfShipping($arrayResponse);
- }
- /**
- * Valida el JSON de los métodos de envío seleccionados.
- * @param array $items Array con los métodos de envío seleccionados. Tiene esta estructura:
- * ["id_producto_1" => "metodo_1", "id_producto_2" => "metodo_2", ... => ...]
- * @return bool True si el JSON es válido (los id de los productos existen y los métodos de pago existen),
- * False si no es un JSON válido.
- */
- private function validateItemsMethodsJSON($items){
- foreach ($items as $id => $method){
- if(array_key_exists($id,$this->itemsIdMethods)){
- if(array_key_exists($method,$this->preOrder->compras_items[$this->itemsIdMethods[$id]]->opciones_envio)){
- return true;
- }else{
- return false;
- }
- }else{
- return false;
- }
- }
- return false;
- }
- /**
- * Completa los ítems de la orden con el campo envio_seleccionado
- * @param string $json_items Cadena JSON que contiene las opciones de envío de los ítems.
- * @return array Array de respuesta que contiene la Orden con el flete calculado de los ítems o
- * un array de error en caso de fallo.
- * @throws Exception
- */
- private function fillShippingOptions($json_items){
- $items_methods = null;
- if($json_items) {
- $items_methods = json_decode($json_items);
- if (!$this->validateItemsMethodsJSON($items_methods)) {
- throw new Exception("Hay un error con el JSON de los métodos seleccionados de los items");
- }
- }
- foreach($this->preOrder->compras_items as $index => $item){
- //Obtenemos el primer y único elemento en el caso de que sólo haya un método seleccionado
- if(isset($item->opciones_envio)){
- $key = key($item->opciones_envio);
- $item->envio_seleccionado = ($key==="metodo_aex")?"aex":$key;
- }
- }
- //Completamos los campos necesarios para envio_seleccionado
- if($json_items){
- foreach($items_methods as $id => $method){
- $this->preOrder->compras_items[$this->itemsIdMethods[$id]]->envio_seleccionado = $method;
- }
- }
- //Calculamos el flete
- return $this->getShippingCost();
- }
- /**
- * Calcula el flete de los ítems si es necesario
- * @return array Array de respuesta que contiene la Orden con el flete calculado de los ítems o
- * un array de error en caso de fallo.
- */
- private function getShippingCost(){
- $token = $this->generateToken(self::TOKEN_TIPO_FLETE);
- $args = ['token'=>$token,'token_publico'=>$this->order->publicKey, 'dato'=> json_encode($this->preOrder)];
- $response = $this->runCurl($args, self::URL_FLETE);
- return json_decode($response);
- }
- /**
- * Reenvía el pedido a Pagopar (con el flete calculado) y genera una transacción. Si tiene éxito al
- * generar el pedido, redirecciona a la página de pago de Pagopar.
- * @param string $json_items Cadena JSON que contiene las opciones de envío de los ítems.
- * Si no recibe este parámetro, se asume que no existen opciones de envío o que sólo existe una opción
- * de envío para cada ítem.
- * @throws Exception
- */
- public function newPagoparTransaction($json_items=null){
- $response = $this->fillShippingOptions($json_items);
- //Verificar si hay error
- if(isset($response->respuesta) && !$response->respuesta){
- throw new Exception($response->resultado);
- }
- //Asignamos de nuevo a la orden la respuesta
- $this->preOrder = $response;
- //Generamos de nuevo el token
- $this->preOrder->token = $this->order->generateOrderHash(
- $this->preOrder->id_pedido_comercio,
- $this->preOrder->monto_total,
- $this->order->privateKey
- );
- $response = $this->runCurl($this->preOrder, self::URL_COMERCIOS);
- $arrayResponse = json_decode($response);
- //Verificar si hay error
- if(!$arrayResponse->respuesta){
- throw new Exception($arrayResponse->resultado);
- }
- $this->hashOrder = $arrayResponse->resultado[0]->data;
- $this->db->insertTransaction(
- $this->preOrder->id_pedido_comercio,
- $this->preOrder->tipo_pedido,
- $this->preOrder->monto_total,
- $this->hashOrder,
- $this->preOrder->fecha_maxima_pago,
- $this->preOrder->descripcion_resumen
- );
- $this->redirectToPagopar($this->hashOrder);
- }
- /**
- * Redirecciona a la página de Pagopar
- * @param string $hash Hash del pedido
- */
- public function redirectToPagopar($hash){
- $url = sprintf(self::URL_REDIRECT, $hash);
- //Redireccionamos a Pagopar
- header('Location: '. $url);
- exit();
- }
- /**
- * Inicia la transacción con Pagopar y si tiene éxito al generar el pedido con valores de prueba,
- * redirecciona a la página de pago de Pagopar.
- */
- public function newTestPagoparTransaction($public_key,$private_key,$seller_public_key){
- //Creamos el comprador
- $buyer = new BuyerPagopar();
- $buyer->name = 'Juan Perez';
- $buyer->cityId = 1;
- $buyer->tel = '0972200046';
- $buyer->typeDoc = 'CI';
- $buyer->doc = '352221';
- $buyer->addr = 'Mexico 840';
- $buyer->addRef = 'alado de credicentro';
- $buyer->addrCoo = '-25.2844638,-57.6480038';
- $buyer->ruc = null;
- $buyer->socialReason = null;
- //Agregamos el comprador
- $this->order->addPagoparBuyer($buyer);
- //Creamos los productos
- $item1 = new ItemPagopar();
- $item1->name = "Válido 1 persona";
- $item1->qty = 1;
- $item1->price = 1000;
- $item1->cityId = 1;
- $item1->desc = "producto";
- $item1->url_img = "http://www.clipartkid.com/images/318/tickets-for-the-film-festival-are-for-the-two-day-event-admission-is-lPOEYl-clipart.png";
- $item1->weight = '0.1';
- $item1->category = 3;
- $item1->productId = 100;
- $item1->sellerPhone = '0985885487';
- $item1->sellerAddress = 'dr paiva ca cssssom gaa';
- $item1->sellerAddressRef = '';
- $item1->sellerAddressCoo = '-28.75438,-57.1580038';
- $item1->sellerPublicKey = $seller_public_key;
- $item2 = new ItemPagopar();
- $item2->name = "Heladera";
- $item2->qty = 1;
- $item2->price = 785000;
- $item2->cityId = 1;
- $item2->desc = "producto";
- $item2->url_img = "https://cdn1.hendyla.com/archivos/imagenes/2017/04/09/publicacion-564c19b86b235526160f43483c76a69ee1a85c96c976c33e3e21ce6a5f9009b9-234_A.jpg";
- $item2->weight = '5.0';
- $item2->category = 8;//Cambiar a 8
- $item2->productId = 2;
- $item2->sellerPhone = '0985885487';
- $item2->sellerAddress = 'dr paiva ca cssssom gaa';
- $item2->sellerAddressRef = '';
- $item2->sellerAddressCoo = '-28.75438,-57.1580038';
- $item2->sellerPublicKey = $seller_public_key;
- //Agregamos los productos al pedido
- $this->order->addPagoparItem($item1);
- $this->order->addPagoparItem($item2);
- $this->order->publicKey = $public_key;
- $this->order->privateKey = $private_key;
- $this->order->typeOrder = 'VENTA-COMERCIO';
- $this->order->desc = 'Entrada Retiro';
- $this->order->periodOfDaysForPayment = 1;
- $this->order->periodOfHoursForPayment = 0;
- /*Calculamos el flete y... si es algún ítem tiene opcion_envio metodo_aex Y propio, entonces retornamos
- un json en el que el usuario debe seleccionar el método de pago para el ítem específico*/
- $json_pedido_con_flete = $this->getMethodsOfShipping();
- if( !$json_pedido_con_flete ){
- $this->newPagoparTransaction();
- }else{
- /*Seleccionamos los métodos de envío y después generamos la nueva transacción
- Método para seleccionar el envío, debe ser una pantalla intermedia
- y hay que llamar a la función newPagoparTransaction con un array como parámetro.
- El json debe tener esta forma:
- {
- id_producto_1 : metodo_seleccionado_1,
- id_producto_2 : metodo_seleccionado_2,
- ... : ...
- }
- */
- //$json_items_metodo_seleccionado = '{"100":"aex"}';
- //$pedidoPagopar->newPagoparTransaction($json_items_metodo_seleccionado);
- }
- }
- /**
- * Obtiene un JSON con el estado del pedido
- * @param int $id Id del pedido
- * @return JSON con el estado del Pedido
- * @throws Exception
- */
- public function getPagoparOrderStatus($id){
- $this->idOrder = $id;
- $orderData = $this->db->selectTransaction("id=$id");
- if($orderData){
- $this->hashOrder = $orderData['hash'];
- }else{
- throw new Exception("Hay un error con el hash");
- }
- $token = $this->generateToken(self::TOKEN_TIPO_CONSULTA);
- $args = ['hash_pedido'=>$this->hashOrder, 'token'=>$token, 'token_publico'=>$this->publicKey];
- $arrayResponse = $this->runCurl($args, self::URL_PEDIDOS);
- return $arrayResponse;
- }
- /**
- * Genera un Token para el pedido
- * @param string $typeOfToken Tipo de token generado
- * @return string Token generado
- */
- private function generateToken($typeOfToken){
- $key = ($this->privateKey)?$this->privateKey:$this->order->privateKey;
- return sha1($key.$typeOfToken);
- }
- #registrar usuario
- public function registrarUsuario(array $json){
- $url = self::URL_BASE.'usuario/1.1/registro';
- $args = $json;
- $arrayResponse = $this->runCurl($args, $url);
- return $arrayResponse;
- }
- }
|