/
Copy of LLD For Invoice Generation as per Billing Cycle

What is the purpose of this article?

This article explains how Eshopbox integrates with ZOHO Billing to automate the invoicing process for unbilled shipping and fulfillment charges. It outlines the steps to generate invoices based on customer billing cycles, manage payments using available wallet balances, and update the invoice statuses. The article also covers how payments are processed (either fully or partially) and recorded in the system, ensuring accurate tracking and real-time updates in both the internal database and Zoho Billing.

Workflow Steps

1. Calculate Last Billing Date and Next Billing Date based on Shipping Billing Cycle

We will store billing information for customers created in Zoho Billing within our system by adding new columns to the existing accounts table:
The following columns will be added:

  • shipping_billing_cycle – Stores the customer's billing frequency (Weekly, Fortnightly, or Monthly).

  • last_billing_date – Initially set to the subscription_created_at date.

  • next_billing_date – Calculated based on the billing cycle.

Create a Webhook in Zoho

To ensure real-time updates of the shipping_billing_cycle for customers in our system, we will create a webhook in Zoho Billing and configure a listener API to handle incoming updates.

1. Create a Webhook in Zoho Billing

  • In Zoho Billing, navigate to Settings → Automation → Webhooks.

  • Configure a new webhook that triggers on updates to the shipping_billing_cycle field.

  • Set the webhook method to POST and provide the endpoint URL of our API.

Sample webhook Payload (POST):

1{ 2 "event": "customer_update", 3 "data": { 4 "customer_id": "982000000567001", 5 "shipping_billing_cycle": "Monthly", 6 "updated_time": "2025-02-04T10:00:00Z" 7 } 8}

2. Create a Listener API

  • Create an API to receive webhook notifications from Zoho.

API Endpoint :

1POST /api/v1/webhooks/billingCycle

Request Body (Received from Zoho Webhook)

1{ 2 "event": "customer_update", 3 "data": { 4 "customer_id": "982000000567001", 5 "shipping_billing_cycle": "Monthly", 6 "updated_time": "2025-02-04T10:00:00Z" 7 } 8}
  • Extract the relevant details from the request payload, including:

    • customer_id

    • shipping_billing_cycle

  • Update the accounts table with the new shipping_billing_cycle value and next_billing_date(Calculate it on the basis of shipping_billing_cycle ).

2. Invoice Creation

A cron job will run daily at a scheduled time, triggering the /createInvoiceCron API to automate invoice generation for customers based on their billing cycle. The process includes:

  1. Fetching accounts details for invoice creation.

  2. Calculating invoice amounts based on SKU and invoicing category.

  3. Creating invoices for Self-Fulfillment and Eshopbox Fulfillment Centers (FCs) using Zoho Billing API.

Detailed Workflow

Step 1: Scheduled Cron Job Execution

  • Trigger the API Endpoint:

    • POST /createInvoiceCron

    • Runs daily at a pre-defined schedule.

Step 2: Fetch Accounts Eligible for Invoicing

Fetch accounts where next_billing_date matches the current date (SYSDATE).
SQL Query:

1SELECT 2 customer_id, 3 account_id, 4 next_billing_date 5FROM 6 accounts 7WHERE 8 next_billing_date = SYSDATE;

Result: List of customers (customer_id) and their corresponding accounts (account_id) due for invoicing.

Step 3: Task Queue Creation for Invoice Processing

For each eligible account_id, create two Task Queues:

  • Self-Fulfillment Invoice Task Queue (Single shipping invoice per account)

  • Eshopbox Location Invoice Task Queue (2 invoices(Shipping and Fullfillment) per FC for that account)

Now inside each taskqueue we will calculate the invoice amount.

3.1 Self-Fulfillment Invoice Calculation
  • Calculate total invoice amount for all Zoho SKUs in the 'Shipping' category for self-Fullfillment only.

SQL query::

1SELECT 2 td.zoho_zoho_sku AS zoho_sku, 3 SUM(wt.amount) AS total_invoice_amount 4FROM 5 wallet_transaction wt 6JOIN 7 workspace_wallet ww ON wt.workspace_wallet_id = ww.id 8JOIN 9 transaction_definitions td ON wt.transaction_definition_id = td.id 10JOIN 11 warehouse wh ON wt.warehouse_id = wh.id 12JOIN 13 facility f ON wh.facility_id = f.id 14WHERE 15 td.invoicingCategory = 'shipping' 16 AND ww.account_id = {account_id} 17 AND wt.invoicingStatus = 'pending' 18 AND f.external_wms_account_id <> 1 19GROUP BY 20 td.zoho__sku;

Result: It will fetch the zohoSku and respective invoiced amount for the account_id.

3.2 Eshopbox Location Invoice Calculation
A. Shipping Invoice Calculation for Each Warehouse(Eshopbox Fcs only)

SQL Query

1SELECT 2 td.zoho_sku AS zoho_sku, 3 wt.warehouse_id AS warehouse_id, 4 SUM(wt.amount) AS total_invoice_amount 5FROM 6 wallet_transaction wt 7JOIN 8 workspace_wallet ww ON wt.workspace_wallet_id = ww.id 9JOIN 10 transaction_definitions td ON wt.transaction_definition_id = td.id 11JOIN 12 warehouse wh ON wt.warehouse_id = wh.id 13JOIN 14 facility f ON wh.facility_id = f.id 15WHERE 16 td.invoicingCategory = 'shipping' 17 AND ww.account_id = {account_id} 18 AND wt.invoicingStatus = 'pending' 19 AND f.external_wms_account_id = 1 20GROUP BY 21 td.zoho_sku, wh.id;

Result: It will fetch invoiced amount for each warehouse and sku for shipping category.

B. Fulfillment Invoice Calculation for Each Warehouse
  • Include only 'Fulfillment' category transactions for Eshopbox FCs

SQL Query

1SELECT 2 td.zoho_sku AS zoho_sku, 3 wt.warehouse_id AS warehouse_id, 4 SUM(wt.amount) AS total_invoice_amount 5FROM 6 wallet_transaction wt 7JOIN 8 workspace_wallet ww ON wt.workspace_wallet_id = ww.id 9JOIN 10 transaction_definitions td ON wt.transaction_definition_id = td.id 11JOIN 12 warehouse wh ON wt.warehouse_id = wh.id 13JOIN 14 facility f ON wh.facility_id = f.id 15WHERE 16 td.invoicingCategory = 'fulfillment' 17 AND ww.account_id = {account_id} 18 AND wt.invoicingStatus = 'pending' 19 AND f.external_wms_account_id = 1 20GROUP BY 21 td.zoho_sku, wh.id;

Result: It will fetch invoiced amount for each warehouse and sku for Fullfillment category.

Now we will call the Zoho Invoice creation API based on above(step 3) invoice calculation.

Step 4: Make a call to Zoho API for Invoice creation:

1POST https://www.zohoapis.com/billing/v1/invoices

Sample Request Body:
Note: The request body should include all the zoho_sku in line_items for a particular invoicing categories.

1{ 2 "customer_id": "903000000000099", 3 "place_of_supply": "TN", 4 "gst_no": "22AAAAA0000A1Z5", 5 "payment_terms": 15, 6 "date": "2025-01-01" 7 "custom_fields": [ 8 { 9 "label": "Accrual month", 10 "value": "Feb" //Current Month 11 }, 12 { 13 "label": "Accrual Year", 14 "value": "2025" //Current Year 15 } 16 ], 17 "invoice_items": [ 18 { 19 "item_id": 982000000567021, //ZOHO SKU 20 "name": "Shipping Charges", 21 "description": "Shipping costs for the billing period", 22 "rate": 100, 23 "quantity": 5 24 }, 25 { 26 "item_id": 982000000567043, //ZOHO SKU 27 "name": "Shipping Charges", 28 "description": "Shipping costs for the billing period", 29 "rate": 50, 30 "quantity": 10 31 } 32 ], 33}

Sample Response Body:

1{ 2code": 0, 3 "message": "The invoice has been created.", 4 "invoice": { 5 "invoice_id": "4947764000002227106", 6 "number": "IN/HAR/2425/626", 7 "invoice_number": "IN/HAR/2425/626", 8 "invoice_date": "2025-02-03", 9 "date": "2025-02-03", 10 "due_date": "2025-02-18", 11 "reference_number": "", 12 "status": "draft", 13 "place_of_supply": "UP", 14 "customer_id": "4947764000002177048", 15 "customer_name": "Harshva05" 16 } 17}

Step 5: Save invoices details in Invoices table

Extract relevant fields (e.g., invoice_id, invoice_number, status) from the API response and insert them into the Invoices table.

Step 6: Record payment in Invoices_payment table

This step involves recording the payment in the invoices_payment table based on the available unapplied_amount.

Retrieve unapplied_amount from wallet_transaction table using account_id, considering the max wallet_transaction_id for fetching unapplied amount.

SQL Query

1SELECT unapplied_amount 2FROM wallet_transaction 3WHERE account_id = :account_id 4ORDER BY wallet_transaction_id DESC 5LIMIT 1;
  • Case 1: Unapplied Amount ≥ Invoiced Amount

    • Pay the full invoiced amount from unapplied_amount.

    • Insert a record into the invoices_payment table with:

      • invoice_id

      • wallet_transaction_id (from the latest wallet transaction)

      • amount_paid (full invoiced amount)

  • Case 2: Unapplied Amount < Invoiced Amount (Partial Payment)

    • Pay the entire unapplied_amount from the balance.

    • Insert a record into the invoices_payment table with:

      • invoice_id

      • wallet_transaction_id

      • amount_paid (equal to unapplied_amount)

  • Case 3: Unapplied Amount = 0 (No Payment Available)

    • No payment entry is made in the invoices_payment table.

Step 7: Update status in Invoices table

After recording the payment in the invoices_payment table, update the invoice_status in the Invoices table based on the amount paid.

Step 8: Update unapplied_amount in wallet_transaction table

Update the unapplied amount in wallet_transaction table as per the amount_paid in step-5

Step 9: Update the invoice status in zoho

Finally, we will update the invoice status in zoho. This is done by calling a Create Payment API, based on the amount we recorded status will be updated in zoho. For e.g If full amount is recorded then status will change to ‘paid', in case of partial amount recorded it will change to 'partiallly_paid’.

ZOHO API Endpoint:

1POST https://www.zohoapis.com/billing/v1/payments

Sample Request Body:

1{ 2 "customer_id": "903000000000099", 3 "payment_mode": "cash", 4 "amount": 450, 5 "date": "2016-06-05", 6 "reference_number": "INV-384", 7 "description": "Payment has been added to INV-384", 8 "invoices": [ 9 { 10 "invoice_id": "90300000079426", 11 "amount_applied": 450 12 } 13 ] 14}

3. Automated Invoice Payment on Wallet Recharge

To automate invoice payments when a brand's wallet is recharged:

  • Implement a Pub/Sub Mechanism:

    • A Pub/Sub topic will be created, where a message is published as soon as the wallet is recharged.

    • A subscriber API will listen to this topic and process the payment updates.

Subscriber API Workflow:

  • Retrieve the latest unapplied_amount from the wallet_transaction table by considering the most recent wallet_transaction_id.

  • Fetch the list of unpaid or partially paid invoices from the invoice table.

  • Process the invoices one by one, prioritizing the oldest invoice first.

  • For each invoice:

  • Case 1: Unpaid Invoice

    • If invoiced_amount < unapplied_amount, pay the full invoice amount, update its status to paid in invoice table, and record the transaction in invoices_payment, also update the unapplied amount and process next invoice.

    • If invoiced_amount > unapplied_amount, pay the available amount, mark the invoice as partially paid in invoice table, record the transaction in invoices_payment, and update unapplied_amount in wallet_transaction table and update the invoice status in zoho.

  • Case 2: Partially Paid Invoice

    • Fetch the previously paid amount from invoices_payment.

    • Calculate the remaining amount to be paid (invoiced_amount - paid_amount).

    • Compare it with unapplied_amount and pay accordingly.

    • Update invoices_payment, invoices, and wallet_transaction tables.

    • Finally, update the status in Zoho using the following API:

Zoho API:

1POST https://www.zohoapis.com/billing/v1/payments