Saltar a contenido

FlowProvider

FlowProvider

Bases: BasicProvider

FlowProvider es una clase que proporciona integración con Flow para procesar pagos. Inicializa una instancia de FlowProvider con el key y el secreto de Flow.

Parameters:

Name Type Description Default
api_key str

ApiKey entregada por Flow.

required
api_secret str

ApiSecret entregada por Flow.

required
api_medio int | None

Versión de la API de notificaciones a utilizar (Valor por defecto: 9).

9
api_endpoint str

Ambiente flow, puede ser "live" o "sandbox" (Valor por defecto: live).

'live'
**kwargs int

Argumentos adicionales.

{}
Source code in django_payments_chile/FlowProvider.py
class FlowProvider(BasicProvider):
    """
    FlowProvider es una clase que proporciona integración con Flow para procesar pagos.
    Inicializa una instancia de FlowProvider con el key y el secreto de Flow.

    Args:
        api_key (str): ApiKey entregada por Flow.
        api_secret (str): ApiSecret entregada por Flow.
        api_medio (int | None): Versión de la API de notificaciones a utilizar (Valor por defecto: 9).
        api_endpoint (str): Ambiente flow, puede ser "live" o "sandbox" (Valor por defecto: live).
        **kwargs: Argumentos adicionales.
    """

    form_class = BasePaymentForm
    api_endpoint: str
    api_key: str = None
    api_secret: str = None
    api_medio: int

    def __init__(
        self,
        api_key: str,
        api_secret: str,
        api_endpoint: str = "live",
        api_medio: int = 9,
        **kwargs: int,
    ):
        super().__init__(**kwargs)
        self.api_endpoint = api_endpoint
        self.api_key = api_key
        self.api_secret = api_secret
        self.api_medio = api_medio
        if self.api_endpoint == "live":
            self.api_endpoint = "https://www.flow.cl/api"
        elif self.api_endpoint == "sandbox":
            self.api_endpoint = "https://sandbox.flow.cl/api"

    def get_form(self, payment, data: Optional[dict] = None) -> Any:
        """
        Genera el formulario de pago para redirigir a la página de pago de Flow.

        Args:
            payment ("Payment"): Objeto de pago Django Payments.
            data (dict | None): Datos del formulario (opcional).

        Returns:
            Any: Formulario de pago redirigido a la página de pago de Flow.

        Raises:
            RedirectNeeded: Redirige a la página de pago de Flow.

        """
        if not payment.transaction_id:
            datos_para_flow = {
                "apiKey": self.api_key,
                "commerceOrder": str(payment.token),
                "urlReturn": payment.get_success_url(),
                "urlConfirmation": payment.get_process_url(),
                "subject": payment.description,
                "amount": int(payment.total),
                "paymentMethod": self.api_medio,
                "currency": payment.currency,
            }

            if payment.billing_email:
                datos_para_flow.update({"email": payment.billing_email})

            datos_para_flow.update(**self._extra_data(payment.attrs))

            try:
                payment.attrs.datos_payment_create_flow = datos_para_flow
                payment.save()
            except Exception as e:  # noqa
                # Dificil llegar acá, y si llegamos es problema de django-payments
                raise PaymentError(f"Ocurrió un error al guardar attrs.datos_flow: {e}")  # noqa
            datos_para_flow = dict(sorted(datos_para_flow.items()))
            firma_datos = ClienteAPI.genera_firma(datos_para_flow, self.api_secret)
            datos_para_flow.update({"s": firma_datos})

            try:
                pago_req = requests.post(f"{self.api_endpoint}/payment/create", data=datos_para_flow, timeout=5)
                pago_req.raise_for_status()

            except Exception as pe:
                payment.change_status(PaymentStatus.ERROR, str(pe))
                raise PaymentError(pe)
            else:
                pago = pago_req.json()
                payment.transaction_id = pago["token"]
                payment.attrs.respuesta_flow = {
                    "url": pago["url"],
                    "token": pago["token"],
                    "flowOrder": pago["flowOrder"],
                }
                payment.save()
                payment.change_status(PaymentStatus.WAITING)

            raise RedirectNeeded(f"{pago['url']}?token={pago['token']}")

    def process_data(self, payment, request) -> JsonResponse:
        """
        Procesa los datos del pago recibidos desde Flow.

        Args:
            payment ("Payment"): Objeto de pago Django Payments.
            request ("HttpRequest"): Objeto de solicitud HTTP de Django.

        Returns:
            JsonResponse: Respuesta JSON que indica el procesamiento de los datos del pago.

        """
        if "token" not in request.POST:
            return HttpResponseBadRequest("token no está en post")

        if payment.status in [PaymentStatus.WAITING, PaymentStatus.PREAUTH]:
            self.actualiza_estado(payment=payment)

        return JsonResponse({"status": "ok"})

    def actualiza_estado(self, payment) -> dict:
        """Actualiza el estado del pago con Flow

        Args:
            payment ("Payment): Objeto de pago Django Payments.

        Returns:
            dict: Diccionario con valores del objeto `PaymentStatus`.
        """
        datos_para_flow = {"apiKey": self.api_key, "token": payment.token}
        datos_para_flow = dict(sorted(datos_para_flow.items()))
        firma_datos = ClienteAPI.genera_firma(datos_para_flow, self.api_secret)
        datos_para_flow.update({"s": firma_datos})

        try:
            # status = FlowPayment.getStatus(self._client, payment.transaction_id)
            estado_req = requests.get(f"{self.api_endpoint}/payment/getStatus", data=datos_para_flow, timeout=5)
            estado_req.raise_for_status()

        except Exception as e:
            raise e
        else:
            status = estado_req.json()
            if status["status"] == 2:
                payment.change_status(PaymentStatus.CONFIRMED)
            elif status["status"] == 3:
                payment.change_status(PaymentStatus.REJECTED)
            elif status["status"] == 4:
                payment.change_status(PaymentStatus.ERROR)
        return status

    def _extra_data(self, attrs) -> dict:
        """Busca los datos que son enviandos por django-payments y los saca del diccionario

        Args:
            attrs ("PaymentAttributeProxy"): Obtenido desde PaymentModel.extra_data

        Returns:
            dict: Diccionario con valores permitidos.
        """
        try:
            data = attrs.datos_extra
        except AttributeError:
            return {}

        prohibidos = [
            "commerceOrder",
            "urlReturn",
            "urlConfirmation",
            "amount",
            "subject",
            "paymentMethod",
            "currency",
        ]
        for valor in prohibidos:
            if valor in data:
                del data[valor]

        return data

    def refund(self, payment, amount: Optional[int] = None) -> int:
        """
        Realiza un reembolso del pago.
        El seguimiendo se debe hacer directamente en Flow

        Args:
            payment ("Payment"): Objeto de pago Django Payments.
            amount (int | None): Monto a reembolsar (opcional).

        Returns:
            int: Monto de reembolso solicitado.

        Raises:
            PaymentError: Error al crear el reembolso.

        """
        if payment.status != PaymentStatus.CONFIRMED:
            raise PaymentError("El pago debe estar confirmado para reversarse.")

        to_refund = amount or payment.total

        datos_reembolso = {
            "apiKey": self.api_key,
            "refundCommerceOrder": payment.token,
            "receiverEmail": payment.billing_email,
            "amount": to_refund,
            "urlCallBack": payment.get_process_url(),
            "commerceTrxId": payment.token,
            "flowTrxId": payment.attrs.respuesta_flow["flowOrder"],
        }
        try:
            refun_req = requests.post(f"{self.api_endpoint}/refund/create", data=datos_reembolso, timeout=5)
            refun_req.raise_for_status()
        except Exception as pe:
            raise PaymentError(pe)
        else:
            payment.attrs.solicitud_reembolso = refun_req.json()
            payment.save()
            payment.change_status(PaymentStatus.REFUNDED)
            return to_refund

actualiza_estado(payment)

Actualiza el estado del pago con Flow

Parameters:

Name Type Description Default
payment "Payment

Objeto de pago Django Payments.

required

Returns:

Name Type Description
dict dict

Diccionario con valores del objeto PaymentStatus.

Source code in django_payments_chile/FlowProvider.py
def actualiza_estado(self, payment) -> dict:
    """Actualiza el estado del pago con Flow

    Args:
        payment ("Payment): Objeto de pago Django Payments.

    Returns:
        dict: Diccionario con valores del objeto `PaymentStatus`.
    """
    datos_para_flow = {"apiKey": self.api_key, "token": payment.token}
    datos_para_flow = dict(sorted(datos_para_flow.items()))
    firma_datos = ClienteAPI.genera_firma(datos_para_flow, self.api_secret)
    datos_para_flow.update({"s": firma_datos})

    try:
        # status = FlowPayment.getStatus(self._client, payment.transaction_id)
        estado_req = requests.get(f"{self.api_endpoint}/payment/getStatus", data=datos_para_flow, timeout=5)
        estado_req.raise_for_status()

    except Exception as e:
        raise e
    else:
        status = estado_req.json()
        if status["status"] == 2:
            payment.change_status(PaymentStatus.CONFIRMED)
        elif status["status"] == 3:
            payment.change_status(PaymentStatus.REJECTED)
        elif status["status"] == 4:
            payment.change_status(PaymentStatus.ERROR)
    return status

get_form(payment, data=None)

Genera el formulario de pago para redirigir a la página de pago de Flow.

Parameters:

Name Type Description Default
payment Payment

Objeto de pago Django Payments.

required
data dict | None

Datos del formulario (opcional).

None

Returns:

Name Type Description
Any Any

Formulario de pago redirigido a la página de pago de Flow.

Raises:

Type Description
RedirectNeeded

Redirige a la página de pago de Flow.

Source code in django_payments_chile/FlowProvider.py
def get_form(self, payment, data: Optional[dict] = None) -> Any:
    """
    Genera el formulario de pago para redirigir a la página de pago de Flow.

    Args:
        payment ("Payment"): Objeto de pago Django Payments.
        data (dict | None): Datos del formulario (opcional).

    Returns:
        Any: Formulario de pago redirigido a la página de pago de Flow.

    Raises:
        RedirectNeeded: Redirige a la página de pago de Flow.

    """
    if not payment.transaction_id:
        datos_para_flow = {
            "apiKey": self.api_key,
            "commerceOrder": str(payment.token),
            "urlReturn": payment.get_success_url(),
            "urlConfirmation": payment.get_process_url(),
            "subject": payment.description,
            "amount": int(payment.total),
            "paymentMethod": self.api_medio,
            "currency": payment.currency,
        }

        if payment.billing_email:
            datos_para_flow.update({"email": payment.billing_email})

        datos_para_flow.update(**self._extra_data(payment.attrs))

        try:
            payment.attrs.datos_payment_create_flow = datos_para_flow
            payment.save()
        except Exception as e:  # noqa
            # Dificil llegar acá, y si llegamos es problema de django-payments
            raise PaymentError(f"Ocurrió un error al guardar attrs.datos_flow: {e}")  # noqa
        datos_para_flow = dict(sorted(datos_para_flow.items()))
        firma_datos = ClienteAPI.genera_firma(datos_para_flow, self.api_secret)
        datos_para_flow.update({"s": firma_datos})

        try:
            pago_req = requests.post(f"{self.api_endpoint}/payment/create", data=datos_para_flow, timeout=5)
            pago_req.raise_for_status()

        except Exception as pe:
            payment.change_status(PaymentStatus.ERROR, str(pe))
            raise PaymentError(pe)
        else:
            pago = pago_req.json()
            payment.transaction_id = pago["token"]
            payment.attrs.respuesta_flow = {
                "url": pago["url"],
                "token": pago["token"],
                "flowOrder": pago["flowOrder"],
            }
            payment.save()
            payment.change_status(PaymentStatus.WAITING)

        raise RedirectNeeded(f"{pago['url']}?token={pago['token']}")

process_data(payment, request)

Procesa los datos del pago recibidos desde Flow.

Parameters:

Name Type Description Default
payment Payment

Objeto de pago Django Payments.

required
request HttpRequest

Objeto de solicitud HTTP de Django.

required

Returns:

Name Type Description
JsonResponse JsonResponse

Respuesta JSON que indica el procesamiento de los datos del pago.

Source code in django_payments_chile/FlowProvider.py
def process_data(self, payment, request) -> JsonResponse:
    """
    Procesa los datos del pago recibidos desde Flow.

    Args:
        payment ("Payment"): Objeto de pago Django Payments.
        request ("HttpRequest"): Objeto de solicitud HTTP de Django.

    Returns:
        JsonResponse: Respuesta JSON que indica el procesamiento de los datos del pago.

    """
    if "token" not in request.POST:
        return HttpResponseBadRequest("token no está en post")

    if payment.status in [PaymentStatus.WAITING, PaymentStatus.PREAUTH]:
        self.actualiza_estado(payment=payment)

    return JsonResponse({"status": "ok"})

refund(payment, amount=None)

Realiza un reembolso del pago. El seguimiendo se debe hacer directamente en Flow

Parameters:

Name Type Description Default
payment Payment

Objeto de pago Django Payments.

required
amount int | None

Monto a reembolsar (opcional).

None

Returns:

Name Type Description
int int

Monto de reembolso solicitado.

Raises:

Type Description
PaymentError

Error al crear el reembolso.

Source code in django_payments_chile/FlowProvider.py
def refund(self, payment, amount: Optional[int] = None) -> int:
    """
    Realiza un reembolso del pago.
    El seguimiendo se debe hacer directamente en Flow

    Args:
        payment ("Payment"): Objeto de pago Django Payments.
        amount (int | None): Monto a reembolsar (opcional).

    Returns:
        int: Monto de reembolso solicitado.

    Raises:
        PaymentError: Error al crear el reembolso.

    """
    if payment.status != PaymentStatus.CONFIRMED:
        raise PaymentError("El pago debe estar confirmado para reversarse.")

    to_refund = amount or payment.total

    datos_reembolso = {
        "apiKey": self.api_key,
        "refundCommerceOrder": payment.token,
        "receiverEmail": payment.billing_email,
        "amount": to_refund,
        "urlCallBack": payment.get_process_url(),
        "commerceTrxId": payment.token,
        "flowTrxId": payment.attrs.respuesta_flow["flowOrder"],
    }
    try:
        refun_req = requests.post(f"{self.api_endpoint}/refund/create", data=datos_reembolso, timeout=5)
        refun_req.raise_for_status()
    except Exception as pe:
        raise PaymentError(pe)
    else:
        payment.attrs.solicitud_reembolso = refun_req.json()
        payment.save()
        payment.change_status(PaymentStatus.REFUNDED)
        return to_refund