Introduction
Welcome to the Afrinvoice API documentation. This guide provides all required information for developers integrating with our e-Invoicing platform, including authentication, invoice validation, IRN generation, QR generation, and response formats.
The API follows RESTful principles and accepts and returns data in JSON format. All requests must be authenticated using your Access Key and Access Secret obtained from the authentication endpoint.
Main Capabilities
- Generate IRN (client-side pattern)
- Validate Invoice Payload
- Generate encrypted QR Code (FIRS standard)
- Download & Confirm Invoices
Authentication
Authenticate using email & password to receive access_token and access_secret.
LIVEPOST /api/login
// Login endpoint POST /api/login Content-Type: application/json Authenticate with email + password and receive a token pair used in all requests.
{
"email": "user@example.com",
"password": "yourpassword"
}
{
"status": "success",
"access_token": "YXNkZjEyMzQtYWJjZC01Njc4LWpoazAtc2VjcmV0dG9rZW4=",
"access_secret": "c2VjcmV0LXZhbHVlLWFzZGZhc2RmLWhhbmRsZW1l"
}
| Field | Required | Description |
|---|---|---|
| Yes | User login email | |
| password | Yes | User login password |
curl -X POST "https://your-backend.example.com/api/login" \
-H "Content-Type: application/json" \
-d '{
"email": "user@example.com",
"password": "yourpassword"
}'
$payload = [ "email" => "user@example.com", "password" => "yourpassword" ]; $curl = curl_init(); curl_setopt_array($curl, [ CURLOPT_URL => "https://your-backend.example.com/api/login", CURLOPT_RETURNTRANSFER => true, CURLOPT_POST => true, CURLOPT_HTTPHEADER => ["Content-Type: application/json"], CURLOPT_POSTFIELDS => json_encode($payload) ]); $response = curl_exec($curl); curl_close($curl); echo $response;
import axios from "axios";
axios.post("https://your-backend.example.com/api/login", {
email: "user@example.com",
password: "yourpassword"
})
.then(res => console.log(res.data))
.catch(err => console.error(err));
import requests
payload = {
"email": "user@example.com",
"password": "yourpassword"
}
res = requests.post(
"https://your-backend.example.com/api/login",
json=payload
)
print(res.json())
System Integrator API Endpoints
Generate IRN
Client-side IRN formation pattern (server only receives the resulting IRN).
LOCALClient: generate IRN string
// IRN format (client-side)
{invoiceNumber}-{FIRS_SERVICE_ID}-{YYYYMMDD}
INV001-SERVICEID-20251005
| Field | Required | Description |
|---|---|---|
| invoiceNumber | Yes | Alphanumeric only (no spaces/special chars) |
| FIRS_SERVICE_ID | Yes | Your assigned service id from FIRS |
| invoiceDate | Yes | YYYY-MM-DD (used to produce YYYYMMDD) |
// Client should produce IRN and send that string to backend endpoints below. // (No server-side call required for generation itself)
Validate Invoice
Validate invoice payload with FIRS via your backend proxy.
POST/api/validate-invoice
{
"business_id": "6389237e-4f6e-4371-aece-aea5d2536df3",
"irn": "INV001-SERVICEID-20251005",
"issue_date": "2025-10-05",
"due_date": "2025-10-05",
"issue_time": "13:45:00",
"invoice_type_code": "396",
"payment_status": "PENDING",
"note": "Optional notes",
"tax_point_date": "2025-10-05",
"document_currency_code": "NGN",
"tax_currency_code": "NGN",
"accounting_cost": "100232 NGN",
"invoice_delivery_period": {
"start_date": "2025-10-05",
"end_date": "2025-10-25"
},
"order_reference": "",
"accounting_supplier_party": {
"party_name": "Supplier Ltd",
"tin": "32492774-0001",
"email": "developer@afrinvoice.com",
"telephone": "+2348026664444",
"postal_address": {
"street_name": "Supplier address",
"city_name": "ANTHONY",
"postal_zone": "100232",
"lga": "70",
"state": "4",
"country": "NG"
}
},
"accounting_customer_party": {
"party_name": "Customer Inc",
"tin": "33577312-0001",
"email": "mark@markodenore.com",
"telephone": "+2348026665555",
"postal_address": {
"street_name": "Customer address",
"city_name": "ANTHONY",
"postal_zone": "100232",
"lga": "70",
"state": "4",
"country": "NG"
}
},
"tax_total": [
{
"tax_amount": 250,
"tax_subtotal": [
{
"taxable_amount": 2500,
"tax_amount": 250,
"tax_category": {
"id": "LOCAL_SALES_TAX",
"percent": 10
}
}
]
}
],
"legal_monetary_total": {
"line_extension_amount": 2500,
"tax_exclusive_amount": 2500,
"tax_inclusive_amount": 2750,
"payable_amount": 2750
},
"invoice_line": [
{
"hsn_code": "CC-001",
"product_category": "Consulting Services",
"discount_rate": 0,
"fee_rate": 10,
"invoiced_quantity": 1,
"line_extension_amount": 2500,
"item": {
"name": "Service A",
"description": "Consulting",
"sellers_item_identification": "item-1"
},
"price": {
"price_amount": 2500,
"base_quantity": 1,
"price_unit": "NGN per 1"
}
}
]
}
{
"message": "FIRS validation successful",
"data": {
"code": 200,
"data": {
"ok": true
}
}
}
| Header | Value / Notes |
|---|---|
| Content-Type | application/json |
| x-api-key | Backend injects when proxying to FIRS (do not expose) |
| x-api-secret | Backend injects when proxying to FIRS (do not expose) |
Top-level required fields: business_id,
irn, issue_date,
invoice_line,
legal_monetary_total.
curl -X POST https://yourdomain.com/api/validate-invoice \
-H "Content-Type: application/json" \
-d '{
"business_id": "6389237e-4f6e-4371-aece-aea5d2536df3",
"irn": "INV123-147-20251012",
"issue_date": "2025-10-12",
"due_date": "2025-10-12",
"issue_time": "12:05:22",
"invoice_type_code": "396",
"payment_status": "PENDING",
"note": "",
"tax_point_date": "2025-10-12",
"document_currency_code": "NGN",
"tax_currency_code": "NGN",
"accounting_cost": "100232 NGN",
"buyer_reference": "",
"invoice_delivery_period": {
"start_date": "2025-10-12",
"end_date": "2025-10-25"
},
"order_reference": "",
"billing_reference": null,
"dispatch_document_reference": null,
"receipt_document_reference": null,
"originator_document_reference": null,
"contract_document_reference": null,
"additional_document_reference": null,
"accounting_supplier_party": {
"party_name": "Business Name",
"tin": "32492774-0001",
"email": "developer@afrinvoice.com",
"telephone": "+2348026664444",
"business_description": "Consulting services",
"postal_address": {
"street_name": "Business Address",
"city_name": "ANTHONY",
"postal_zone": "100232",
"lga": "70",
"state": "4",
"country": "NG"
}
},
"accounting_customer_party": {
"party_name": "Customer Name",
"tin": "33577312-0001",
"email": "mark@markodenore.com",
"telephone": "+2348026665555",
"business_description": null,
"postal_address": {
"street_name": "Customer Address",
"city_name": "ANTHONY",
"postal_zone": "100232",
"lga": "70",
"state": "4",
"country": "NG"
}
},
"payee_party": null,
"bill_party": null,
"ship_party": null,
"tax_representative_party": null,
"actual_delivery_date": "2025-10-12",
"payment_means": null,
"payment_terms_note": null,
"allowance_charge": [
{ "charge_indicator": true, "amount": 800.6 },
{ "charge_indicator": false, "amount": 100.50 }
],
"tax_total": [
{
"tax_amount": 250.00,
"tax_subtotal": [
{
"taxable_amount": 1500.00,
"tax_amount": 250.00,
"tax_category": {
"id": "LOCAL_SALES_TAX",
"percent": 17.5
}
}
]
}
],
"legal_monetary_total": {
"line_extension_amount": 1500.00,
"tax_exclusive_amount": 1400.00,
"tax_inclusive_amount": 1650.00,
"payable_amount": 1650.00
},
"invoice_line": [
{
"hsn_code": "CC-001",
"product_category": "Consulting Services",
"discount_rate": 10,
"discount_amount": 50.00,
"fee_rate": 17.5,
"fee_amount": 87.50,
"invoiced_quantity": 2,
"line_extension_amount": 1000.00,
"item": {
"name": "Product A",
"description": "my wonderful product",
"sellers_item_identification": "item-1"
},
"price": {
"price_amount": 500.00,
"base_quantity": 1,
"price_unit": "NGN per 1"
}
}
]
}'
<?php
$payload = [
"business_id" => "6389237e-4f6e-4371-aece-aea5d2536df3",
"irn" => "INV123-147-20251012",
"issue_date" => "2025-10-12",
"due_date" => "2025-10-12",
"issue_time" => "12:05:22",
"invoice_type_code" => "396",
"payment_status" => "PENDING",
"note" => "",
"tax_point_date" => "2025-10-12",
"document_currency_code" => "NGN",
"tax_currency_code" => "NGN",
"accounting_cost" => "100232 NGN",
"buyer_reference" => "",
"invoice_delivery_period" => [
"start_date" => "2025-10-12",
"end_date" => "2025-10-25"
],
"order_reference" => "",
"billing_reference" => null,
"dispatch_document_reference" => null,
"receipt_document_reference" => null,
"originator_document_reference" => null,
"contract_document_reference" => null,
"additional_document_reference" => null,
"accounting_supplier_party" => [
"party_name" => "Business Name",
"tin" => "32492774-0001",
"email" => "developer@afrinvoice.com",
"telephone" => "+2348026664444",
"business_description" => "Consulting services",
"postal_address" => [
"street_name" => "Business Address",
"city_name" => "ANTHONY",
"postal_zone" => "100232",
"lga" => "70",
"state" => "4",
"country" => "NG"
]
],
"accounting_customer_party" => [
"party_name" => "Customer Name",
"tin" => "33577312-0001",
"email" => "mark@markodenore.com",
"telephone" => "+2348026665555",
"business_description" => null,
"postal_address" => [
"street_name" => "Customer Address",
"city_name" => "ANTHONY",
"postal_zone" => "100232",
"lga" => "70",
"state" => "4",
"country" => "NG"
]
],
"payee_party" => null,
"bill_party" => null,
"ship_party" => null,
"tax_representative_party" => null,
"actual_delivery_date" => "2025-10-12",
"payment_means" => null,
"payment_terms_note" => null,
"allowance_charge" => [
["charge_indicator" => true, "amount" => 800.6],
["charge_indicator" => false, "amount" => 100.50]
],
"tax_total" => [
[
"tax_amount" => 250.00,
"tax_subtotal" => [
[
"taxable_amount" => 1500.00,
"tax_amount" => 250.00,
"tax_category" => [
"id" => "LOCAL_SALES_TAX",
"percent" => 17.5
]
]
]
]
],
"legal_monetary_total" => [
"line_extension_amount" => 1500.00,
"tax_exclusive_amount" => 1400.00,
"tax_inclusive_amount" => 1650.00,
"payable_amount" => 1650.00
],
"invoice_line" => [
[
"hsn_code" => "CC-001",
"product_category" => "Consulting Services",
"discount_rate" => 10,
"discount_amount" => 50.00,
"fee_rate" => 17.5,
"fee_amount" => 87.50,
"invoiced_quantity" => 2,
"line_extension_amount" => 1000.00,
"item" => [
"name" => "Product A",
"description" => "my wonderful product",
"sellers_item_identification" => "item-1"
],
"price" => [
"price_amount" => 500.00,
"base_quantity" => 1,
"price_unit" => "NGN per 1"
]
]
]
];
$ch = curl_init("https://yourdomain.com/api/validate-invoice");
curl_setopt($ch, CURLOPT_POST, true);
curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode($payload));
curl_setopt($ch, CURLOPT_HTTPHEADER, ["Content-Type: application/json"]);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$response = curl_exec($ch);
curl_close($ch);
echo $response;
?>
import axios from "axios";
axios.post("https://yourdomain.com/api/validate-invoice", {
business_id: "6389237e-4f6e-4371-aece-aea5d2536df3",
irn: "INV123-147-20251012",
issue_date: "2025-10-12",
due_date: "2025-10-12",
issue_time: "12:05:22",
invoice_type_code: "396",
payment_status: "PENDING",
note: "",
tax_point_date: "2025-10-12",
document_currency_code: "NGN",
tax_currency_code: "NGN",
accounting_cost: "100232 NGN",
buyer_reference: "",
invoice_delivery_period: {
start_date: "2025-10-12",
end_date: "2025-10-25"
},
order_reference: "",
billing_reference: null,
dispatch_document_reference: null,
receipt_document_reference: null,
originator_document_reference: null,
contract_document_reference: null,
additional_document_reference: null,
accounting_supplier_party: {
party_name: "Business Name",
tin: "32492774-0001",
email: "developer@afrinvoice.com",
telephone: "+2348026664444",
business_description: "Consulting services",
postal_address: {
street_name: "Business Address",
city_name: "ANTHONY",
postal_zone: "100232",
lga: "70",
state: "4",
country: "NG"
}
},
accounting_customer_party: {
party_name: "Customer Name",
tin: "33577312-0001",
email: "mark@markodenore.com",
telephone: "+2348026665555",
business_description: null,
postal_address: {
street_name: "Customer Address",
city_name: "ANTHONY",
postal_zone: "100232",
lga: "70",
state: "4",
country: "NG"
}
},
payee_party: null,
bill_party: null,
ship_party: null,
tax_representative_party: null,
actual_delivery_date: "2025-10-12",
payment_means: null,
payment_terms_note: null,
allowance_charge: [
{ charge_indicator: true, amount: 800.6 },
{ charge_indicator: false, amount: 100.50 }
],
tax_total: [
{
tax_amount: 250.0,
tax_subtotal: [
{
taxable_amount: 1500.0,
tax_amount: 250.0,
tax_category: {
id: "LOCAL_SALES_TAX",
percent: 17.5
}
}
]
}
],
legal_monetary_total: {
line_extension_amount: 1500.0,
tax_exclusive_amount: 1400.0,
tax_inclusive_amount: 1650.0,
payable_amount: 1650.0
},
invoice_line: [
{
hsn_code: "CC-001",
product_category: "Consulting Services",
discount_rate: 10,
discount_amount: 50.0,
fee_rate: 17.5,
fee_amount: 87.5,
invoiced_quantity: 2,
line_extension_amount: 1000.0,
item: {
name: "Product A",
description: "my wonderful product",
sellers_item_identification: "item-1"
},
price: {
price_amount: 500.0,
base_quantity: 1,
price_unit: "NGN per 1"
}
}
]
})
.then(res => console.log(res.data))
.catch(err => console.error(err));
import requests
payload = {
"business_id": "6389237e-4f6e-4371-aece-aea5d2536df3",
"irn": "INV123-147-20251012",
"issue_date": "2025-10-12",
"due_date": "2025-10-12",
"issue_time": "12:05:22",
"invoice_type_code": "396",
"payment_status": "PENDING",
"note": "",
"tax_point_date": "2025-10-12",
"document_currency_code": "NGN",
"tax_currency_code": "NGN",
"accounting_cost": "100232 NGN",
"buyer_reference": "",
"invoice_delivery_period": {
"start_date": "2025-10-12",
"end_date": "2025-10-25"
},
"order_reference": "",
"billing_reference": None,
"dispatch_document_reference": None,
"receipt_document_reference": None,
"originator_document_reference": None,
"contract_document_reference": None,
"additional_document_reference": None,
"accounting_supplier_party": {
"party_name": "Business Name",
"tin": "32492774-0001",
"email": "developer@afrinvoice.com",
"telephone": "+2348026664444",
"business_description": "Consulting services",
"postal_address": {
"street_name": "Business Address",
"city_name": "ANTHONY",
"postal_zone": "100232",
"lga": "70",
"state": "4",
"country": "NG"
}
},
"accounting_customer_party": {
"party_name": "Customer Name",
"tin": "33577312-0001",
"email": "mark@markodenore.com",
"telephone": "+2348026665555",
"business_description": None,
"postal_address": {
"street_name": "Customer Address",
"city_name": "ANTHONY",
"postal_zone": "100232",
"lga": "70",
"state": "4",
"country": "NG"
}
},
"payee_party": None,
"bill_party": None,
"ship_party": None,
"tax_representative_party": None,
"actual_delivery_date": "2025-10-12",
"payment_means": None,
"payment_terms_note": None,
"allowance_charge": [
{ "charge_indicator": True, "amount": 800.6 },
{ "charge_indicator": False, "amount": 100.50 }
],
"tax_total": [
{
"tax_amount": 250.00,
"tax_subtotal": [
{
"taxable_amount": 1500.00,
"tax_amount": 250.00,
"tax_category": {
"id": "LOCAL_SALES_TAX",
"percent": 17.5
}
}
]
}
],
"legal_monetary_total": {
"line_extension_amount": 1500.00,
"tax_exclusive_amount": 1400.00,
"tax_inclusive_amount": 1650.00,
"payable_amount": 1650.00
},
"invoice_line": [
{
"hsn_code": "CC-001",
"product_category": "Consulting Services",
"discount_rate": 10,
"discount_amount": 50.00,
"fee_rate": 17.5,
"fee_amount": 87.50,
"invoiced_quantity": 2,
"line_extension_amount": 1000.00,
"item": {
"name": "Product A",
"description": "my wonderful product",
"sellers_item_identification": "item-1"
},
"price": {
"price_amount": 500.00,
"base_quantity": 1,
"price_unit": "NGN per 1"
}
}
]
}
response = requests.post("https://yourdomain.com/api/validate-invoice", json=payload)
print(response.json())
Generate QR Code
Encrypt IRN + server-side timestamp with FIRS public key and return QR image (base64 data URL).
POST/api/generate-qr
{
"irn": "INV001-SERVICEID-20251005"
}
{
"success": true,
"qrCodeDataUrl": "",
"encryptedBase64": "o24MzpmIqWG55j97118zzhHvRuV9pSxTsobReMpzqU2hUEeNVmQhUCH92a7yIEDRn+lLvjP5KAQ2Z1R1cZnrE/GRfmM1Jz08N0M9LARUdGQw1oWSTlKQy0FBC3zQaG8mxkdHBaO8DKexH0BISJyFEugVqP6/2nZUtSl3MxB0BO3yncOkSTo8aiLz60abNaV44kZBainwBT89ikN2GTn73VKAd34iBSpmDmHFtvj54g+bDCI+r8ivJO9nT6gPE/QdMUOz6rtCbHWgxu2rvhJfEkk10R178GQxwS6+xfcrS2wW6LmpfPqtUUQDd+oTV6R0yDkmd7iiJhhheyNUHI4uGA==",
"message": "QR generated successfully"
}
| Field | Type | Required | Description |
|---|---|---|---|
| irn | string | Yes | IRN generated by client (server appends timestamp internally) |
Server will append a unix timestamp
(..
curl -X POST "https://your-backend.example.com/api/generate-qr" \
-H "Content-Type: application/json" \
-d '{
"irn": "INV001-SERVICEID-20251005"
}'
$payload = [ "irn" => "INV001-SERVICEID-20251005" ]; $curl = curl_init(); curl_setopt_array($curl, [ CURLOPT_URL => "https://your-backend.example.com/api/generate-qr", CURLOPT_RETURNTRANSFER => true, CURLOPT_POST => true, CURLOPT_HTTPHEADER => ["Content-Type: application/json"], CURLOPT_POSTFIELDS => json_encode($payload) ]); $response = curl_exec($curl); curl_close($curl); echo $response;
import axios from "axios";
const payload = {
irn: "INV001-SERVICEID-20251005"
};
axios
.post("https://your-backend.example.com/api/generate-qr", payload)
.then((res) => console.log(res.data))
.catch((err) => console.error(err));
import requests
payload = {
"irn": "INV001-SERVICEID-20251005"
}
r = requests.post(
"https://your-backend.example.com/api/generate-qr",
json=payload
)
print(r.json())
Error Codes
The API uses conventional HTTP response codes to indicate the success or failure of a request. In general:
- Codes in the
2xxrange indicate success. -
Codes in the
4xxrange indicate client-side errors (e.g., invalid data, missing fields). -
Codes in the
5xxrange indicate server-side errors (contact support if persistent).
| Code | Status | Description |
|---|---|---|
200 |
OK | Request succeeded. Response includes requested data. |
400 |
Bad Request | Malformed JSON, missing required fields, or invalid data format. |
401 |
Unauthorized | Missing or invalid authentication credentials. |
403 |
Forbidden | Your account lacks permission to access this resource. |
404 |
Not Found | The requested endpoint or resource does not exist. |
422 |
Unprocessable Entity | Semantic errors (e.g., IRN format invalid, TIN mismatch, FIRS validation failed). |
500 |
Internal Server Error | An unexpected error occurred on our server. Retry or contact support. |
502/503 |
Service Unavailable | FIRS gateway is temporarily unreachable. Implement retry logic with exponential backoff. |
Error Response Format
{
"status": "error",
"message": "Validation failed: IRN must follow pattern {invoiceNumber}-{SERVICE_ID}-{YYYYMMDD}",
"code": 400
}
Integration Examples
Below are end-to-end examples showing how to authenticate, validate an invoice, and generate a FIRS-compliant QR code.
1. Full Invoice Workflow (cURL)
# Step 1: Login to obtain tokens
curl -X POST https://api.afrinvoice.com/api/login \
-H "Content-Type: application/json" \
-d '{"email":"integrator@example.com","password":"secure123"}'
# Response:
# { "access_token": "abc...", "access_secret": "xyz..." }
# Step 2: Validate invoice
curl -X POST https://api.afrinvoice.com/api/validate-invoice \
-H "Content-Type: application/json" \
-d '{
"business_id": "6389237e-4f6e-4371-aece-aea5d2536df3",
"irn": "INV2025-147-20251123",
"issue_date": "2025-11-23",
"invoice_line": [...],
"legal_monetary_total": {...},
"accounting_supplier_party": {...},
"accounting_customer_party": {...}
}'
# Step 3: Generate QR code
curl -X POST https://api.afrinvoice.com/api/generate-qr \
-H "Content-Type: application/json" \
-d '{"irn":"INV2025-147-20251123"}'
2. Python Integration Snippet
import requests
# Authenticate
auth = requests.post("https://api.afrinvoice.com/api/login", json={
"email": "integrator@example.com",
"password": "secure123"
}).json()
# Validate
invoice_data = { /* full invoice payload as per /validate-invoice spec */ }
validation = requests.post(
"https://api.afrinvoice.com/api/validate-invoice",
json=invoice_data
).json()
if validation.get("data", {}).get("ok"):
# Generate QR
qr = requests.post("https://api.afrinvoice.com/api/generate-qr", json={
"irn": invoice_data["irn"]
}).json()
print("QR Base64:", qr["qrCodeDataUrl"])
{invoiceNumber}-{FIRS_SERVICE_ID}-{YYYYMMDD} before
sending to /validate-invoice.