API Documentation

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
Base URL: https://api.e-afrinvoice.com

Authentication

Authenticate using email & password to receive access_token and access_secret.

LIVE POST /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
email 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).

LOCAL Client: 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 (..) before encrypting, per FIRS spec.

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 2xx range indicate success.
  • Codes in the 4xx range indicate client-side errors (e.g., invalid data, missing fields).
  • Codes in the 5xx range 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"])
💡 Tip: Always generate the IRN client-side using the pattern {invoiceNumber}-{FIRS_SERVICE_ID}-{YYYYMMDD} before sending to /validate-invoice.