🚀 TemanQRIS API Documentation
RESTful API untuk mengkonversi QRIS statis menjadi QRIS dinamis dengan nominal custom, fee, dan payment link yang bisa di-share.
https://temanqris.com/api/qris
Terima notifikasi real-time saat status pembayaran berubah. Lihat dokumentasi →
Authentication
Semua API endpoint memerlukan autentikasi. Gunakan API Key yang bisa Anda dapatkan dari Dashboard Settings.
API Key
Gunakan API Key untuk semua request ke TemanQRIS API.
X-API-Key: YOUR_API_KEY
Jangan pernah expose API Key di frontend atau commit ke repository. Gunakan environment variables.
Quick Start
Tiga langkah untuk mulai menggunakan TemanQRIS API:
Upload QRIS Statis
Upload gambar atau string QRIS statis Anda (sekali saja)
Generate QRIS Dinamis
Buat QRIS dengan nominal kapan saja
Share atau Embed
Gunakan payment link atau embed widget
Rate Limits
Batas request harian berdasarkan tier akun:
| Tier | Daily Limit | Stored QRIS | Features |
|---|---|---|---|
| Free | 50 requests/day | 1 QRIS | Basic features |
| Premium | 1000 requests/day | Unlimited* | All features + Priority support |
GET /api/qris/usage untuk monitoring penggunaan API real-time.Upload Static QRIS
Upload QRIS statis Anda. Ini hanya perlu dilakukan SEKALI saja.
Penting!
Upload QRIS statis hanya diperlukan sekali saat pertama kali setup.
Setelah berhasil upload, Anda bisa langsung menggunakan endpoint /generate
untuk membuat QRIS dinamis tanpa perlu upload ulang. Gunakan endpoint /my-qris
untuk mengecek apakah Anda sudah pernah upload QRIS.
Option 1: Upload Image
Upload file gambar QRIS (jpg, png, webp)
# Upload image file
curl -X POST https://temanqris.com/api/qris/upload \
-H "X-API-Key: YOUR_API_KEY" \
-F "qris_image=@/path/to/qris.png"
Option 2: Upload String
Jika sudah punya QRIS string (hasil decode)
# Upload QRIS string
curl -X POST https://temanqris.com/api/qris/upload \
-H "X-API-Key: YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"qris_string": "00020101021126670016COM.NOBUBANK.WWW..."
}'
Response
200 OK{
"message": "QRIS uploaded successfully",
"qris_string": "00020101021126...",
"merchant": {
"name": "TOKO EXAMPLE",
"city": "JAKARTA"
}
}
Check QRIS Status
Cek apakah Anda sudah pernah upload QRIS statis. Gunakan sebelum memanggil /upload untuk menghindari upload yang tidak perlu.
Request
# Check QRIS status
curl -X GET https://temanqris.com/api/qris/my-qris \
-H "X-API-Key: YOUR_API_KEY"
Response (QRIS exists)
200 OK{
"success": true,
"has_qris": true,
"qris": {
"id": 123,
"qris_string": "00020101021126...",
"merchant_name": "TOKO EXAMPLE",
"merchant_city": "JAKARTA",
"created_at": "2024-01-15T10:30:00Z",
"updated_at": "2024-01-15T10:30:00Z"
}
}
Response (No QRIS yet)
200 OK{
"success": true,
"has_qris": false,
"message": "No QRIS found. Please upload your static QRIS first (only needed once)."
}
Generate Dynamic QRIS
Endpoint utama untuk membuat QRIS dinamis dengan nominal transaksi.
Request Body
| Field | Type | Description |
|---|---|---|
amount |
number required | Nominal transaksi (contoh: 50000) |
fee_type |
string optional | "rupiah" atau "percent" |
fee_value |
number optional | Nilai fee yang ditambahkan |
qris_id |
number optional | ID QRIS spesifik (jika punya multiple) |
order_id |
string optional | ID unik order Anda (max 30 chars). Auto-generate jika kosong. |
webhook_url |
string optional | URL untuk menerima notifikasi webhook saat status berubah. |
callback_url |
string optional | URL redirect customer setelah klik "Sudah Bayar". |
Example Request
# Generate QRIS Rp 50.000 + fee Rp 1.000 with webhook
curl -X POST https://temanqris.com/api/qris/generate \
-H "X-API-Key: YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"amount": 50000,
"fee_type": "rupiah",
"fee_value": 1000,
"order_id": "INV-2025-001",
"webhook_url": "https://yoursite.com/webhook"
}'
Response
200 OK{
"success": true,
"qris": "00020101021226...",
"qr_image": "data:image/png;base64,...",
"amount": 50000,
"fee": { "type": "rupiah", "value": 1000 },
"expires_at": "2025-12-25T10:00:00.000Z",
"payment_link": {
"id": 456,
"link_code": "BTN3EV2X",
"order_id": "INV-2025-001",
"url": "/p/BTN3EV2X",
"amount": 50000,
"merchant_name": "TOKO EXAMPLE",
"webhook_url": "https://yoursite.com/webhook",
"expires_at": "2025-12-26T10:00:00.000Z"
}
}
qr_image berisi Base64 image yang bisa langsung ditampilkan dengan
<img src="...">
Render QR Image
Generate gambar QR code dari string QRIS.
Provide either qris_string or qris_data_id.
Option 1: Using String
curl -X POST https://temanqris.com/api/qris/render \
-H "X-API-Key: YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"qris_string": "00020101021126..."
}'
Option 2: Using Saved ID
Jika sudah punya QRIS yang tersimpan (dari /my-qris), cukup kirim ID-nya.
curl -X POST https://temanqris.com/api/qris/render \
-H "X-API-Key: YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"qris_data_id": 123
}'
Response
{
"success": true,
"qr_image": "..."
}
Create Payment Link
Buat link pembayaran yang bisa di-share ke pelanggan.
Request Body
| Field | Type | Description |
|---|---|---|
amount |
number required | Nominal pembayaran |
description |
string optional | Deskripsi pembayaran (e.g. "Order #123") |
order_id |
string optional | ID unik order Anda (max 30 chars). Auto-generate jika kosong. |
webhook_url |
string optional | URL untuk menerima notifikasi webhook saat status berubah. |
callback_url |
string optional | URL untuk redirect customer setelah klik "Sudah Bayar". |
curl -X POST https://temanqris.com/api/qris/payment-link \
-H "X-API-Key: YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"amount": 75000,
"description": "Pembayaran Order #12345",
"order_id": "INV-2025-001",
"webhook_url": "https://yoursite.com/webhook",
"callback_url": "https://yoursite.com/thank-you"
}'
Response
{
"success": true,
"payment_link": {
"id": 456,
"link_code": "ABC12345",
"order_id": "INV-2025-001",
"url": "/p/ABC12345",
"amount": 75000,
"description": "Pembayaran Order #12345",
"webhook_url": "https://yoursite.com/webhook",
"callback_url": "https://yoursite.com/thank-you",
"expires_at": "2025-12-26T14:00:00.000Z"
}
}
https://temanqris.com/p/ABC12345 - Share ke pelanggan via
WhatsApp, email, dll.List Payment Links
Ambil daftar semua payment link Anda.
curl https://temanqris.com/api/qris/payment-links \
-H "X-API-Key: YOUR_API_KEY"
Response
{
"payment_links": [
{
"id": 456,
"link_code": "ABC12345",
"url": "/p/ABC12345",
"amount": 75000,
"description": "Order #12345",
"merchant_name": "TOKO EXAMPLE",
"is_active": true,
"view_count": 5,
"expires_at": "2025-12-31T23:59:59Z",
"created_at": "2025-12-24T10:00:00.000Z",
"is_expired": false
}
]
}
Check Order Status
Cek status pembayaran order/payment link via API. Berguna untuk integrasi backend-to-backend.
Request
# Check specific order by order_id
curl -X GET https://temanqris.com/api/qris/orders/ORD-ABC123 \
-H "X-API-Key: YOUR_API_KEY"
Response
200 OKOrder Status Definitions
| Status | Description |
|---|---|
| pending | Menunggu pembayaran dari customer. |
| awaiting_confirmation | Pembayaran dikonfirmasi customer (widget), menunggu verifikasi. |
| paid | Pembayaran berhasil dan terverifikasi. |
| expired | Batas waktu pembayaran telah habis. |
| cancelled | Order dibatalkan. |
{
"success": true,
"order": {
"order_id": "ORD-ABC123",
"title": "Pembayaran Invoice #001",
"amount": 50000,
"total_amount": 51000,
"status": "paid",
"is_paid": true,
"is_expired": false,
"created_at": "2024-01-15T10:30:00Z",
"paid_at": "2024-01-15T10:35:00Z"
}
}
Query Parameters
| Parameter | Type | Description |
|---|---|---|
status |
string | Filter by status: pending, paid, expired, cancelled |
limit |
number | Max results (default: 20) |
offset |
number | Offset for pagination |
Request
# Get all paid orders
curl -X GET "https://temanqris.com/api/qris/orders?status=paid&limit=10" \
-H "X-API-Key: YOUR_API_KEY"
Response
200 OK{
"success": true,
"orders": [
{
"order_id": "ORD-ABC123",
"title": "Invoice #001",
"amount": 50000,
"status": "paid",
"is_paid": true
}
],
"pagination": {
"total": 25,
"limit": 10,
"offset": 0
}
}
Verify Order (Merchant)
Verifikasi pembayaran order via API. Merchant bisa langsung mark order sebagai paid tanpa
perlu buka dashboard.
awaiting_confirmation), merchant bisa verifikasi via API dan otomatis update ke
paid.
URL Parameter
| Parameter | Type | Description |
|---|---|---|
orderId |
string | Required. Order ID Anda (e.g., INV-2025-001) |
Request Body (optional)
| Field | Type | Description |
|---|---|---|
payer_name |
string optional | Nama pembayar |
payer_note |
string optional | Catatan pembayaran |
Request
# Verify order as paid
curl -X POST https://temanqris.com/api/qris/orders/INV-2025-001/verify \
-H "X-API-Key: YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"payer_name": "John Doe",
"payer_note": "Pembayaran via OVO"
}'
Response
200 OK{
"success": true,
"message": "Payment verified successfully",
"order": {
"order_id": "INV-2025-001",
"amount": 75000,
"status": "paid",
"paid_at": "2025-12-27T16:30:00.000Z",
"confirmed_by": "api"
}
}
Error Responses
404 Not Found{
"success": false,
"error": "Order not found",
"message": "No order found with order_id \"INV-999\" for your account."
}
400 Bad Request
{
"success": false,
"error": "Order already verified",
"message": "This order has already been marked as paid."
}
Integration Flow
- Buat payment link via
POST /api/qris/payment-linkatauPOST /api/qris/generate - Share link ke customer
- Customer scan QRIS dan bayar
- Customer klik "Sudah Bayar" → status =
awaiting_confirmation - Terima webhook
payment.awaiting_confirmation🔔 - Cek status via
GET /api/qris/orders/:orderId - Verifikasi dana masuk di e-wallet/rekening
- Call
POST /api/qris/orders/:orderId/verify→ status =paid✅
Cancel Order
Batalkan order/payment link yang belum dibayar.
Request
# Cancel an order
curl -X POST https://temanqris.com/api/qris/orders/INV-2025-001/cancel \
-H "X-API-Key: YOUR_API_KEY"
Response
200 OK{
"success": true,
"message": "Order cancelled successfully",
"order": {
"order_id": "INV-2025-001",
"status": "cancelled"
}
}
Error Responses
400 Bad Request{
"success": false,
"error": "Cannot cancel paid order",
"message": "This order has already been verified as paid and cannot be cancelled."
}
Delete Payment Link
Hapus/nonaktifkan payment link yang sudah tidak digunakan.
curl -X DELETE https://temanqris.com/api/qris/payment-link/456 \
-H "X-API-Key: YOUR_API_KEY"
Response
{
"message": "Payment link deleted successfully"
}
Webhooks
Terima notifikasi real-time saat status pembayaran berubah. Webhook dikirim ke URL yang Anda tentukan saat membuat payment link.
Event Types
| Event | Trigger | Description |
|---|---|---|
payment.awaiting_confirmation |
Customer klik "Sudah Bayar" | Customer mengklaim sudah bayar, menunggu verifikasi Anda |
payment.confirmed |
Merchant approve di Dashboard atau via API | Pembayaran sudah diverifikasi dan dikonfirmasi (via
POST /orders/:orderId/verify atau Dashboard)
|
Webhook Payload
{
"event": "payment.awaiting_confirmation",
"timestamp": "2025-12-27T16:00:00.000Z",
"data": {
"order_id": "INV-2025-001",
"link_code": "ABC12345",
"amount": 75000,
"description": "Pembayaran Order #12345",
"status": "awaiting_confirmation",
"paid_at": null,
"created_at": "2025-12-27T15:55:00.000Z"
}
}
Webhook Headers
| Header | Description |
|---|---|
X-TemanQRIS-Signature |
HMAC-SHA256 signature untuk verifikasi (format: sha256=...) |
X-TemanQRIS-Event |
Nama event (e.g. payment.awaiting_confirmation) |
X-TemanQRIS-Retry |
Nomor retry (jika sebelumnya gagal) |
Verifikasi Signature (Node.js)
const crypto = require('crypto');
function verifyWebhook(payload, signature, secret) {
const expected = 'sha256=' + crypto
.createHmac('sha256', secret)
.update(JSON.stringify(payload))
.digest('hex');
return crypto.timingSafeEqual(
Buffer.from(signature),
Buffer.from(expected)
);
}
// Contoh penggunaan di Express
app.post('/webhook', (req, res) => {
const signature = req.headers['x-temanqris-signature'];
const isValid = verifyWebhook(req.body, signature, 'YOUR_WEBHOOK_SECRET');
if (!isValid) {
return res.status(401).json({ error: 'Invalid signature' });
}
// Process webhook...
if (req.body.event === 'payment.confirmed') {
// Update order status di database Anda
}
res.json({ received: true });
});
Customer Sudah Bayar (Public)
API publik untuk mencatat bahwa customer mengklaim sudah membayar. Status transaksi akan
berubah menjadi awaiting_confirmation dan merchant perlu memverifikasi di dashboard.
Endpoint ini hanya mengubah status ke
awaiting_confirmation. Merchant tetap harus
verifikasi manual bahwa dana sudah masuk di e-wallet/rekening mereka.
🔓 Public API - Tidak Perlu API Key
Endpoint ini dipanggil dari frontend saat customer klik tombol "Sudah Bayar" di halaman pembayaran.
URL Parameter
| Parameter | Type | Description |
|---|---|---|
link_code |
string | Required. Kode payment link (e.g., ABC12345) |
Request
# Customer confirms payment via link code
curl -X POST https://temanqris.com/api/pay/ABC12345/confirm \
-H "Content-Type: application/json"
JavaScript Example
// Call from payment page after customer finishes paying
const linkCode = 'ABC12345';
fetch(`/api/pay/${linkCode}/confirm`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(data => {
if (data.success) {
alert('Konfirmasi pembayaran berhasil!');
}
});
Response
200 OK{
"success": true,
"message": "Konfirmasi pembayaran berhasil dikirim",
"order_id": "ORD-ABC12345XYZ",
"status": "awaiting_confirmation"
}
Flow Explanation
- Customer membuka halaman pembayaran (
/p/ABC12345) - Customer scan QRIS dan melakukan pembayaran di e-wallet
- Customer klik tombol "Sudah Bayar"
- Frontend memanggil
POST /api/pay/ABC12345/confirm - Status berubah menjadi
awaiting_confirmation(belum confirmed!) - 🔔 Webhook dikirim ke URL yang di-configure (jika ada)
- 📩 Merchant menerima Email Notifikasi
- 🔗 Customer di-redirect ke callback_url (jika ada)
- Merchant verifikasi manual di dashboard atau e-wallet
- Setelah verifikasi, merchant approve via API atau Dashboard
→ status =
paid
Error Response
404 Not Found{
"error": "Link pembayaran tidak ditemukan"
}
API Usage Statistics
Monitor penggunaan API dan sisa limit Anda.
curl https://temanqris.com/api/qris/usage \
-H "X-API-Key: YOUR_API_KEY"
Response
{
"tier": "free",
"daily_limit": 100,
"usage": {
"today": {
"total": 15,
"remaining": 85,
"by_endpoint": [
{ "endpoint": "/qris/generate", "request_count": 10 },
{ "endpoint": "/qris/payment-link", "request_count": 5 }
]
},
"this_month": {
"total": 250,
"by_endpoint": [...]
}
}
}
Embed Widget (widget.js)
Pasang tombol pembayaran QRIS di website Anda dengan satu baris kode. Widget akan menampilkan modal pembayaran yang cantik dan responsif.
Installation
Tambahkan kode ini di halaman website Anda:
<script
src="https://temanqris.com/widget.js"
data-merchant="YOUR_MERCHANT_ID"
></script>
💡 Mendapatkan Merchant ID: Login ke Dashboard → Widget untuk melihat Merchant ID Anda.
Result
Widget akan otomatis menampilkan tombol seperti ini:
Preview tombol widget
Widget Options
Kustomisasi widget sesuai kebutuhan Anda dengan data attributes:
| Attribute | Type | Description |
|---|---|---|
data-merchant |
string * | Merchant ID Anda (wajib) |
data-amount |
number | Nominal tetap (jika tidak diisi, user bisa input sendiri) |
data-button-text |
string | Teks tombol (default: "Bayar dengan QRIS") |
data-button-color |
string | Warna tombol dalam hex (default: "#ff6b35") |
data-description |
string | Deskripsi pembayaran (opsional) |
data-callback |
string (URL) | Callback URL - Redirect browser customer setelah klik "Sudah Bayar". Query
params: order_id, amount, status |
data-webhook |
string (URL) | 🔔 Webhook URL - Notifikasi server-to-server saat status berubah (untuk update database) |
Full Example with Callback & Webhook
<script
src="https://temanqris.com/widget.js"
data-merchant="abc123xyz"
data-amount="50000"
data-button-text="Bayar Sekarang"
data-button-color="#22c55e"
data-description="Pembayaran Order #123"
data-callback="https://yoursite.com/thank-you"
data-webhook="https://yoursite.com/webhook"
></script>
🔗 Callback URL (data-callback): Customer di-redirect ke URL
setelah klik "Sudah Bayar":
https://yoursite.com/thank-you?order_id=xxx&amount=50000&status=confirmed
🔔 Webhook URL (data-webhook): Server TemanQRIS akan mengirim
POST request ke URL Anda dengan payload JSON:
{ "event": "payment.awaiting_confirmation", "order_id": "xxx", ... }
Lihat dokumentasi webhook untuk verifikasi signature.
⚠️ Catatan Keamanan: Jangan tampilkan Merchant ID di repositori publik. Widget ini aman digunakan di frontend karena pembayaran diproses via server TemanQRIS.