Tamara Payment Gateway Integration for Django - Buy Now Pay Later (BNPL) solution supporting Saudi Arabia, UAE, Kuwait, Bahrain, Qatar, and Oman.
Project description
Django Tamara Payment Gateway
A professional Django package for integrating Tamara - the leading Buy Now Pay Later (BNPL) payment solution in the Middle East. Supports Saudi Arabia, UAE, Kuwait, Bahrain, Qatar, and Oman.
Features
- ✅ Full Tamara Online Checkout flow
- ✅ Payment Types lookup
- ✅ Create Checkout Sessions
- ✅ Authorise / Capture / Cancel / Refund orders
- ✅ Webhook management (register, list, update, delete)
- ✅ Sandbox & Production environments
- ✅ Multi-currency support (SAR, AED, KWD, BHD, QAR, OMR)
- ✅ Multi-country support (SA, AE, KW, BH, QA, OM)
- ✅ In-store checkout support
- ✅ Requests HTTP client (no raw cURL)
- ✅ Configurable URL prefix & middleware
- ✅ Transaction logging model
- ✅ Django 4.2, 5.0 & 5.1 compatible
- ✅ Python 3.9+
Requirements
| Django | Python | Package Version |
|---|---|---|
| 4.2.x | 3.9+ | ^1.0 |
| 5.0.x | 3.10+ | ^1.0 |
| 5.1.x | 3.10+ | ^1.0 |
Installation
pip install aghfatehi-django-tamara
Configuration
1. Add to INSTALLED_APPS
# settings.py
INSTALLED_APPS = [
...
'tamara',
]
2. Environment / Settings Variables
# settings.py
# ─── Tamara Payment Gateway Settings ──────────────────────────────
TAMARA_SANDBOX_MODE = True
# bool | Sandbox mode (testing) when True, Production mode when False
TAMARA_API_TOKEN = "your-api-token-here"
# string | Your Tamara API token from Tamara dashboard
TAMARA_COUNTRY_CODE = "SA"
# string (2 chars) | Country code
# SA = Saudi Arabia | AE = UAE | KW = Kuwait | BH = Bahrain | QA = Qatar | OM = Oman
TAMARA_CURRENCY = "SAR"
# string (3 chars) | Currency code
# SAR = Saudi Riyal | AED = UAE Dirham | KWD = Kuwaiti Dinar | BHD = Bahraini Dinar | QAR = Qatari Riyal | OMR = Omani Riyal
TAMARA_INSTALMENTS = 3
# int | Number of instalments (3, 4, or 6)
TAMARA_PAYMENT_TYPE = "PAY_BY_INSTALMENTS"
# string | Payment type - "PAY_BY_INSTALMENTS" or "PAY_NEXT_MONTH" or "SINGLE_PAYMENT"
TAMARA_LOCALE = "en_US"
# string | Locale - "en_US" or "ar_SA"
TAMARA_ROUTE_PREFIX = "tamara"
# string | URL prefix for all Tamara routes
3. Include URLs
# urls.py
from django.urls import include, path
urlpatterns = [
...
path('tamara/', include('tamara.urls')),
]
4. Run Migrations
python manage.py migrate tamara
Usage
Quick Start - Frontend Checkout
from tamara.client import client
# Get available payment types
types = client.get_payment_types('SA', 'SAR', 500)
# Create a checkout session
response = client.create_checkout({
'total_amount': {'amount': 500, 'currency': 'SAR'},
'order_reference_id': 'ORD-' + str(int(__import__('time').time())),
'order_number': 'INV-2024-001',
'consumer': {
'email': 'customer@example.com',
'first_name': 'Ahmed',
'last_name': 'Ali',
'phone_number': '500000000',
},
'country_code': 'SA',
'merchant_url': {
'success': 'https://example.com/tamara/callback',
'failure': 'https://example.com/tamara/failure',
'cancel': 'https://example.com/tamara/cancel',
'notification': 'https://example.com/tamara/webhook',
},
})
# Redirect customer to Tamara checkout
if 'checkout_url' in response:
# return redirect(response['checkout_url'])
print(f"Redirect to: {response['checkout_url']}")
Using Routes
The package registers these routes under the configured prefix (/tamara by default):
| Method | URI | Name | Description |
|---|---|---|---|
| GET | /tamara/payment/types |
tamara:payment_types |
Get eligible payment types |
| POST | /tamara/pay |
tamara:pay |
Initiate checkout |
| ANY | /tamara/callback |
tamara:callback |
Payment callback |
| GET | /tamara/cancel |
tamara:cancel |
Cancel handler |
| GET | /tamara/failure |
tamara:failure |
Failure handler |
| POST | /tamara/webhook |
tamara:webhook |
Webhook receiver |
| POST | /tamara/authorise |
tamara:authorise |
Authorise order |
API Methods
from tamara.client import client
# Payment Types
types = client.get_payment_types('SA', 'SAR', 500, '500000000')
# Checkout
checkout = client.create_checkout(data)
# Order Management
order = client.get_order('order-id-here')
order = client.get_order_by_reference_id('ref-id-here')
# Authorise / Capture / Cancel / Refund
authorised = client.authorise_order('order-id')
captured = client.capture_order('order-id', {'amount': 500})
cancelled = client.cancel_order('order-id', {'reason': 'out_of_stock'})
refunded = client.refund_order('order-id', 500, 'SAR', 'Refund comment')
# Webhook Management
webhook = client.webhook_register(
'https://example.com/webhook',
['order_approved', 'order_declined', 'order_authorised', 'order_captured', 'order_refunded'],
)
webhook_list = client.webhook_list()
webhook_detail = client.webhook_get('webhook-id')
client.webhook_delete('webhook-id')
client.webhook_update('webhook-id', 'https://example.com/webhook', ['order_approved'])
Handling Webhooks
The webhook endpoint logs all incoming events. Extend the view or listen to the log to implement your business logic:
# views.py (custom webhook handler)
import json
import logging
from django.http import JsonResponse
from django.views import View
from django.utils.decorators import method_decorator
from django.views.decorators.csrf import csrf_exempt
logger = logging.getLogger(__name__)
@method_decorator(csrf_exempt, name='dispatch')
class CustomWebhookView(View):
def post(self, request):
payload = json.loads(request.body)
event = payload.get('event_type')
order_id = payload.get('order_id')
status = payload.get('status')
if event == 'order_approved':
# Mark order as approved
pass
elif event == 'order_captured':
# Fulfill the order
pass
elif event == 'order_refunded':
# Process refund
pass
return JsonResponse({'success': True})
Development
# Install dev dependencies
pip install -e ".[dev]"
# Run tests
pytest
# Lint
ruff check src/ tests/
Changelog
See CHANGELOG for recent changes.
Security
If you discover security issues, please email fathi.a.n2002@gmail.com instead of using the issue tracker.
License
This package is open-sourced software licensed under the MIT license.
Support
- Issues: GitHub Issues
- Tamara Docs: https://docs.tamara.co
- Author: AL-AGHBARI Fatehi (fathi.a.n2002@gmail.com)
Countries & Currencies
| Country | Code | Currency | Code |
|---|---|---|---|
| Saudi Arabia | SA | Riyal | SAR |
| UAE | AE | Dirham | AED |
| Kuwait | KW | Dinar | KWD |
| Bahrain | BH | Dinar | BHD |
| Qatar | QA | Riyal | QAR |
| Oman | OM | Riyal | OMR |
| "# django-tamara" |
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file aghfatehi_django_tamara-1.0.0.tar.gz.
File metadata
- Download URL: aghfatehi_django_tamara-1.0.0.tar.gz
- Upload date:
- Size: 11.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
2405e1937cb6b03809325c85e10606a94c4be3a89daf6b8231a07920532731a4
|
|
| MD5 |
48d4e4f498eddf66c9b8ac856157dce3
|
|
| BLAKE2b-256 |
492cd3d0cb3243d185ba7965a90a65f1942808381b1bc8dd6b3637fcd9a32eb0
|
File details
Details for the file aghfatehi_django_tamara-1.0.0-py3-none-any.whl.
File metadata
- Download URL: aghfatehi_django_tamara-1.0.0-py3-none-any.whl
- Upload date:
- Size: 12.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
38beb18ca5e628318d072df52f167a092564e5303e2a6e480cdce59374f15f93
|
|
| MD5 |
8a7ed3b0206c0a0ea83f0ea39b207747
|
|
| BLAKE2b-256 |
2896f574102ce7647302703241d828b057c71c6d760b67fc1c5e61d094c4aa2c
|