Billing gateways¶
Adding a new gateway to fleio¶
The following gateways already exist in Fleio: Paypal, Stripe, Romcard and PayU. Those are located in /var/webapps/fleio/
project/fleio/billing/gateways/
(location is available after running fleio bash command) and you can find some inspiration
from there when adding your desired gateway. To configure an existing gateway read the gateways configuration
documentation
A gateway template exists here https://github.com/fleio/gateway-template, however the following instructions show information about each step required to create a new gateway template. Also, please note that each gateway has its own implementation and the following information might not be all that needs to be done so that the new gateway to work properly. For implementing a new gateway we highly recommend to talk to a python developer.
First, we will have to change directory to /home/fleio/compose and create a new directory where we will place our custom gateway files:
cd /home/fleio/compose
mkdir custombackend
cd custombackend
git clone https://github.com/fleio/gateway-template.git
Create the following Dockerfile related files in /home/fleio/compose/custombackend/ with the code from below:
Important
Depending on edition, you might need to add _web
or _openstack
on the 4th line of each Dockerfile
e.g. FROM ${FLEIO_DOCKER_HUB}/fleio_backend${FLEIO_RELEASE_SUFFIX}
will be FROM ${FLEIO_DOCKER_HUB}/fleio_backend_openstack${FLEIO_RELEASE_SUFFIX}
or
FROM ${FLEIO_DOCKER_HUB}/fleio_backend_web${FLEIO_RELEASE_SUFFIX}
Dockerfile-backend
ARG FLEIO_DOCKER_HUB
ARG FLEIO_RELEASE_SUFFIX
FROM ${FLEIO_DOCKER_HUB}/fleio_backend${FLEIO_RELEASE_SUFFIX}
ENV INSTALL_PATH="/var/webapps/fleio"
COPY --chown=fleio:fleio gateway-template "$INSTALL_PATH/project/fleio/billing/gateways/"
Dockerfile-updated
ARG FLEIO_DOCKER_HUB
ARG FLEIO_RELEASE_SUFFIX
FROM ${FLEIO_DOCKER_HUB}/fleio_updated${FLEIO_RELEASE_SUFFIX}
ENV INSTALL_PATH="/var/webapps/fleio"
COPY --chown=fleio:fleio gateway-template "$INSTALL_PATH/project/fleio/billing/gateways/"
Dockerfile-celery
ARG FLEIO_DOCKER_HUB
ARG FLEIO_RELEASE_SUFFIX
FROM ${FLEIO_DOCKER_HUB}/fleio_celery${FLEIO_RELEASE_SUFFIX}
ENV INSTALL_PATH="/var/webapps/fleio"
COPY --chown=fleio:fleio gateway-template "$INSTALL_PATH/project/fleio/billing/gateways/"
Dockerfile-celerybeat
ARG FLEIO_DOCKER_HUB
ARG FLEIO_RELEASE_SUFFIX
FROM ${FLEIO_DOCKER_HUB}/fleio_celerybeat${FLEIO_RELEASE_SUFFIX}
ENV INSTALL_PATH="/var/webapps/fleio"
COPY --chown=fleio:fleio gateway-template "$INSTALL_PATH/project/fleio/billing/gateways/"
Dockerfile-operations
ARG FLEIO_DOCKER_HUB
ARG FLEIO_RELEASE_SUFFIX
FROM ${FLEIO_DOCKER_HUB}/fleio_operations${FLEIO_RELEASE_SUFFIX}
ENV INSTALL_PATH="/var/webapps/fleio"
COPY --chown=fleio:fleio gateway-template "$INSTALL_PATH/project/fleio/billing/gateways/"
This will add the newgateway folder (with its content) in /var/webapps/fleio/project/fleio/billing/gateways/ to all needed images which will be built in the next step.
Add this to your /home/fleio/compose/docker-compose.override.yml file:
# Add here your Docker Compose customizations
# docker-compose.override.yml is not overwritten by Fleio
# (while docker-compose.yml may be OVERWRITTEN on Fleio upgrades)
version: "3.7"
services:
backend:
build:
context: ./customgateway/
dockerfile: Dockerfile-backend
args:
- FLEIO_DOCKER_HUB
- FLEIO_RELEASE_SUFFIX
image: fleio_backend_custom
pull_policy: never
updated:
build:
context: ./customgateway/
dockerfile: Dockerfile-updated
args:
- FLEIO_DOCKER_HUB
- FLEIO_RELEASE_SUFFIX
image: fleio_updated_custom
pull_policy: never
celery:
build:
context: ./customgateway/
dockerfile: Dockerfile-celery
args:
- FLEIO_DOCKER_HUB
- FLEIO_RELEASE_SUFFIX
image: fleio_celery_custom
pull_policy: never
celerybeat:
build:
context: ./customgateway/
dockerfile: Dockerfile-celerybeat
args:
- FLEIO_DOCKER_HUB
- FLEIO_RELEASE_SUFFIX
image: fleio_celerybeat_custom
pull_policy: never
operations:
build:
context: ./customgateway/
dockerfile: Dockerfile-operations
args:
- FLEIO_DOCKER_HUB
- FLEIO_RELEASE_SUFFIX
image: fleio_operations_custom
pull_policy: never
At this point, you should have the following directory structure:
compose
├── customgateway
│ ├── Dockerfile-backend
│ ├── Dockerfile-celery
│ ├── Dockerfile-celerybeat
│ ├── Dockerfile-operations
│ ├── Dockerfile-updated
│ └── gateway-template
│ ├── README.md
│ └── newgateway
│ ├── __init__.py
│ ├── apps.py
│ ├── conf.py
│ ├── models.py
│ └── newgateway.py
├── docker-compose.override.yml
├── docker-compose.yml
├── env_template
└── secrets
If everything looks ok, you can build your new image and make sure that the affected containers are being reloaded and they’re using thew new custom build images:
cd /home/fleio/compose
docker compose build && docker compose up -d
Run docker ps -a command to confirm that the rebuild containers reloaded (uptime should be a few seconds) and they’re using the fleio custom images:
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
c52140d9f894 hub.fleio.com/fleio_web-2023-02:1 "/docker-entrypoint.…" 4 seconds ago Up 3 seconds 80/tcp, 0.0.0.0:80->8080/tcp, :::80->8080/tcp fleio-web-1
fa8228540bbe fleio_backend_custom "/var/webapps/fleio/…" 11 seconds ago Up 6 seconds 8000/tcp fleio-backend-1
a227f0badff4 fleio_updated_custom "/var/webapps/fleio/…" 11 seconds ago Up 6 seconds fleio-updated-1
605eb2e6286b fleio_celerybeat_custom "/var/webapps/fleio/…" 11 seconds ago Up 6 seconds fleio-celerybeat-1
a267943154f1 fleio_celery_custom "/var/webapps/fleio/…" 11 seconds ago Up 6 seconds fleio-celery-1
e2d260de66bf fleio_operations_custom "/var/webapps/fleio/…" 11 seconds ago Up 6 seconds fleio-operations-1
706aeca45b84 hub.fleio.com/fleio_incomingmail-2023-02:1 "/var/webapps/fleio/…" 45 minutes ago Up 24 minutes fleio-incomingmail-1
f18bd6760827 hub.fleio.com/fleio_frontend-2023-02:1 "/docker-entrypoint.…" 45 minutes ago Up 24 minutes 80/tcp fleio-frontend-1
dfea54b07fab hub.fleio.com/fleio_fluentd-2023-02:1 "tini -- /bin/entryp…" 45 minutes ago Up 24 minutes 5140/tcp, 127.0.0.1:24224->24224/tcp, 127.0.0.1:24224->24224/udp fleio-fluentd-1
688fd8203263 redis:7.0.7-alpine "docker-entrypoint.s…" 45 minutes ago Up 24 minutes 6379/tcp fleio-redis-1
c8dc28269128 mariadb:10.9.4-focal "docker-entrypoint.s…" 48 minutes ago Up 24 minutes 3306/tcp fleio-db-1
The next step is to add the newgateway to the django installed apps. Run fleio edit settings.py and add the following line at the end of the file:
INSTALLED_APPS += ('fleio.billing.gateways.newgateway', )
NEWGATEWAY_SETTINGS = {
'secret_key': 'whatever!@#$',
'test_mode': True,
'callback_url': 'https://fleio.com',
}
Save and exit and answer with “y” when asked if you want to restart Fleio. At this point, the gateway should be visible when going to the staff/billing/gateways page in the frontend dashboard panel.
Notes on files present in newgateway directory:¶
apps.py
from django.utils.translation import ugettext_lazy as _
from django.apps import AppConfig
class NewgatewayConfig(AppConfig):
name = "fleio.billing.gateways.newgateway"
verbose_name = _("NewGateway")
fleio_module_type = 'payment_gateway'
module_settings = {
'capabilities': {
'can_process_payments': True,
'returns_fee_information': False
}
}
The returns_fee_information attribute from module_settings specifies if this gateway returns fee information. If so, fees will automatically be added to transactions otherwise they may be added from frontend on the /staff/billing/gateways page.
conf.py
from django.conf import settings
class Conf(object):
def __init__(self):
self.newgateway_settings = getattr(settings, 'NEWGATEWAY_SETTINGS', {})
self.test_mode = self.newgateway_settings.get('test_mode')
self.secret_key = self.newgateway_settings.get('secret_key')
self.callback_url = self.newgateway_settings.get('callback_url')
conf = Conf()
As you can see, the first line in the __init__(self) method gets the gateway settings dictionary from the settings.py file. You defined a dictionary in settings.py earlier, containing all the information and variables required by your gateway.
The lines just retrieve each value from the dictionary and assigns them to variables defined on the Conf class.
The dictionary defined in settings.py may look like this given the above example:
NEWGATEWAY_SETTINGS = {
'secret_key': 'whatever!@#$',
'test_mode': True,
'callback_url': 'https://fleio.com',
}
newgateway.py
from django.conf import settings
from django.http import HttpResponseRedirect
from .conf import conf
from fleio.billing.gateways.decorators import gateway_action, staff_gateway_action
from fleio.billing.gateways import exceptions as gateway_exceptions
from fleio.core.utils import fleio_join_url
def process_test_payment(test_mode):
pass
@gateway_action(methods=['GET'])
def pay_invoice(request):
invoice_id = request.query_params.get('invoice')
try:
process_test_payment(test_mode=conf.test_mode)
except Exception as create_payment_exception:
# handle possible exception
raise gateway_exceptions.InvoicePaymentException(
message=str(create_payment_exception),
invoice_id=invoice_id
)
# redirect to invoice
relative_url = 'billing/invoices/{}'.format(invoice_id)
Note on 4th line (from fleio.billing.gateways.decorators import gateway_action, staff_gateway_action):
These decorators are used for functions that can be done by end-users or staff users. There is also the callback(request) function. This will be used in case you want to implement callbacks from the gateway (see the example from stripe.py file).
An example of pay_invoice function can be this:
from django.conf import settings
from django.http import HttpResponseRedirect
from .conf import conf
from fleio.billing.gateways.decorators import gateway_action, staff_gateway_action
from fleio.billing.gateways import exceptions as gateway_exceptions
from fleio.core.utils import fleio_join_url
def process_test_payment(test_mode):
pass
@gateway_action(methods=['GET'])
def pay_invoice(request):
invoice_id = request.query_params.get('invoice')
try:
process_test_payment(test_mode=conf.test_mode)
except Exception as create_payment_exception:
# handle possible exception
raise gateway_exceptions.InvoicePaymentException(
message=str(create_payment_exception),
invoice_id=invoice_id
)
# redirect to invoice (or maybe render a gateway template)
relative_url = 'billing/invoices/{}'.format(invoice_id)
return HttpResponseRedirect(fleio_join_url(settings.FRONTEND_URL, relative_url))
You will now see that when paying an invoice this code gets executed.
Other decorated methods you can use:
Using the gateway_action decorator: pay_invoice, confirm_payment, cancel_payment,
Using the staff_gateway_action decorator: refund, capture
For rendering custom html pages you may find inspiration in the existing Stripe or Romcard gateways.