The Problem
Within my business, Coast Tutoring, the use of accounting software such as MYOB has been an integral part of managing invoicing and payroll across all aspects of operations. For over 18 months, I benefited from this service at no cost, which significantly streamlined administrative tasks without impacting my bottom line. However, recent price increases have transformed this once-free service into a notable expense, now exceeding $60 per month. As someone who strives for efficiency and sustainability in business decisions, I find myself grappling with the value of this investment. Given that I only utilise a fraction of the software's capabilities, this recurring expense feels increasingly unjustifiable. It has become clear that a change is necessary—whether by seeking a more cost-effective alternative, renegotiating terms, or re-evaluating my reliance on such external tools altogether.
Idea
The concept arose from the need to tackle inefficiencies and mounting costs in my current invoicing system, while retaining the simplicity for both me, the user, and customers. I began by examining the fundamental requirements of the process: retrieving customer data, generating invoices, and maintaining accurate records.
Realising that much of this could be streamlined, I considered the resources already at my disposal, such as Python for automation and Excel for data storage. This prompted a structured brainstorming process, during which I outlined how each stage—filtering data by invoice number, generating PDF invoices, and organising outputs—could be automated.
I also identified potential edge cases in invoicing, such as incorporating due date tracking and accommodating multi-line invoices.
By focusing on these essentials, I conceptualised a tailored solution that would eliminate unnecessary features while meeting the specific operational needs of my business, allowing me to create an MVP.
Solution
The resulting Python code shown below loads customer data from a local Excel file, also used as a record of invoicing, asks user for invoice number before creating a PDF invoice and saving to the specified local folder.
import pandas as pd
from fpdf import FPDF
import datetime
# datetime object containing current date and time
now = datetime.datetime.now()
dt_now = now.strftime("%d/%m/%Y")
dt_due_long = now + datetime.timedelta(days=14)
dt_due = dt_due_long.strftime("%d/%m/%Y")
print(dt_due)
def import_customer_data(file_path, invoice_number):
"""Imports customer data from an Excel file filtered by invoice number."""
try:
data = pd.read_excel(file_path)
# Debug: Print data and types
print("Loaded data:")
print(data.head())
print("Invoice Number column type:", data['Invoice Number'].dtype)
# Ensure invoice number comparison works correctly
if data['Invoice Number'].dtype != type(invoice_number):
invoice_number = str(invoice_number) if data['Invoice Number'].dtype == object else int(invoice_number)
filtered_data = data[data['Invoice Number'] == invoice_number]
if filtered_data.empty:
print(f"No data found for Invoice Number: {invoice_number}")
return None
return filtered_data
except Exception as e:
print(f"Error importing data: {e}")
return None
def generate_invoice(customer_data, output_path):
"""Generates a PDF invoice for the provided customer data."""
try:
pdf = FPDF()
pdf.add_page()
pdf.set_font("Arial", size=12)
# Add Logo
pdf.image(r"D:\Users\conno\OneDrive\Desktop\Coast Tutoring\Website\Coast_Tutoring_Logo.png", x=10, y=8, w=30)
# Add header
pdf.ln(5)
pdf.set_font("Arial", style='B', size=14)
pdf.cell(0, 10, txt="Coast Tutoring Invoice", ln=True, align="C")
pdf.ln(25)
# Add invoice number and date
invoice_number = customer_data['Invoice Number'].iloc[0]
pdf.set_font("Arial", style= 'B', size=11)
pdf.cell(0, 5, txt=f"Invoice Number: {invoice_number}", ln=True, align="L")
pdf.set_font("Arial", size=11)
pdf.cell(0, 10, txt=f"Issue Date: {dt_now}", ln=False, align="L")
pdf.cell(0, 10, txt=f"Due Date: {dt_due}", ln=True, align="R")
# Add customer details
customer_name = customer_data['Customer Name'].iloc[0]
#customer_address = customer_data['Customer Address'].iloc[0]
pdf.cell(0, 5, txt=f"Name: {customer_name}", ln=True, align="L")
#pdf.cell(0, 5, txt=f"Address: {customer_address}", ln=True, align="L")
# Add line break
pdf.ln(10)
# Add table header
pdf.set_font("Arial", style='B', size=11)
pdf.cell(100, 10, txt="Description", border=1, align='C')
pdf.cell(30, 10, txt="Quantity", border=1, align='C')
pdf.cell(30, 10, txt="Unit Cost", border=1, align='C')
pdf.cell(30, 10, txt="Subtotal", border=1, align='C')
pdf.ln(10)
# Add table rows for all items in the invoice
pdf.set_font("Arial", size=11)
total = 0
for _, row in customer_data.iterrows():
description = row['Description']
quantity = row['Quantity']
unit_cost = row['Unit Cost']
subtotal = row['Total']
total += subtotal
pdf.cell(100, 10, txt=f"{description}", border=1, align='L')
pdf.cell(30, 10, txt=f"{quantity}", border=1, align='C')
pdf.cell(30, 10, txt=f"${unit_cost:.2f}", border=1, align='C')
pdf.cell(30, 10, txt=f"${subtotal:.2f}", border=1, align='R')
pdf.ln(10)
# Add total
pdf.set_font("Arial", style='B', size=11)
pdf.cell(160, 10, txt="Total", border=1, align='R')
pdf.cell(30, 10, txt=f"${total:.2f}", border=1, align='R')
pdf.ln(10)
# Payment instructions
pdf.ln(10)
pdf.set_font("Arial", style='B', size=11)
pdf.cell(0, 10, txt=f"Due Date: {dt_due}", ln=True, align="L")
pdf.ln(5)
pdf.set_font("Arial", style='B', size=12)
pdf.cell(0, 10, txt="Payment Instructions", ln=True, align="L")
pdf.cell(0, 10, txt="Bank deposit via EFT", ln=True, align="L")
pdf.set_font("Arial", size=11)
pdf.cell(0, 6, txt="Bank: NAB", ln=True, align="L")
pdf.cell(0, 6, txt="Name: COAST TUTORING", ln=True, align="L")
pdf.cell(0, 6, txt="BSB: 082356", ln=True, align="L")
pdf.cell(0, 6, txt="AC#: 355918718", ln=True, align="L")
pdf.cell(0, 6, txt=f"Ref#: {invoice_number} ", ln=True, align="L")
# Customer Help
pdf.ln(20)
pdf.set_font("Arial", style='B', size=12)
pdf.cell(0, 10, txt="Need Help?", ln=True, align="L")
pdf.set_font("Arial", size=11)
pdf.cell(0, 6, txt="If you have any questions about this invoice, please contact: connor@coasttutoring.net or 0481 679 212", ln=True, align="L")
# Add footer
pdf.set_y(-40)
pdf.set_font("Arial", style='I', size=8)
pdf.cell(0, 10, txt=f"Thank you for your business {customer_name}!", align='C')
pdf.ln(5)
pdf.cell(0, 10, txt=f"Issue Date: {dt_now}", ln=False, align="L")
pdf.cell(-195, 10, txt="Coast Tutoring", align='C')
pdf.cell(0, 10, txt="ABN: 123456789", align='R')
pdf.ln(-5)
# Save the invoice as a PDF
customer_file = f"{output_path}/Coast_Tutoring_Invoice_{invoice_number}_{customer_name.replace(' ', '_')}.pdf"
pdf.output(customer_file)
print(f"Invoice generated for {customer_name} at {customer_file}")
except Exception as e:
print(f"Error generating invoice: {e}")
if __name__ == "__main__":
# Specify the Excel file path and output directory
#excel_file_path = r"C:\Users\conno\OneDrive\Desktop\Coast Tutoring\Clients\Invoices\InvoiceData\InvoiceDataFile.xlsx"
excel_file_path = r"D:\Users\conno\OneDrive\Desktop\Coast Tutoring\Clients\Invoices\InvoiceData\InvoiceDataFile.xlsx"
#output_directory = r"C:\Users\conno\OneDrive\Desktop\Coast Tutoring\Clients\Invoices\InvoiceData"
output_directory = r"D:\Users\conno\OneDrive\Desktop\Coast Tutoring\Clients\Invoices\2025"
# Prompt user for invoice number
invoice_number = input("Enter the Invoice Number: ")
# Import data
customer_data = import_customer_data(excel_file_path, invoice_number)
if customer_data is not None:
# Generate invoices
generate_invoice(customer_data, output_directory)
Below is the PDF export for an example of a single line invoice and a multi-line invoice.
Single Line Invoice
Multiple Line Invoice
For the Future
The future development of this program envisions the creation of a web application, designed to function similarly to the MYOB platform previously utilised. This advancement would enable the majority of the Python code already written to remain a core component of the backend infrastructure, ensuring continuity and leveraging the work already completed.
Key changes required for this transition would include the development of a dynamic table within the web app for inputting and managing data, replacing the reliance on local Excel files. Additionally, features such as invoice due date tracking and the ability to download invoices directly through the web interface—rather than saving them to a local folder—would need to be implemented. These upgrades aim to enhance the program’s accessibility, functionality, and efficiency, ultimately offering a more streamlined and user-friendly experience for both myself and any potential collaborators.